# HG changeset patch # User Florian Pose # Date 1137166778 0 # Node ID c0405659a74a86ea7d769f57e7b5b2b9e5eb6d0c # Parent b3beaa00640f1ef6a73eab294eee0a3fe3c62ce7 Neues Interface. diff -r b3beaa00640f -r c0405659a74a drivers/ec_domain.c --- a/drivers/ec_domain.c Fri Jan 13 13:44:22 2006 +0000 +++ b/drivers/ec_domain.c Fri Jan 13 15:39:38 2006 +0000 @@ -24,28 +24,11 @@ void EtherCAT_domain_init(EtherCAT_domain_t *dom) { dom->number = 0; - dom->data = NULL; dom->data_size = 0; dom->logical_offset = 0; dom->response_count = 0; -} -/*****************************************************************************/ - -/** - Destruktor eines Prozessdatenobjekts. - - @param dom Zeiger auf die zu löschenden Prozessdaten -*/ - -void EtherCAT_domain_clear(EtherCAT_domain_t *dom) -{ - if (dom->data) { - kfree(dom->data); - dom->data = NULL; - } - - dom->data_size = 0; + memset(dom->data, 0x00, ECAT_FRAME_BUFFER_SIZE); } /*****************************************************************************/ diff -r b3beaa00640f -r c0405659a74a drivers/ec_domain.h --- a/drivers/ec_domain.h Fri Jan 13 13:44:22 2006 +0000 +++ b/drivers/ec_domain.h Fri Jan 13 15:39:38 2006 +0000 @@ -29,8 +29,8 @@ unsigned int number; /*<< Domänen-Identifikation */ EtherCAT_command_t command; /**< Kommando zum Senden und Empfangen der Prozessdaten */ - unsigned char *data; /**< Zeiger auf Speicher mit Prozessdaten */ - unsigned int data_size; /**< Größe des Prozessdatenspeichers */ + unsigned char data[ECAT_FRAME_BUFFER_SIZE]; /**< Prozessdaten-Array */ + unsigned int data_size; /**< Größe der Prozessdaten */ unsigned int logical_offset; /**< Logische Basisaddresse */ unsigned int response_count; /**< Anzahl antwortender Slaves */ } @@ -39,7 +39,6 @@ /*****************************************************************************/ void EtherCAT_domain_init(EtherCAT_domain_t *); -void EtherCAT_domain_clear(EtherCAT_domain_t *); /*****************************************************************************/ diff -r b3beaa00640f -r c0405659a74a drivers/ec_master.c --- a/drivers/ec_master.c Fri Jan 13 13:44:22 2006 +0000 +++ b/drivers/ec_master.c Fri Jan 13 15:39:38 2006 +0000 @@ -27,6 +27,8 @@ void EtherCAT_master_init(EtherCAT_master_t *master) { + master->bus_slaves = NULL; + master->bus_slaves_count = 0; master->dev = NULL; master->command_index = 0x00; master->tx_data_length = 0; @@ -51,11 +53,9 @@ void EtherCAT_master_clear(EtherCAT_master_t *master) { - unsigned int i; - - // Remove domains - for (i = 0; i < master->domain_count; i++) { - EtherCAT_domain_clear(master->domains + i); + if (master->bus_slaves) { + kfree(master->bus_slaves); + master->bus_slaves = NULL; } master->domain_count = 0; @@ -353,42 +353,19 @@ /*****************************************************************************/ /** - Überprüft die angeschlossenen Slaves. - - Vergleicht die an den Bus angeschlossenen Slaves mit - den im statischen-Slave-Array vorgegebenen Konfigurationen. - Stimmen Anzahl oder Typen nicht überein, gibt diese - Methode einen Fehler aus. + Durchsucht den Bus nach Slaves. @param master Der EtherCAT-Master - @param slaves Zeiger auf ein statisches Slave-Array - @param slave_count Anzahl der Slaves im Array @return 0 bei Erfolg, sonst < 0 */ -int EtherCAT_check_slaves(EtherCAT_master_t *master, - EtherCAT_slave_t *slaves, - unsigned int slave_count) +int EtherCAT_scan_for_slaves(EtherCAT_master_t *master) { EtherCAT_command_t cmd; EtherCAT_slave_t *cur; - unsigned int i, j, found, size, offset; + unsigned int i, j; unsigned char data[2]; - EtherCAT_domain_t *dom; - - // Clear domains - for (i = 0; i < master->domain_count; i++) { - printk(KERN_DEBUG "EtherCAT: Clearing domain %i!\n", - master->domains[i].number); - EtherCAT_domain_clear(master->domains + i); - } - master->domain_count = 0; - - if (unlikely(!slave_count)) { - printk(KERN_ERR "EtherCAT: No slaves in list!\n"); - return -1; - } // Determine number of slaves on bus @@ -397,44 +374,24 @@ if (unlikely(EtherCAT_simple_send_receive(master, &cmd) < 0)) return -1; - if (unlikely(cmd.working_counter != slave_count)) { - printk(KERN_ERR "EtherCAT: Wrong number of slaves on bus: %i / %i\n", - cmd.working_counter, slave_count); - return -1; - } - - printk("EtherCAT: Found all %i slaves.\n", slave_count); + master->bus_slaves_count = cmd.working_counter; + printk("EtherCAT: Found %i slaves on bus.\n", master->bus_slaves_count); + + if (!master->bus_slaves_count) return 0; + + if (!(master->bus_slaves = + (EtherCAT_slave_t *) kmalloc(master->bus_slaves_count * + sizeof(EtherCAT_slave_t), GFP_KERNEL))) { + printk(KERN_ERR "EtherCAT: Could not allocate memory for bus slaves!\n"); + return -1; + } // For every slave in the list - for (i = 0; i < slave_count; i++) - { - cur = &slaves[i]; - - if (unlikely(!cur->desc)) { - printk(KERN_ERR "EtherCAT: Slave %i has no description.\n", i); - return -1; - } - - // Set ring position - cur->ring_position = -i; - cur->station_address = i + 1; - - // Write station address - - data[0] = cur->station_address & 0x00FF; - data[1] = (cur->station_address & 0xFF00) >> 8; - - EtherCAT_command_position_write(&cmd, cur->ring_position, - 0x0010, 2, data); - - if (unlikely(EtherCAT_simple_send_receive(master, &cmd) < 0)) - return -1; - - if (unlikely(cmd.working_counter != 1)) { - printk(KERN_ERR "EtherCAT: Slave %i did not repond" - " while writing station address!\n", i); - return -1; - } + for (i = 0; i < master->bus_slaves_count; i++) + { + cur = master->bus_slaves + i; + + EtherCAT_slave_init(cur); // Read base data @@ -495,107 +452,42 @@ // Search for identification in "database" - found = 0; - for (j = 0; j < slave_ident_count; j++) { if (unlikely(slave_idents[j].vendor_id == cur->vendor_id && slave_idents[j].product_code == cur->product_code)) { - found = 1; - - if (unlikely(cur->desc != slave_idents[j].desc)) { - printk(KERN_ERR "EtherCAT: Unexpected slave device" - " \"%s %s\" at position %i. Expected: \"%s %s\"\n", - slave_idents[j].desc->vendor_name, - slave_idents[j].desc->product_name, i, - cur->desc->vendor_name, cur->desc->product_name); - return -1; - } - + cur->desc = slave_idents[j].desc; break; } } - if (unlikely(!found)) { - printk(KERN_ERR "EtherCAT: Unknown slave device" - " (vendor %X, code %X) at position %i.\n", - cur->vendor_id, cur->product_code, i); - return -1; - } - - // Check, if process data domain already exists... - found = 0; - for (j = 0; j < master->domain_count; j++) { - if (cur->domain == master->domains[j].number) { - found = 1; - } - } - - // Create process data domain - if (!found) { - if (master->domain_count + 1 >= ECAT_MAX_DOMAINS) { - printk(KERN_ERR "EtherCAT: Too many domains!\n"); - return -1; - } - - EtherCAT_domain_init(&master->domains[master->domain_count]); - master->domains[master->domain_count].number = cur->domain; - master->domain_count++; - } - } - - // Calculate domain sizes and addresses - - offset = 0; - for (i = 0; i < master->domain_count; i++) - { - dom = master->domains + i; - - dom->logical_offset = offset; - - // Calculate size of the domain - size = 0; - for (j = 0; j < slave_count; j++) { - if (slaves[j].domain == dom->number) { - size += slaves[j].desc->process_data_size; - } - } - - if (size > ECAT_FRAME_BUFFER_SIZE - 14) { - printk(KERN_ERR "EtherCAT: Oversized domain %i: %i / %i Bytes!\n", - dom->number, size, ECAT_FRAME_BUFFER_SIZE - 14); - return -1; - } - - if (!(dom->data = (unsigned char *) kmalloc(sizeof(unsigned char) - * size, GFP_KERNEL))) { - printk(KERN_ERR "EtherCAT: Could not allocate %i bytes of domain" - " data.\n", size); - return -1; - } - - dom->data_size = size; - memset(dom->data, 0x00, size); - - printk(KERN_INFO "EtherCAT: Domain %i: Offset 0x%04X, %i Bytes of" - " process data.\n", dom->number, dom->logical_offset, size); - - // Set logical addresses and data pointers of domain slaves - size = 0; - for (j = 0; j < slave_count; j++) { - if (slaves[j].domain == dom->number) { - slaves[j].process_data = dom->data + size; - slaves[j].logical_address = dom->logical_offset + size; - size += slaves[j].desc->process_data_size; - - printk(KERN_INFO "EtherCAT: Slave %i: Logical Address 0x%04X, %i" - " bytes of process data.\n", j, slaves[j].logical_address, - slaves[j].desc->process_data_size); - } - } - - offset += size; + if (unlikely(!cur->desc)) { + printk(KERN_ERR "EtherCAT: Unknown slave device (vendor %X, code %X) at " + " position %i.\n", cur->vendor_id, cur->product_code, i); + return -1; + } + + // Set ring position + cur->ring_position = -i; + cur->station_address = i + 1; + + // Write station address + + data[0] = cur->station_address & 0x00FF; + data[1] = (cur->station_address & 0xFF00) >> 8; + + EtherCAT_command_position_write(&cmd, cur->ring_position, + 0x0010, 2, data); + + if (unlikely(EtherCAT_simple_send_receive(master, &cmd) < 0)) + return -1; + + if (unlikely(cmd.working_counter != 1)) { + printk(KERN_ERR "EtherCAT: Slave %i did not repond" + " while writing station address!\n", i); + return -1; + } } return 0; @@ -604,6 +496,83 @@ /*****************************************************************************/ /** + Registriert einen Slave beim Master. + + @param master Der EtherCAT-Master + + @return 0 bei Erfolg, sonst < 0 +*/ + +void *EtherCAT_register_slave(EtherCAT_master_t *master, + unsigned int bus_index, + const char *vendor_name, + const char *product_name, + unsigned int domain) +{ + EtherCAT_slave_t *slave; + EtherCAT_domain_t *dom; + unsigned int j; + + if (bus_index >= master->bus_slaves_count) { + printk(KERN_ERR "EtherCAT: Illegal bus index! (%i / %i)\n", bus_index, + master->bus_slaves_count); + return NULL; + } + + slave = master->bus_slaves + bus_index; + + if (slave->process_data) { + printk(KERN_ERR "EtherCAT: Slave %i is already registered!\n", bus_index); + return NULL; + } + + if (strcmp(vendor_name, slave->desc->vendor_name) || + strcmp(product_name, slave->desc->product_name)) { + printk(KERN_ERR "Invalid Slave Type! Requested: \"%s %s\", present: \"%s" + "%s\".\n", vendor_name, product_name, slave->desc->vendor_name, + slave->desc->product_name); + return NULL; + } + + // Check, if process data domain already exists... + dom = NULL; + for (j = 0; j < master->domain_count; j++) { + if (domain == master->domains[j].number) { + dom = master->domains + j; + } + } + + // Create process data domain + if (!dom) { + if (master->domain_count > ECAT_MAX_DOMAINS - 1) { + printk(KERN_ERR "EtherCAT: Too many domains!\n"); + return NULL; + } + + dom = master->domains + master->domain_count; + EtherCAT_domain_init(dom); + dom->number = domain; + dom->logical_offset = master->domain_count * ECAT_FRAME_BUFFER_SIZE; + master->domain_count++; + } + + if (dom->data_size + slave->desc->process_data_size + > ECAT_FRAME_BUFFER_SIZE - 14) { + printk(KERN_ERR "EtherCAT: Oversized domain %i: %i / %i Bytes!\n", + dom->number, dom->data_size + slave->desc->process_data_size, + ECAT_FRAME_BUFFER_SIZE - 14); + return NULL; + } + + slave->process_data = dom->data + dom->data_size; + dom->data_size += slave->desc->process_data_size; + + return slave->process_data; +} + +/*****************************************************************************/ + +/** Liest Daten aus dem Slave-Information-Interface eines EtherCAT-Slaves. @@ -1142,11 +1111,8 @@ /*****************************************************************************/ -EXPORT_SYMBOL(EtherCAT_master_init); -EXPORT_SYMBOL(EtherCAT_master_clear); EXPORT_SYMBOL(EtherCAT_master_open); EXPORT_SYMBOL(EtherCAT_master_close); -EXPORT_SYMBOL(EtherCAT_check_slaves); EXPORT_SYMBOL(EtherCAT_activate_slave); EXPORT_SYMBOL(EtherCAT_deactivate_slave); EXPORT_SYMBOL(EtherCAT_process_data_cycle); diff -r b3beaa00640f -r c0405659a74a drivers/ec_master.h --- a/drivers/ec_master.h Fri Jan 13 13:44:22 2006 +0000 +++ b/drivers/ec_master.h Fri Jan 13 15:39:38 2006 +0000 @@ -27,6 +27,8 @@ struct EtherCAT_master { + EtherCAT_slave_t *bus_slaves; /**< Array von Slaves auf dem Bus */ + unsigned int bus_slaves_count; /**< Anzahl Slaves auf dem Bus */ EtherCAT_device_t *dev; /**< Zeiger auf das zugewiesene EtherCAT-Gerät */ unsigned char command_index; /**< Aktueller Kommando-Index */ unsigned char tx_data[ECAT_FRAME_BUFFER_SIZE]; /**< Statischer Speicher @@ -47,6 +49,17 @@ /*****************************************************************************/ +// Public methods + +void *EtherCAT_register_slave(EtherCAT_master_t *, unsigned int, + const char *, const char *, unsigned int); +int EtherCAT_activate_slave(EtherCAT_master_t *, EtherCAT_slave_t *); +int EtherCAT_deactivate_slave(EtherCAT_master_t *, EtherCAT_slave_t *); +int EtherCAT_process_data_cycle(EtherCAT_master_t *, unsigned int, + unsigned int); + +// Private Methods + // Master creation and deletion void EtherCAT_master_init(EtherCAT_master_t *); void EtherCAT_master_clear(EtherCAT_master_t *); @@ -61,20 +74,14 @@ int EtherCAT_simple_receive(EtherCAT_master_t *, EtherCAT_command_t *); // Slave management -int EtherCAT_check_slaves(EtherCAT_master_t *, EtherCAT_slave_t *, - unsigned int); +int EtherCAT_scan_for_slaves(EtherCAT_master_t *); int EtherCAT_read_slave_information(EtherCAT_master_t *, unsigned short int, unsigned short int, unsigned int *); -int EtherCAT_activate_slave(EtherCAT_master_t *, EtherCAT_slave_t *); -int EtherCAT_deactivate_slave(EtherCAT_master_t *, EtherCAT_slave_t *); int EtherCAT_state_change(EtherCAT_master_t *, EtherCAT_slave_t *, unsigned char); -// Process data -int EtherCAT_process_data_cycle(EtherCAT_master_t *, unsigned int, - unsigned int); +// Misc. -// Private functions void output_debug_data(const EtherCAT_master_t *); void ecat_output_lost_frames(EtherCAT_master_t *); diff -r b3beaa00640f -r c0405659a74a drivers/ec_module.c --- a/drivers/ec_module.c Fri Jan 13 13:44:22 2006 +0000 +++ b/drivers/ec_module.c Fri Jan 13 15:39:38 2006 +0000 @@ -24,6 +24,9 @@ #include "ec_module.h" +int __init ecat_init_module(void); +void __exit ecat_cleanup_module(void); + /*****************************************************************************/ #define LIT(X) #X @@ -229,6 +232,11 @@ return NULL; } + if (EtherCAT_scan_for_slaves(&ecat_masters[index]) != 0) { + printk(KERN_ERR "EtherCAT: Could not scan for slaves!\n"); + return NULL; + } + ecat_masters_reserved[index] = 1; printk(KERN_INFO "EtherCAT: Reserved master %i.\n", index); diff -r b3beaa00640f -r c0405659a74a drivers/ec_module.h --- a/drivers/ec_module.h Fri Jan 13 13:44:22 2006 +0000 +++ b/drivers/ec_module.h Fri Jan 13 15:39:38 2006 +0000 @@ -26,9 +26,6 @@ /*****************************************************************************/ -int __init ecat_init_module(void); -void __exit ecat_cleanup_module(void); - // Registration of devices int EtherCAT_register_device(int, EtherCAT_device_t *); void EtherCAT_unregister_device(int, EtherCAT_device_t *); diff -r b3beaa00640f -r c0405659a74a drivers/ec_slave.c --- a/drivers/ec_slave.c Fri Jan 13 13:44:22 2006 +0000 +++ b/drivers/ec_slave.c Fri Jan 13 15:39:38 2006 +0000 @@ -49,21 +49,6 @@ /*****************************************************************************/ /** - EtherCAT-Slave-Destruktor. - - Im Moment ohne Funktionalität. - - @param slave Zeiger auf den zu zerstörenden Slave -*/ - -void EtherCAT_slave_clear(EtherCAT_slave_t *slave) -{ - // Nothing yet... -} - -/*****************************************************************************/ - -/** Liest einen bestimmten Kanal des Slaves als Integer-Wert. Prüft zuerst, ob der entsprechende Slave eine diff -r b3beaa00640f -r c0405659a74a drivers/ec_slave.h --- a/drivers/ec_slave.h Fri Jan 13 13:44:22 2006 +0000 +++ b/drivers/ec_slave.h Fri Jan 13 15:39:38 2006 +0000 @@ -62,7 +62,6 @@ // Slave construction and deletion void EtherCAT_slave_init(EtherCAT_slave_t *); -void EtherCAT_slave_clear(EtherCAT_slave_t *); int EtherCAT_read_value(EtherCAT_slave_t *, unsigned int); void EtherCAT_write_value(EtherCAT_slave_t *, unsigned int, int);