diff --git a/TODO b/TODO index 5a03535334022bcab456696fd53a79aead272047..8f5061df7dbab615143522208a3f501cec7793e3 100644 --- a/TODO +++ b/TODO @@ -13,8 +13,6 @@ Version 1.5.0: * Ethernet drivers: - Fix link detection in generic driver. - Add native drivers from 2.6.24 up to 2.6.31. -* Finish library implementation. - - Remove stdio uses? * Rescan command. * Change SDO index at runtime for SDO request. * Output skipped datagrams again. @@ -86,6 +84,7 @@ Future issues: * Simplify master fsm by introducing a common request state to handle external requests (replace write_sii, sdo_request, etc). * Write PDO mapping/assignment by default? +* Remove stdio uses in userspace library? Smaller issues: diff --git a/lib/common.c b/lib/common.c index f9f6a77102ed66dc96abb992c4c19c40bf9c6066..35702376eacd388895ccac2c99c6cf3de300bc4f 100644 --- a/lib/common.c +++ b/lib/common.c @@ -55,9 +55,9 @@ ec_master_t *ecrt_request_master(unsigned int master_index) ec_master_t *master = ecrt_open_master(master_index); if (master) { if (ecrt_master_reserve(master) < 0) { - close(master->fd); + ec_master_clear(master); free(master); - master = 0; + master = NULL; } } @@ -82,19 +82,21 @@ ec_master_t *ecrt_open_master(unsigned int master_index) master->process_data = NULL; master->process_data_size = 0; + master->first_domain = NULL; + master->first_config = NULL; snprintf(path, MAX_PATH_LEN - 1, "/dev/EtherCAT%u", master_index); master->fd = open(path, O_RDWR); if (master->fd == -1) { fprintf(stderr, "Failed to open %s: %s\n", path, strerror(errno)); - goto out_free; + goto out_clear; } if (ioctl(master->fd, EC_IOCTL_MODULE, &module_data) < 0) { fprintf(stderr, "Failed to get module information from %s: %s\n", path, strerror(errno)); - goto out_close; + goto out_clear; } if (module_data.ioctl_version_magic != EC_IOCTL_VERSION_MAGIC) { @@ -102,14 +104,13 @@ ec_master_t *ecrt_open_master(unsigned int master_index) " %s: %u, libethercat: %u.\n", path, module_data.ioctl_version_magic, EC_IOCTL_VERSION_MAGIC); - goto out_close; + goto out_clear; } return master; -out_close: - close(master->fd); -out_free: +out_clear: + ec_master_clear(master); free(master); return 0; } @@ -118,11 +119,7 @@ out_free: void ecrt_release_master(ec_master_t *master) { - if (master->process_data) { - munmap(master->process_data, master->process_data_size); - } - - close(master->fd); + ec_master_clear(master); free(master); } diff --git a/lib/domain.c b/lib/domain.c index 8b36cdc3da62a4854bd7a46f575b0dedfff584b5..98ae332c94b16640f6dda4734fac35c447557d3d 100644 --- a/lib/domain.c +++ b/lib/domain.c @@ -47,6 +47,13 @@ /*****************************************************************************/ +void ec_domain_clear(ec_domain_t *domain) +{ + // nothing to do +} + +/*****************************************************************************/ + int ecrt_domain_reg_pdo_entry_list(ec_domain_t *domain, const ec_pdo_entry_reg_t *regs) { diff --git a/lib/domain.h b/lib/domain.h index 10ca2947b065c9a7a074f71c24e73712a381a2a8..3d9690fa3ba4f606e4403e511a6a842b45d904e9 100644 --- a/lib/domain.h +++ b/lib/domain.h @@ -33,9 +33,14 @@ /*****************************************************************************/ struct ec_domain { + ec_domain_t *next; unsigned int index; ec_master_t *master; uint8_t *process_data; }; /*****************************************************************************/ + +void ec_domain_clear(ec_domain_t *); + +/*****************************************************************************/ diff --git a/lib/master.c b/lib/master.c index 5b4f05807760a82922baef8caee098d622b3bdde..193d7e576847489c0f97a1078f6bac440ef0f763 100644 --- a/lib/master.c +++ b/lib/master.c @@ -53,6 +53,51 @@ int ecrt_master_reserve(ec_master_t *master) /*****************************************************************************/ +void ec_master_clear(ec_master_t *master) +{ + ec_domain_t *d, *next_d; + ec_slave_config_t *c, *next_c; + + if (master->process_data) { + munmap(master->process_data, master->process_data_size); + } + + d = master->first_domain; + while (d) { + next_d = d->next; + ec_domain_clear(d); + d = next_d; + } + + c = master->first_config; + while (c) { + next_c = c->next; + ec_slave_config_clear(c); + c = next_c; + } + + if (master->fd != -1) { + close(master->fd); + } +} + +/*****************************************************************************/ + +void ec_master_add_domain(ec_master_t *master, ec_domain_t *domain) +{ + if (master->first_domain) { + ec_domain_t *d = master->first_domain; + while (d->next) { + d = d->next; + } + d->next = domain; + } else { + master->first_domain = domain; + } +} + +/*****************************************************************************/ + ec_domain_t *ecrt_master_create_domain(ec_master_t *master) { ec_domain_t *domain; @@ -71,14 +116,33 @@ ec_domain_t *ecrt_master_create_domain(ec_master_t *master) return 0; } + domain->next = NULL; domain->index = (unsigned int) index; domain->master = master; domain->process_data = NULL; + + ec_master_add_domain(master, domain); + return domain; } /*****************************************************************************/ +void ec_master_add_slave_config(ec_master_t *master, ec_slave_config_t *sc) +{ + if (master->first_config) { + ec_slave_config_t *c = master->first_config; + while (c->next) { + c = c->next; + } + c->next = sc; + } else { + master->first_config = sc; + } +} + +/*****************************************************************************/ + ec_slave_config_t *ecrt_master_slave_config(ec_master_t *master, uint16_t alias, uint16_t position, uint32_t vendor_id, uint32_t product_code) @@ -105,10 +169,16 @@ ec_slave_config_t *ecrt_master_slave_config(ec_master_t *master, return 0; } + sc->next = NULL; sc->master = master; sc->index = data.config_index; sc->alias = alias; sc->position = position; + sc->first_sdo_request = NULL; + sc->first_voe_handler = NULL; + + ec_master_add_slave_config(master, sc); + return sc; } diff --git a/lib/master.h b/lib/master.h index 46dfbe2d97cd0e510799f7f721aac592def9af29..d092c39d77a19219f08c3414b08b397b22abd665 100644 --- a/lib/master.h +++ b/lib/master.h @@ -36,6 +36,13 @@ struct ec_master { int fd; uint8_t *process_data; size_t process_data_size; + + ec_domain_t *first_domain; + ec_slave_config_t *first_config; }; /*****************************************************************************/ + +void ec_master_clear(ec_master_t *); + +/*****************************************************************************/ diff --git a/lib/sdo_request.c b/lib/sdo_request.c index 0b53d80d6b06831ccb032824a2b860ff1d1f765a..8999801ce1c98f7ac4f63a9617b234c7428ae85a 100644 --- a/lib/sdo_request.c +++ b/lib/sdo_request.c @@ -43,8 +43,17 @@ #include "slave_config.h" #include "master.h" +/*****************************************************************************/ + +void ec_sdo_request_clear(ec_sdo_request_t *req) +{ + if (req->data) { + free(req->data); + } +} + /***************************************************************************** - * Realtime interface. + * Application interface. ****************************************************************************/ void ecrt_sdo_request_timeout(ec_sdo_request_t *req, uint32_t timeout) diff --git a/lib/sdo_request.h b/lib/sdo_request.h index 3719a44a60462cc6b2d597ccffdf4bca3686e272..1e55fee21deefc6d3825e5308e312aba0b441cef 100644 --- a/lib/sdo_request.h +++ b/lib/sdo_request.h @@ -33,6 +33,7 @@ /*****************************************************************************/ struct ec_sdo_request { + ec_sdo_request_t *next; /**< List header. */ ec_slave_config_t *config; /**< Parent slave configuration. */ unsigned int index; /**< Request index (identifier). */ uint16_t sdo_index; /**< SDO index. */ @@ -43,3 +44,7 @@ struct ec_sdo_request { }; /*****************************************************************************/ + +void ec_sdo_request_clear(ec_sdo_request_t *); + +/*****************************************************************************/ diff --git a/lib/slave_config.c b/lib/slave_config.c index 49ce635b4fcb85bc0bb17e6985878bad3c2310d3..468452106096522aae4903811186c1badd97cee8 100644 --- a/lib/slave_config.c +++ b/lib/slave_config.c @@ -43,6 +43,29 @@ /*****************************************************************************/ +void ec_slave_config_clear(ec_slave_config_t *sc) +{ + ec_sdo_request_t *r, *next_r; + ec_voe_handler_t *v, *next_v; + + r = sc->first_sdo_request; + while (r) { + next_r = r->next; + ec_sdo_request_clear(r); + r = next_r; + } + + + v = sc->first_voe_handler; + while (v) { + next_v = v->next; + ec_voe_handler_clear(v); + v = next_v; + } +} + +/*****************************************************************************/ + int ecrt_slave_config_sync_manager(ec_slave_config_t *sc, uint8_t sync_index, ec_direction_t dir, ec_watchdog_mode_t watchdog_mode) { @@ -361,6 +384,22 @@ int ecrt_slave_config_sdo32(ec_slave_config_t *sc, uint16_t index, /*****************************************************************************/ +void ec_slave_config_add_sdo_request(ec_slave_config_t *sc, + ec_sdo_request_t *req) +{ + if (sc->first_sdo_request) { + ec_sdo_request_t *r = sc->first_sdo_request; + while (r->next) { + r = r->next; + } + r->next = req; + } else { + sc->first_sdo_request = req; + } +} + +/*****************************************************************************/ + ec_sdo_request_t *ecrt_slave_config_create_sdo_request(ec_slave_config_t *sc, uint16_t index, uint8_t subindex, size_t size) { @@ -393,23 +432,42 @@ ec_sdo_request_t *ecrt_slave_config_create_sdo_request(ec_slave_config_t *sc, if (ioctl(sc->master->fd, EC_IOCTL_SC_SDO_REQUEST, &data) == -1) { fprintf(stderr, "Failed to create SDO request: %s\n", strerror(errno)); - if (req->data) - free(req->data); + ec_sdo_request_clear(req); free(req); return NULL; } + req->next = NULL; req->config = sc; req->index = data.request_index; req->sdo_index = data.sdo_index; req->sdo_subindex = data.sdo_subindex; req->data_size = size; req->mem_size = size; + + ec_slave_config_add_sdo_request(sc, req); + return req; } /*****************************************************************************/ +void ec_slave_config_add_voe_handler(ec_slave_config_t *sc, + ec_voe_handler_t *voe) +{ + if (sc->first_voe_handler) { + ec_voe_handler_t *v = sc->first_voe_handler; + while (v->next) { + v = v->next; + } + v->next = voe; + } else { + sc->first_voe_handler = voe; + } +} + +/*****************************************************************************/ + ec_voe_handler_t *ecrt_slave_config_create_voe_handler(ec_slave_config_t *sc, size_t size) { @@ -441,16 +499,19 @@ ec_voe_handler_t *ecrt_slave_config_create_voe_handler(ec_slave_config_t *sc, if (ioctl(sc->master->fd, EC_IOCTL_SC_VOE, &data) == -1) { fprintf(stderr, "Failed to create VoE handler: %s\n", strerror(errno)); - if (voe->data) - free(voe->data); + ec_voe_handler_clear(voe); free(voe); return NULL; } + voe->next = NULL; voe->config = sc; voe->index = data.voe_index; voe->data_size = size; voe->mem_size = size; + + ec_slave_config_add_voe_handler(sc, voe); + return voe; } diff --git a/lib/slave_config.h b/lib/slave_config.h index 53729457448ffa9a55ad998efd2d4f55e20e4ff0..a3e269558ea7433cbee45d6c84ae7b33ba2ade77 100644 --- a/lib/slave_config.h +++ b/lib/slave_config.h @@ -33,10 +33,17 @@ /*****************************************************************************/ struct ec_slave_config { + ec_slave_config_t *next; ec_master_t *master; unsigned int index; uint16_t alias; uint16_t position; + ec_sdo_request_t *first_sdo_request; + ec_voe_handler_t *first_voe_handler; }; /*****************************************************************************/ + +void ec_slave_config_clear(ec_slave_config_t *); + +/*****************************************************************************/ diff --git a/lib/voe_handler.c b/lib/voe_handler.c index e1645a4f12222f2706ef9b96e0e801c007128b69..ff87109a1072f6726d185f9c0c29faf6d3d82c2f 100644 --- a/lib/voe_handler.c +++ b/lib/voe_handler.c @@ -47,6 +47,14 @@ /*****************************************************************************/ +void ec_voe_handler_clear(ec_voe_handler_t *voe) +{ + if (voe->data) + free(voe->data); +} + +/*****************************************************************************/ + void ecrt_voe_handler_send_header(ec_voe_handler_t *voe, uint32_t vendor_id, uint16_t vendor_type) { diff --git a/lib/voe_handler.h b/lib/voe_handler.h index df9542dfc04a550d723318140c0870256048e856..523c138036e12fb7b699df464f46791cc6990759 100644 --- a/lib/voe_handler.h +++ b/lib/voe_handler.h @@ -33,6 +33,7 @@ /*****************************************************************************/ struct ec_voe_handler { + ec_voe_handler_t *next; ec_slave_config_t *config; unsigned int index; size_t data_size; @@ -41,3 +42,7 @@ struct ec_voe_handler { }; /*****************************************************************************/ + +void ec_voe_handler_clear(ec_voe_handler_t *); + +/*****************************************************************************/