master/fsm_foe.c
branch1.4-foe
changeset 1707 11ec009e145d
child 1336 e27b37e80a99
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/master/fsm_foe.c	Mon Jan 19 10:18:41 2009 +0000
@@ -0,0 +1,851 @@
+/******************************************************************************
+ *
+ *  $Id:$
+ *
+ *  Copyright (C) 2008  Olav Zarges, imc Meßsysteme GmbH
+ *
+ *  This file is part of the IgH EtherCAT Master.
+ *
+ *  The IgH EtherCAT Master is free software; you can redistribute it
+ *  and/or modify it under the terms of the GNU General Public License
+ *  as published by the Free Software Foundation; either version 2 of the
+ *  License, or (at your option) any later version.
+ *
+ *  The IgH EtherCAT Master is distributed in the hope that it will be
+ *  useful, but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with the IgH EtherCAT Master; if not, write to the Free Software
+ *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ *
+ *  The right to use EtherCAT Technology is granted and comes free of
+ *  charge under condition of compatibility of product made by
+ *  Licensee. People intending to distribute/sell products based on the
+ *  code, have to sign an agreement to guarantee that products using
+ *  software based on IgH EtherCAT master stay compatible with the actual
+ *  EtherCAT specification (which are released themselves as an open
+ *  standard) as the (only) precondition to have the right to use EtherCAT
+ *  Technology, IP and trade marks.
+ *
+ *****************************************************************************/
+
+/**
+   \file
+   EtherCAT FoE state machines.
+*/
+
+/*****************************************************************************/
+
+#include "globals.h"
+#include "master.h"
+#include "mailbox.h"
+#include "fsm_foe.h"
+#include "foe.h"
+
+/*****************************************************************************/
+
+/** Maximum time in ms to wait for responses when reading out the dictionary.
+ */
+#define EC_FSM_FOE_TIMEOUT 3000
+
+#define EC_MBOX_TYPE_FILEACCESS 0x04
+
+/*****************************************************************************/
+
+int		ec_foe_prepare_data_send( ec_fsm_foe_t * );
+int		ec_foe_prepare_wrq_send( ec_fsm_foe_t * );
+int		ec_foe_prepare_rrq_send( ec_fsm_foe_t * );
+int		ec_foe_prepare_send_ack( ec_fsm_foe_t * );
+
+void	ec_foe_set_tx_error( ec_fsm_foe_t *, uint32_t );
+void	ec_foe_set_rx_error( ec_fsm_foe_t *, uint32_t );
+
+void	ec_fsm_foe_write(ec_fsm_foe_t * );
+void	ec_fsm_foe_read(ec_fsm_foe_t * );
+void 	ec_fsm_foe_end( ec_fsm_foe_t * );
+void 	ec_fsm_foe_error( ec_fsm_foe_t * );
+
+void	ec_fsm_foe_state_wrq_sent( ec_fsm_foe_t * );
+void	ec_fsm_foe_state_rrq_sent( ec_fsm_foe_t * );
+
+void	ec_fsm_foe_state_ack_check( ec_fsm_foe_t * );
+void	ec_fsm_foe_state_ack_read( ec_fsm_foe_t * );
+
+void	ec_fsm_foe_state_data_sent( ec_fsm_foe_t * );
+
+void	ec_fsm_foe_state_data_check( ec_fsm_foe_t * );
+void	ec_fsm_foe_state_data_read ( ec_fsm_foe_t * );
+void	ec_fsm_foe_state_sent_ack( ec_fsm_foe_t * );
+
+void	ec_fsm_foe_write_start( ec_fsm_foe_t * );
+void	ec_fsm_foe_read_start(ec_fsm_foe_t * );
+
+/*****************************************************************************/
+
+/**
+   Constructor.
+*/
+
+void ec_fsm_foe_init(ec_fsm_foe_t *fsm, /**< finite state machine */
+                     ec_datagram_t *datagram /**< datagram */
+                     )
+{
+    fsm->state     = NULL;
+    fsm->datagram  = datagram;
+    fsm->rx_errors = 0;
+    fsm->tx_errors = 0;
+}
+
+/*****************************************************************************/
+
+/**
+   Destructor.
+*/
+
+void ec_fsm_foe_clear(ec_fsm_foe_t *fsm /**< finite state machine */)
+{
+}
+
+/*****************************************************************************/
+
+/**
+   Executes the current state of the state machine.
+   \return false, if state machine has terminated
+*/
+
+int ec_fsm_foe_exec(ec_fsm_foe_t *fsm /**< finite state machine */)
+{
+    fsm->state(fsm);
+
+    return fsm->state != ec_fsm_foe_end && fsm->state != ec_fsm_foe_error;
+}
+
+/*****************************************************************************/
+
+/**
+   Returns, if the state machine terminated with success.
+   \return non-zero if successful.
+*/
+
+int ec_fsm_foe_success(ec_fsm_foe_t *fsm /**< Finite state machine */)
+{
+    return fsm->state == ec_fsm_foe_end;
+}
+
+/*****************************************************************************/
+
+void ec_fsm_foe_transfer(
+        ec_fsm_foe_t *fsm, /**< State machine. */
+        ec_slave_t *slave, /**< EtherCAT slave. */
+        ec_foe_request_t *request /**< Sdo request. */
+        )
+{
+    fsm->slave = slave;
+    fsm->request = request;
+    if (request->dir == EC_DIR_OUTPUT) {
+        fsm->state = ec_fsm_foe_write;
+    }
+    else {
+        fsm->state = ec_fsm_foe_read;
+    }
+}
+
+/*****************************************************************************/
+
+/**
+   State: ERROR.
+*/
+
+void ec_fsm_foe_error(ec_fsm_foe_t *fsm /**< finite state machine */)
+{
+#ifdef	myDEBUG
+	printk("ec_fsm_foe_error()\n");
+#endif
+}
+
+/*****************************************************************************/
+
+/**
+   State: END.
+*/
+
+void ec_fsm_foe_end(ec_fsm_foe_t *fsm /**< finite state machine */)
+{
+#ifdef	myDEBUG
+    printk("ec_fsm_foe_end\n");
+#endif
+}
+
+/*****************************************************************************/
+
+#define	EC_MBOX_HEADER_SIZE		6
+// uint16_t	Length
+// uint16_t	Address
+// uint8_t	reserved
+// uint8_t	Type:4
+// uint8_t	Counter:4
+
+#define	EC_FOE_HEADER_SIZE		6
+// uint8_t	OpMode
+// uint8_t	reserved
+// uint32_t	PacketNo, Password, ErrorCode
+
+enum {
+	EC_FOE_OPMODE_RRQ  = 1,
+	EC_FOE_OPMODE_WRQ  = 2,
+	EC_FOE_OPMODE_DATA = 3,
+	EC_FOE_OPMODE_ACK  = 4,
+	EC_FOE_OPMODE_ERR  = 5,
+	EC_FOE_OPMODE_BUSY = 6
+} ec_foe_opmode_t;
+
+/*****************************************************************************/
+/**
+   Sends a file or the next fragment.
+*/
+
+int ec_foe_prepare_data_send( ec_fsm_foe_t *fsm ) {
+    size_t       remaining_size, current_size;
+    uint8_t*     data;
+
+    remaining_size = fsm->tx_buffer_size - fsm->tx_buffer_offset;
+
+    if (remaining_size < fsm->slave->sii.tx_mailbox_size - EC_MBOX_HEADER_SIZE - EC_FOE_HEADER_SIZE) {
+        current_size = remaining_size;
+        fsm->tx_last_packet = 1;
+    }
+    else {
+        current_size = fsm->slave->sii.tx_mailbox_size - EC_MBOX_HEADER_SIZE - EC_FOE_HEADER_SIZE;
+    }
+
+    if (!(data = ec_slave_mbox_prepare_send(fsm->slave, fsm->datagram,
+    		EC_MBOX_TYPE_FILEACCESS, current_size + EC_FOE_HEADER_SIZE)))
+        return -1;
+
+    EC_WRITE_U8 ( data, EC_FOE_OPMODE_DATA ); 		// OpMode = DataBlock req.
+    EC_WRITE_U32( data + 2, fsm->tx_packet_no );	// PacketNo, Password
+
+    memcpy(data + EC_FOE_HEADER_SIZE, fsm->tx_buffer + fsm->tx_buffer_offset, current_size);
+
+    fsm->tx_current_size = current_size;
+
+    return 0;
+}
+
+/*****************************************************************************/
+/**
+   Prepare a write request (WRQ) with filename
+*/
+
+int ec_foe_prepare_wrq_send( ec_fsm_foe_t *fsm ) {
+    size_t current_size;
+    uint8_t *data;
+
+    fsm->tx_buffer_offset = 0;
+    fsm->tx_current_size = 0;
+    fsm->tx_packet_no = 0;
+	fsm->tx_last_packet = 0;
+
+    current_size = fsm->tx_filename_len;
+
+    if (!(data = ec_slave_mbox_prepare_send(fsm->slave, fsm->datagram,
+    		EC_MBOX_TYPE_FILEACCESS, current_size + EC_FOE_HEADER_SIZE)))
+        return -1;
+
+    EC_WRITE_U16( data, EC_FOE_OPMODE_WRQ); // fsm write request
+    EC_WRITE_U32( data + 2, fsm->tx_packet_no );
+
+    memcpy(data + EC_FOE_HEADER_SIZE, fsm->tx_filename, current_size);
+
+    return 0;
+}
+
+/*****************************************************************************/
+
+char		tx_buffer[0x1000];
+void ec_fsm_foe_write(ec_fsm_foe_t *fsm /**< finite state machine */)
+{
+    fsm->tx_buffer = fsm->request->buffer;
+    fsm->tx_buffer_size = fsm->request->data_size;
+    fsm->tx_buffer_offset = 0;
+
+	fsm->tx_filename = fsm->request->file_name;
+	fsm->tx_filename_len = strlen(fsm->tx_filename);
+
+	fsm->state = ec_fsm_foe_write_start;
+
+#ifdef use_ext_buffer
+	{
+		int i;
+		fsm->tx_data = tx_buffer;
+		for (i=0 ; i<sizeof(tx_buffer) ; i++) {
+			tx_buffer[i] = (uint8_t)(i);
+		}
+		fsm->tx_data_len = sizeof(tx_buffer);
+	}
+#endif
+}
+
+/*****************************************************************************/
+/**
+   Initializes the SII write state machine.
+*/
+
+void ec_fsm_foe_write_start(ec_fsm_foe_t *fsm /**< finite state machine */)
+{
+    ec_slave_t *slave = fsm->slave;
+
+    fsm->tx_buffer_offset = 0;
+    fsm->tx_current_size = 0;
+    fsm->tx_packet_no = 0;
+	fsm->tx_last_packet = 0;
+
+#ifdef	myDEBUG
+    printk("ec_fsm_foe_write_start()\n");
+#endif
+
+    if (!(slave->sii.mailbox_protocols & EC_MBOX_FOE)) {
+        ec_foe_set_tx_error(fsm, FOE_MBOX_PROT_ERROR);
+        EC_ERR("Slave %u does not support FoE!\n", slave->ring_position);
+        return;
+    }
+
+    if (ec_foe_prepare_wrq_send(fsm)) {
+        ec_foe_set_tx_error(fsm, FOE_PROT_ERROR);
+        return;
+    }
+
+    fsm->state = ec_fsm_foe_state_wrq_sent;
+}
+
+/*****************************************************************************/
+
+void ec_fsm_foe_state_ack_check( ec_fsm_foe_t *fsm ) {
+    ec_datagram_t *datagram = fsm->datagram;
+    ec_slave_t *slave = fsm->slave;
+
+#ifdef	myDEBUG
+//	printk("ec_fsm_foe_ack_check()\n");
+#endif
+    if (datagram->state != EC_DATAGRAM_RECEIVED) {
+    	ec_foe_set_rx_error(fsm, FOE_RECEIVE_ERROR);
+        EC_ERR("Failed to receive FoE mailbox check datagram for slave %u"
+                " (datagram state %u).\n",
+               slave->ring_position, datagram->state);
+        return;
+    }
+
+    if (datagram->working_counter != 1) {
+    	// slave hat noch nichts in die Mailbox getan
+        ec_foe_set_rx_error(fsm, FOE_WC_ERROR);
+        EC_ERR("Reception of FoE mailbox check datagram failed on slave %u: ",
+               slave->ring_position);
+        ec_datagram_print_wc_error(datagram);
+	    return;
+	}
+
+    if (!ec_slave_mbox_check(datagram)) {
+        unsigned long diff_ms =
+            (datagram->jiffies_received - fsm->jiffies_start) * 1000 / HZ;
+        if (diff_ms >= EC_FSM_FOE_TIMEOUT) {
+        	ec_foe_set_tx_error(fsm, FOE_TIMEOUT_ERROR);
+            EC_ERR("Timeout while waiting for ack response "
+                    "on slave %u.\n", slave->ring_position);
+            return;
+        }
+//        EC_ERR("WAIT!!!!!!!!!!!!!\n");
+        ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
+        fsm->retries = EC_FSM_RETRIES;
+        return;
+    }
+
+    // Fetch response
+    ec_slave_mbox_prepare_fetch(slave, datagram); // can not fail.
+
+    fsm->retries = EC_FSM_RETRIES;
+    fsm->state = ec_fsm_foe_state_ack_read;
+}
+
+/*****************************************************************************/
+
+void ec_fsm_foe_state_ack_read( ec_fsm_foe_t *fsm ) {
+
+    ec_datagram_t *datagram = fsm->datagram;
+    ec_slave_t *slave = fsm->slave;
+    uint8_t *data, mbox_prot;
+    uint16_t opMode;
+    size_t rec_size;
+
+#ifdef	myDEBUG
+	printk("ec_fsm_foe_ack_read()\n");
+#endif
+    if (datagram->state != EC_DATAGRAM_RECEIVED) {
+    	ec_foe_set_rx_error(fsm, FOE_RECEIVE_ERROR);
+        EC_ERR("Failed to receive FoE ack response datagram for"
+               " slave %u (datagram state %u).\n",
+               slave->ring_position, datagram->state);
+        return;
+    }
+
+    if (datagram->working_counter != 1) {
+        ec_foe_set_rx_error(fsm, FOE_WC_ERROR);
+        EC_ERR("Reception of FoE ack response failed on slave %u: ",
+                slave->ring_position);
+        ec_datagram_print_wc_error(datagram);
+        return;
+    }
+
+	if (!(data = ec_slave_mbox_fetch(fsm->slave, datagram, &mbox_prot, &rec_size))) {
+    	ec_foe_set_tx_error(fsm, FOE_PROT_ERROR);
+        return;
+    }
+
+    if (mbox_prot != EC_MBOX_TYPE_FILEACCESS) { // FoE
+    	ec_foe_set_tx_error(fsm, FOE_MBOX_PROT_ERROR);
+        EC_ERR("Received mailbox protocol 0x%02X as response.\n", mbox_prot);
+        return;
+    }
+
+    opMode = EC_READ_U16(data);
+
+    if ( opMode == EC_FOE_OPMODE_BUSY ) {
+    	// slave ist noch nicht bereit
+        if (ec_foe_prepare_data_send(fsm)) {
+        	ec_foe_set_tx_error(fsm, FOE_PROT_ERROR);
+            EC_ERR("Slave is busy.\n");
+        	return;
+        }
+        fsm->state = ec_fsm_foe_state_data_sent;
+        return;
+    }
+
+    if ( opMode == EC_FOE_OPMODE_ACK ) {
+        fsm->tx_packet_no++;
+        fsm->tx_buffer_offset += fsm->tx_current_size;
+
+        if (fsm->tx_last_packet) {
+        	fsm->state = ec_fsm_foe_end;
+        	return;
+        }
+
+        if (ec_foe_prepare_data_send(fsm)) {
+        	ec_foe_set_tx_error(fsm, FOE_PROT_ERROR);
+	    	return;
+	    }
+        fsm->state = ec_fsm_foe_state_data_sent;
+    	return;
+    }
+	ec_foe_set_tx_error(fsm, FOE_ACK_ERROR);
+}
+
+/*****************************************************************************/
+/**
+   State: WRQ SENT.
+   Checks is the previous transmit datagram succeded and sends the next
+   fragment, if necessary.
+*/
+
+void ec_fsm_foe_state_wrq_sent( ec_fsm_foe_t *fsm ) {
+    ec_datagram_t *datagram = fsm->datagram;
+    ec_slave_t *slave = fsm->slave;
+
+#ifdef	myDEBUG
+	printk("ec_foe_state_sent_wrq()\n");
+#endif
+    if (datagram->state != EC_DATAGRAM_RECEIVED) {
+    	ec_foe_set_rx_error(fsm, FOE_RECEIVE_ERROR);
+        EC_ERR("Failed to send FoE WRQ for slave %u"
+                " (datagram state %u).\n",
+                slave->ring_position, datagram->state);
+        return;
+    }
+
+    if (datagram->working_counter != 1) {
+    	// slave hat noch nichts in die Mailbox getan
+        ec_foe_set_rx_error(fsm, FOE_WC_ERROR);
+        EC_ERR("Reception of FoE WRQ failed on slave %u: ",
+                slave->ring_position);
+        ec_datagram_print_wc_error(datagram);
+    	return;
+    }
+
+    fsm->jiffies_start = datagram->jiffies_sent;
+
+    ec_slave_mbox_prepare_check(fsm->slave, datagram); // can not fail.
+
+    fsm->retries = EC_FSM_RETRIES;
+    fsm->state = ec_fsm_foe_state_ack_check;
+}
+
+/*****************************************************************************/
+/**
+   State: WRQ SENT.
+   Checks is the previous transmit datagram succeded and sends the next
+   fragment, if necessary.
+*/
+
+void ec_fsm_foe_state_data_sent( ec_fsm_foe_t *fsm ) {
+    ec_datagram_t *datagram = fsm->datagram;
+    ec_slave_t *slave = fsm->slave;
+
+#ifdef	myDEBUG
+	printk("ec_fsm_foe_state_data_sent()\n");
+#endif
+    if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
+    	ec_foe_set_tx_error(fsm, FOE_RECEIVE_ERROR);
+        EC_ERR("Failed to receive FoE ack response datagram for"
+               " slave %u (datagram state %u).\n",
+               slave->ring_position, datagram->state);
+        return;
+    }
+
+    if (fsm->datagram->working_counter != 1) {
+	    ec_foe_set_tx_error(fsm, FOE_WC_ERROR);
+        EC_ERR("Reception of FoE data send failed on slave %u: ",
+                slave->ring_position);
+        ec_datagram_print_wc_error(datagram);
+        return;
+    }
+
+    ec_slave_mbox_prepare_check(fsm->slave, fsm->datagram);
+    fsm->jiffies_start = jiffies;
+    fsm->retries = EC_FSM_RETRIES;
+    fsm->state = ec_fsm_foe_state_ack_check;
+}
+
+/*****************************************************************************/
+/**
+   Prepare a read request (RRQ) with filename
+*/
+
+int ec_foe_prepare_rrq_send( ec_fsm_foe_t *fsm ) {
+    size_t current_size;
+    uint8_t *data;
+
+    current_size = fsm->rx_filename_len;
+
+    if (!(data = ec_slave_mbox_prepare_send(fsm->slave, fsm->datagram,
+    		EC_MBOX_TYPE_FILEACCESS, current_size + EC_FOE_HEADER_SIZE)))
+        return -1;
+
+    EC_WRITE_U16( data, EC_FOE_OPMODE_RRQ); // fsm read request
+    EC_WRITE_U32( data + 2, 0 );
+
+    memcpy(data + EC_FOE_HEADER_SIZE, fsm->rx_filename, current_size);
+
+    return 0;
+}
+
+
+/*****************************************************************************/
+
+int ec_foe_prepare_send_ack( ec_fsm_foe_t *foe ) {
+    uint8_t *data;
+
+    if (!(data = ec_slave_mbox_prepare_send(foe->slave, foe->datagram,
+    		EC_MBOX_TYPE_FILEACCESS, EC_FOE_HEADER_SIZE)))
+        return -1;
+
+    EC_WRITE_U16( data, EC_FOE_OPMODE_ACK);
+    EC_WRITE_U32( data + 2, foe->rx_expected_packet_no  );
+
+    return 0;
+}
+
+/*****************************************************************************/
+/**
+   State: RRQ SENT.
+   Checks is the previous transmit datagram succeded and sends the next
+   fragment, if necessary.
+*/
+
+void ec_fsm_foe_state_rrq_sent( ec_fsm_foe_t *fsm ) {
+    ec_datagram_t *datagram = fsm->datagram;
+    ec_slave_t *slave = fsm->slave;
+
+#ifdef	myDEBUG
+	printk("ec_foe_state_rrq_sent()\n");
+#endif
+    if (datagram->state != EC_DATAGRAM_RECEIVED) {
+    	ec_foe_set_rx_error(fsm, FOE_RECEIVE_ERROR);
+        EC_ERR("Failed to send FoE RRQ for slave %u"
+                " (datagram state %u).\n",
+                slave->ring_position, datagram->state);
+        return;
+    }
+
+    if (datagram->working_counter != 1) {
+    	// slave hat noch nichts in die Mailbox getan
+        ec_foe_set_rx_error(fsm, FOE_WC_ERROR);
+        EC_ERR("Reception of FoE RRQ failed on slave %u: ",
+                slave->ring_position);
+        ec_datagram_print_wc_error(datagram);
+    	return;
+    }
+
+    fsm->jiffies_start = datagram->jiffies_sent;
+
+    ec_slave_mbox_prepare_check(fsm->slave, datagram); // can not fail.
+
+    fsm->retries = EC_FSM_RETRIES;
+    fsm->state = ec_fsm_foe_state_data_check;
+}
+
+/*****************************************************************************/
+
+#ifdef myDEBUG
+char		rx_buffer[0x8000];
+#endif
+
+void ec_fsm_foe_read(ec_fsm_foe_t *fsm /**< finite state machine */)
+{
+	fsm->state = ec_fsm_foe_read_start;
+	fsm->rx_filename = fsm->request->file_name;
+	fsm->rx_filename_len = strlen(fsm->rx_filename);
+
+	fsm->rx_buffer = fsm->request->buffer;
+	fsm->rx_buffer_size = fsm->request->buffer_size;
+
+#ifdef use_ext_buffer
+	fsm->rx_buffer = rx_buffer;
+	fsm->rx_buffer_size = sizeof(rx_buffer);
+#endif
+}
+
+/*****************************************************************************/
+
+void ec_fsm_foe_read_start(ec_fsm_foe_t *fsm /**< finite state machine */)
+{
+	size_t current_size;
+    ec_slave_t *slave = fsm->slave;
+
+    fsm->rx_buffer_offset = 0;
+    fsm->rx_current_size = 0;
+    fsm->rx_packet_no = 0;
+    fsm->rx_expected_packet_no = 1;
+    fsm->rx_last_packet = 0;
+
+    current_size = fsm->rx_filename_len;
+
+#ifdef	myDEBUG
+	printk("ec_fsm_foe_read_start()\n");
+#endif
+    if (!(slave->sii.mailbox_protocols & EC_MBOX_FOE)) {
+    	ec_foe_set_tx_error(fsm, FOE_MBOX_PROT_ERROR);
+        EC_ERR("Slave %u does not support FoE!\n", slave->ring_position);
+        return;
+    }
+
+    if (ec_foe_prepare_rrq_send(fsm)) {
+        ec_foe_set_rx_error(fsm, FOE_PROT_ERROR);
+        return;
+    }
+
+    fsm->state = ec_fsm_foe_state_rrq_sent;
+}
+
+/*****************************************************************************/
+
+void ec_fsm_foe_state_data_check ( ec_fsm_foe_t *fsm ) {
+    ec_datagram_t *datagram = fsm->datagram;
+    ec_slave_t *slave = fsm->slave;
+
+#ifdef	myDEBUG
+	printk("ec_fsm_foe_state_data_check()\n");
+#endif
+	if (datagram->state != EC_DATAGRAM_RECEIVED) {
+        ec_foe_set_rx_error(fsm, FOE_RECEIVE_ERROR);
+        EC_ERR("Failed to send FoE DATA READ for slave %u"
+                " (datagram state %u).\n",
+                slave->ring_position, datagram->state);
+        return;
+    }
+
+    if (datagram->working_counter != 1) {
+        ec_foe_set_rx_error(fsm, FOE_WC_ERROR);
+        EC_ERR("Reception of FoE DATA READ on slave %u: ",
+                slave->ring_position);
+        ec_datagram_print_wc_error(datagram);
+        return;
+    }
+
+    if (!ec_slave_mbox_check(datagram)) {
+        unsigned long diff_ms =
+            (datagram->jiffies_received - fsm->jiffies_start) * 1000 / HZ;
+        if (diff_ms >= EC_FSM_FOE_TIMEOUT) {
+        	ec_foe_set_tx_error(fsm, FOE_TIMEOUT_ERROR);
+            EC_ERR("Timeout while waiting for ack response "
+                    "on slave %u.\n", slave->ring_position);
+            return;
+        }
+
+        ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
+        fsm->retries = EC_FSM_RETRIES;
+        return;
+    }
+
+    // Fetch response
+    ec_slave_mbox_prepare_fetch(slave, datagram); // can not fail.
+
+    fsm->retries = EC_FSM_RETRIES;
+    fsm->state = ec_fsm_foe_state_data_read;
+
+}
+
+/*****************************************************************************/
+
+void ec_fsm_foe_state_data_read ( ec_fsm_foe_t *fsm ) {
+    size_t 	rec_size;
+    uint8_t *data, opMode, packet_no, mbox_prot;
+
+    ec_datagram_t *datagram = fsm->datagram;
+    ec_slave_t *slave = fsm->slave;
+
+#ifdef	myDEBUG
+	printk("ec_fsm_foe_state_data_read()\n");
+#endif
+    if (datagram->state != EC_DATAGRAM_RECEIVED) {
+    	ec_foe_set_rx_error(fsm, FOE_RECEIVE_ERROR);
+        EC_ERR("Failed to receive FoE DATA READ datagram for"
+               " slave %u (datagram state %u).\n",
+               slave->ring_position, datagram->state);
+        return;
+    }
+
+    if (datagram->working_counter != 1) {
+        ec_foe_set_rx_error(fsm, FOE_WC_ERROR);
+        EC_ERR("Reception of FoE DATA READ failed on slave %u: ",
+                slave->ring_position);
+        ec_datagram_print_wc_error(datagram);
+        return;
+    }
+
+	if (!(data = ec_slave_mbox_fetch(slave, datagram, &mbox_prot, &rec_size))) {
+    	ec_foe_set_rx_error(fsm, FOE_MBOX_FETCH_ERROR);
+        return;
+    }
+
+	if (mbox_prot != EC_MBOX_TYPE_FILEACCESS) { // FoE
+        EC_ERR("Received mailbox protocol 0x%02X as response.\n", mbox_prot);
+        ec_foe_set_rx_error(fsm, FOE_PROT_ERROR);
+        return;
+    }
+
+    opMode = EC_READ_U16(data);
+
+    if (opMode == EC_FOE_OPMODE_BUSY) {
+        if (ec_foe_prepare_send_ack(fsm)) {
+            ec_foe_set_rx_error(fsm, FOE_PROT_ERROR);
+        }
+        return;
+    }
+
+    if (opMode != EC_FOE_OPMODE_DATA) {
+        ec_foe_set_rx_error(fsm, FOE_OPMODE_ERROR);
+        return;
+    }
+
+    packet_no = EC_READ_U16(data + 2);
+    if (packet_no != fsm->rx_expected_packet_no) {
+        ec_foe_set_rx_error(fsm, FOE_PACKETNO_ERROR);
+        return;
+    }
+
+    rec_size -= EC_FOE_HEADER_SIZE;
+
+    if ( fsm->rx_buffer_size >= fsm->rx_buffer_offset + rec_size ) {
+        memcpy ( fsm->rx_buffer + fsm->rx_buffer_offset, data + EC_FOE_HEADER_SIZE, rec_size );
+        fsm->rx_buffer_offset += rec_size;
+    }
+
+    fsm->rx_last_packet = (rec_size + EC_MBOX_HEADER_SIZE + EC_FOE_HEADER_SIZE != fsm->slave->sii.rx_mailbox_size);
+
+    if (fsm->rx_last_packet ||
+    	slave->sii.rx_mailbox_size - EC_MBOX_HEADER_SIZE - EC_FOE_HEADER_SIZE  + fsm->rx_buffer_offset <= fsm->rx_buffer_size) {
+    	// either it was the last packet or a new packet will fit into the delivered buffer
+#ifdef	myDEBUG
+    	printk ("last_packet=true\n");
+#endif
+	    if (ec_foe_prepare_send_ack(fsm)) {
+	    	ec_foe_set_rx_error(fsm, FOE_RX_DATA_ACK_ERROR);
+	    	return;
+	    }
+
+	    fsm->state = ec_fsm_foe_state_sent_ack;
+    }
+    else {
+    	// no more data fits into the deliverd buffer
+    	// ... wait for new read request (an den Treiber)
+    	printk ("ERROR: data doesn't fit in receive buffer\n");
+    	printk ("       rx_buffer_size  = %d\n", fsm->rx_buffer_size);
+    	printk ("       rx_buffer_offset= %d\n", fsm->rx_buffer_offset);
+    	printk ("       rec_size        = %d\n", rec_size);
+    	printk ("       rx_mailbox_size = %d\n", slave->sii.rx_mailbox_size);
+    	printk ("       rx_last_packet  = %d\n", fsm->rx_last_packet);
+//    	fsm->state = ec_fsm_state_wait_next_read;
+    	fsm->request->abort_code = FOE_READY;
+    }
+}
+
+/*****************************************************************************/
+
+void ec_fsm_foe_state_sent_ack( ec_fsm_foe_t *fsm ) {
+
+    ec_datagram_t *datagram = fsm->datagram;
+    ec_slave_t *slave = fsm->slave;
+
+#ifdef	myDEBUG
+	printk("ec_foe_state_sent_ack()\n");
+#endif
+    if (datagram->state != EC_DATAGRAM_RECEIVED) {
+        ec_foe_set_rx_error(fsm, FOE_RECEIVE_ERROR);
+        EC_ERR("Failed to send FoE ACK for slave %u"
+                " (datagram state %u).\n",
+                slave->ring_position, datagram->state);
+        return;
+    }
+
+    if (datagram->working_counter != 1) {
+    	// slave hat noch nichts in die Mailbox getan
+        ec_foe_set_rx_error(fsm, FOE_WC_ERROR);
+        EC_ERR("Reception of FoE ACK failed on slave %u: ",
+                slave->ring_position);
+        ec_datagram_print_wc_error(datagram);
+    	return;
+    }
+
+    fsm->jiffies_start = datagram->jiffies_sent;
+
+    ec_slave_mbox_prepare_check(fsm->slave, datagram); // can not fail.
+
+    if (fsm->rx_last_packet) {
+    	fsm->rx_expected_packet_no = 0;
+    	fsm->request->data_size = fsm->rx_buffer_offset;
+    	fsm->state = ec_fsm_foe_end;
+    }
+    else {
+    	fsm->rx_expected_packet_no++;
+        fsm->retries = EC_FSM_RETRIES;
+        fsm->state = ec_fsm_foe_state_data_check;
+    }
+}
+
+/*****************************************************************************/
+
+void ec_foe_set_tx_error( ec_fsm_foe_t *fsm, uint32_t errorcode ) {
+	fsm->tx_errors++;
+	fsm->request->abort_code = errorcode;
+	fsm->state = ec_fsm_foe_error;
+}
+
+/*****************************************************************************/
+
+void ec_foe_set_rx_error( ec_fsm_foe_t *fsm, uint32_t errorcode ) {
+	fsm->rx_errors++;
+	fsm->request->abort_code = errorcode;
+	fsm->state = ec_fsm_foe_error;
+}
+
+/*****************************************************************************/