martin@1583: /****************************************************************************** martin@1583: * martin@1583: * $Id$ martin@1583: * martin@1583: * Copyright (C) 2006-2008 Florian Pose, Ingenieurgemeinschaft IgH martin@1583: * martin@1583: * This file is part of the IgH EtherCAT Master. martin@1583: * martin@1583: * The IgH EtherCAT Master is free software; you can redistribute it and/or martin@1583: * modify it under the terms of the GNU General Public License version 2, as martin@1583: * published by the Free Software Foundation. martin@1583: * martin@1583: * The IgH EtherCAT Master is distributed in the hope that it will be useful, martin@1583: * but WITHOUT ANY WARRANTY; without even the implied warranty of martin@1583: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General martin@1583: * Public License for more details. martin@1583: * martin@1583: * You should have received a copy of the GNU General Public License along martin@1583: * with the IgH EtherCAT Master; if not, write to the Free Software martin@1583: * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA martin@1583: * martin@1583: * --- martin@1583: * martin@1583: * The license mentioned above concerns the source code only. Using the martin@1583: * EtherCAT technology and brand is only permitted in compliance with the martin@1583: * industrial property and similar rights of Beckhoff Automation GmbH. martin@1583: * martin@1583: *****************************************************************************/ martin@1583: martin@1583: /** \file martin@1583: * EtherCAT slave (SDO) state machine. martin@1583: */ martin@1583: martin@1583: /*****************************************************************************/ martin@1583: martin@1583: #include "globals.h" martin@1583: #include "master.h" martin@1583: #include "mailbox.h" martin@1583: martin@1583: #include "fsm_slave.h" martin@1583: martin@1583: /*****************************************************************************/ martin@1583: martin@1583: void ec_fsm_slave_state_idle(ec_fsm_slave_t *); martin@1601: void ec_fsm_slave_state_ready(ec_fsm_slave_t *); martin@1597: int ec_fsm_slave_action_process_sdo(ec_fsm_slave_t *); martin@1597: int ec_fsm_slave_action_process_foe(ec_fsm_slave_t *); martin@1583: void ec_fsm_slave_state_sdo_request(ec_fsm_slave_t *); martin@1597: void ec_fsm_slave_state_foe_request(ec_fsm_slave_t *); martin@1583: martin@1583: martin@1583: /*****************************************************************************/ martin@1583: martin@1583: /** Constructor. martin@1583: */ martin@1583: void ec_fsm_slave_init( martin@1583: ec_fsm_slave_t *fsm, /**< Slave state machine. */ martin@1583: ec_slave_t *slave, /**< EtherCAT slave. */ martin@1583: ec_datagram_t *datagram /**< Datagram object to use. */ martin@1583: ) martin@1583: { martin@1583: fsm->slave = slave; martin@1583: fsm->datagram = datagram; martin@1583: fsm->datagram->data_size = 0; fp@1804: fp@1804: if (slave->master->debug_level) fp@1804: EC_DBG("Init FSM for slave %u...\n", slave->ring_position); fp@1804: fp@1804: fsm->state = ec_fsm_slave_state_idle; martin@1583: martin@1583: // init sub-state-machines martin@1583: ec_fsm_coe_init(&fsm->fsm_coe, fsm->datagram); martin@1597: ec_fsm_foe_init(&fsm->fsm_foe, fsm->datagram); martin@1583: } martin@1583: martin@1583: /*****************************************************************************/ martin@1583: martin@1583: /** Destructor. martin@1583: */ martin@1583: void ec_fsm_slave_clear( martin@1583: ec_fsm_slave_t *fsm /**< Master state machine. */ martin@1583: ) martin@1583: { martin@1583: // clear sub-state machines martin@1583: ec_fsm_coe_clear(&fsm->fsm_coe); martin@1597: ec_fsm_foe_clear(&fsm->fsm_foe); martin@1583: } martin@1583: martin@1583: /*****************************************************************************/ martin@1583: martin@1583: /** Executes the current state of the state machine. martin@1583: * martin@1583: * If the state machine's datagram is not sent or received yet, the execution martin@1583: * of the state machine is delayed to the next cycle. martin@1583: */ martin@1583: void ec_fsm_slave_exec( martin@1583: ec_fsm_slave_t *fsm /**< Slave state machine. */ martin@1583: ) martin@1583: { martin@1583: if (fsm->datagram->state == EC_DATAGRAM_SENT martin@1583: || fsm->datagram->state == EC_DATAGRAM_QUEUED) { martin@1583: // datagram was not sent or received yet. martin@1583: return; martin@1583: } martin@1583: martin@1583: fsm->state(fsm); martin@1583: return; martin@1583: } martin@1583: martin@1601: martin@1601: /*****************************************************************************/ martin@1601: martin@1601: /** Sets the current state of the state machine to READY martin@1601: * martin@1601: */ martin@1601: void ec_fsm_slave_ready( fp@1804: ec_fsm_slave_t *fsm /**< Slave state machine. */ fp@1804: ) fp@1804: { fp@1804: if (fsm->state == ec_fsm_slave_state_idle) { fp@1804: if (fsm->slave->master->debug_level) { fp@1804: EC_DBG("Slave %u ready for SDO/FOE.\n", fp@1804: fsm->slave->ring_position); fp@1804: } fp@1804: fsm->state = ec_fsm_slave_state_ready; fp@1804: } fp@1804: return; martin@1601: } martin@1601: martin@1583: /****************************************************************************** martin@1583: * Slave state machine martin@1583: *****************************************************************************/ martin@1583: martin@1597: /*****************************************************************************/ martin@1597: martin@1583: /** Slave state: IDLE. martin@1583: * martin@1583: * martin@1583: */ martin@1583: void ec_fsm_slave_state_idle( fp@1804: ec_fsm_slave_t *fsm /**< Slave state machine. */ fp@1804: ) fp@1804: { fp@1804: // do nothing martin@1601: } martin@1601: martin@1601: martin@1601: /*****************************************************************************/ martin@1601: martin@1601: /** Slave state: READY. martin@1601: * martin@1601: * martin@1601: */ martin@1601: void ec_fsm_slave_state_ready( fp@1804: ec_fsm_slave_t *fsm /**< Slave state machine. */ fp@1804: ) fp@1804: { fp@1804: // Check for pending external SDO requests fp@1804: if (ec_fsm_slave_action_process_sdo(fsm)) fp@1804: return; fp@1804: fp@1804: // Check for pending FOE requests fp@1804: if (ec_fsm_slave_action_process_foe(fsm)) fp@1804: return; fp@1804: } martin@1597: martin@1597: /*****************************************************************************/ martin@1597: martin@1597: /** Check for pending SDO requests and process one. martin@1597: * martin@1597: * \return non-zero, if an SDO request is processed. martin@1597: */ martin@1597: int ec_fsm_slave_action_process_sdo( martin@1597: ec_fsm_slave_t *fsm /**< Slave state machine. */ martin@1597: ) martin@1597: { martin@1583: ec_slave_t *slave = fsm->slave; martin@1583: ec_master_t *master = slave->master; martin@1583: ec_master_sdo_request_t *request, *next; martin@1583: martin@1597: // search the first external request to be processed martin@1596: list_for_each_entry_safe(request, next, &slave->slave_sdo_requests, list) { martin@1598: martin@1604: list_del_init(&request->list); // dequeue fp@1804: if (slave->current_state & EC_SLAVE_STATE_ACK_ERR) { fp@1804: EC_WARN("Aborting SDO request, slave %u has ERROR.\n", fp@1804: slave->ring_position); fp@1804: request->req.state = EC_INT_REQUEST_FAILURE; fp@1804: wake_up(&slave->sdo_queue); fp@1804: fsm->sdo_request = NULL; fp@1804: fsm->state = ec_fsm_slave_state_idle; fp@1804: return 0; fp@1804: } fp@1804: fp@1804: if (slave->current_state == EC_SLAVE_STATE_INIT) { fp@1804: EC_WARN("Aborting SDO request, slave %u is in INIT.\n", fp@1804: slave->ring_position); fp@1804: request->req.state = EC_INT_REQUEST_FAILURE; fp@1804: wake_up(&slave->sdo_queue); fp@1804: fsm->sdo_request = NULL; fp@1804: fsm->state = ec_fsm_slave_state_idle; fp@1804: return 0; fp@1804: } fp@1804: fp@1804: request->req.state = EC_INT_REQUEST_BUSY; martin@1583: martin@1583: // Found pending SDO request. Execute it! martin@1583: if (master->debug_level) martin@1583: EC_DBG("Processing SDO request for slave %u...\n", martin@1583: slave->ring_position); martin@1583: martin@1583: // Start SDO transfer martin@1583: fsm->sdo_request = &request->req; martin@1583: fsm->state = ec_fsm_slave_state_sdo_request; martin@1583: ec_fsm_coe_transfer(&fsm->fsm_coe, slave, &request->req); martin@1583: ec_fsm_coe_exec(&fsm->fsm_coe); // execute immediately martin@1597: ec_master_queue_external_datagram(fsm->slave->master,fsm->datagram); martin@1597: return 1; martin@1597: } martin@1597: return 0; martin@1597: } martin@1597: martin@1597: martin@1597: /*****************************************************************************/ martin@1597: martin@1597: /** Check for pending FOE requests and process one. martin@1597: * martin@1597: * \return non-zero, if an FOE request is processed. martin@1597: */ martin@1597: int ec_fsm_slave_action_process_foe( martin@1597: ec_fsm_slave_t *fsm /**< Slave state machine. */ martin@1597: ) martin@1597: { martin@1597: ec_slave_t *slave = fsm->slave; martin@1597: ec_master_t *master = slave->master; martin@1597: ec_master_foe_request_t *request, *next; martin@1597: martin@1597: // search the first request to be processed martin@1597: list_for_each_entry_safe(request, next, &slave->foe_requests, list) { fp@1804: if (slave->current_state & EC_SLAVE_STATE_ACK_ERR) { fp@1804: EC_WARN("Aborting FOE request, slave %u has ERROR.\n", fp@1804: slave->ring_position); fp@1804: request->req.state = EC_INT_REQUEST_FAILURE; fp@1804: wake_up(&slave->sdo_queue); fp@1804: fsm->sdo_request = NULL; fp@1804: fsm->state = ec_fsm_slave_state_idle; fp@1804: return 0; fp@1804: } fp@1804: list_del_init(&request->list); // dequeue martin@1597: request->req.state = EC_INT_REQUEST_BUSY; martin@1597: martin@1597: if (master->debug_level) fp@1804: EC_DBG("Processing FOE request for slave %u.\n", martin@1597: slave->ring_position); martin@1597: martin@1597: fsm->foe_request = &request->req; martin@1597: fsm->state = ec_fsm_slave_state_foe_request; martin@1597: ec_fsm_foe_transfer(&fsm->fsm_foe, slave, &request->req); martin@1597: ec_fsm_foe_exec(&fsm->fsm_foe); martin@1597: ec_master_queue_external_datagram(fsm->slave->master,fsm->datagram); martin@1597: return 1; martin@1597: } martin@1597: return 0; martin@1597: } martin@1597: martin@1597: martin@1583: martin@1583: /*****************************************************************************/ martin@1583: martin@1583: /** Slave state: SDO_REQUEST. martin@1583: */ martin@1583: void ec_fsm_slave_state_sdo_request( martin@1583: ec_fsm_slave_t *fsm /**< Slave state machine. */ martin@1583: ) martin@1583: { martin@1583: ec_slave_t *slave = fsm->slave; martin@1583: ec_master_t *master = slave->master; martin@1583: ec_sdo_request_t *request = fsm->sdo_request; martin@1583: martin@1583: if (ec_fsm_coe_exec(&fsm->fsm_coe)) martin@1583: { martin@1597: ec_master_queue_external_datagram(fsm->slave->master,fsm->datagram); martin@1583: return; martin@1583: } martin@1583: if (!ec_fsm_coe_success(&fsm->fsm_coe)) { martin@1583: EC_DBG("Failed to process SDO request for slave %u.\n", martin@1583: fsm->slave->ring_position); martin@1583: request->state = EC_INT_REQUEST_FAILURE; fp@1804: wake_up(&slave->sdo_queue); fp@1804: fsm->sdo_request = NULL; fp@1804: fsm->state = ec_fsm_slave_state_idle; martin@1583: return; martin@1583: } martin@1583: martin@1597: if (master->debug_level) martin@1597: EC_DBG("Finished SDO request for slave %u.\n", martin@1597: fsm->slave->ring_position); martin@1597: martin@1583: // SDO request finished martin@1583: request->state = EC_INT_REQUEST_SUCCESS; martin@1596: wake_up(&slave->sdo_queue); martin@1583: martin@1597: fsm->sdo_request = NULL; fp@1804: fsm->state = ec_fsm_slave_state_ready; fp@1804: } martin@1597: martin@1597: /*****************************************************************************/ martin@1597: martin@1597: /** Slave state: FOE REQUEST. martin@1597: */ martin@1597: void ec_fsm_slave_state_foe_request( martin@1597: ec_fsm_slave_t *fsm /**< Slave state machine. */ martin@1597: ) martin@1597: { martin@1597: ec_slave_t *slave = fsm->slave; martin@1597: ec_master_t *master = slave->master; martin@1597: ec_foe_request_t *request = fsm->foe_request; martin@1597: martin@1597: if (ec_fsm_foe_exec(&fsm->fsm_foe)) martin@1597: { martin@1597: ec_master_queue_external_datagram(fsm->slave->master,fsm->datagram); martin@1597: return; martin@1597: } martin@1597: martin@1597: if (!ec_fsm_foe_success(&fsm->fsm_foe)) { martin@1597: EC_ERR("Failed to handle FoE request to slave %u.\n", martin@1597: slave->ring_position); martin@1597: request->state = EC_INT_REQUEST_FAILURE; martin@1597: wake_up(&slave->foe_queue); martin@1597: fsm->foe_request = NULL; martin@1597: fsm->state = ec_fsm_slave_state_idle; martin@1597: return; martin@1597: } martin@1597: martin@1597: // finished transferring FoE martin@1583: if (master->debug_level) martin@1597: EC_DBG("Successfully transferred %u bytes of FoE data from/to" martin@1597: " slave %u.\n", request->data_size, slave->ring_position); martin@1597: martin@1597: request->state = EC_INT_REQUEST_SUCCESS; martin@1597: wake_up(&slave->foe_queue); martin@1597: martin@1597: fsm->foe_request = NULL; fp@1804: fsm->state = ec_fsm_slave_state_ready; fp@1804: } fp@1804: fp@1804: /*****************************************************************************/