IgH EtherCAT Master  1.5.2
fsm_soe.c
Go to the documentation of this file.
1 /******************************************************************************
2  *
3  * $Id$
4  *
5  * Copyright (C) 2006-2008 Florian Pose, Ingenieurgemeinschaft IgH
6  *
7  * This file is part of the IgH EtherCAT Master.
8  *
9  * The IgH EtherCAT Master is free software; you can redistribute it and/or
10  * modify it under the terms of the GNU General Public License version 2, as
11  * published by the Free Software Foundation.
12  *
13  * The IgH EtherCAT Master is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
16  * Public License for more details.
17  *
18  * You should have received a copy of the GNU General Public License along
19  * with the IgH EtherCAT Master; if not, write to the Free Software
20  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
21  *
22  * ---
23  *
24  * The license mentioned above concerns the source code only. Using the
25  * EtherCAT technology and brand is only permitted in compliance with the
26  * industrial property and similar rights of Beckhoff Automation GmbH.
27  *
28  *****************************************************************************/
29 
35 /*****************************************************************************/
36 
37 #include "globals.h"
38 #include "master.h"
39 #include "mailbox.h"
40 #include "fsm_soe.h"
41 
42 /*****************************************************************************/
43 
46 enum {
51 };
52 
55 #define EC_SOE_SIZE 0x04
56 
59 #define EC_SOE_HEADER_SIZE (EC_MBOX_HEADER_SIZE + EC_SOE_SIZE)
60 
63 #define EC_SOE_RESPONSE_TIMEOUT 1000
64 
65 /*****************************************************************************/
66 
72 
78 
81 
82 /*****************************************************************************/
83 
84 extern const ec_code_msg_t soe_error_codes[];
85 
86 /*****************************************************************************/
87 
90 void ec_print_soe_error(const ec_slave_t *slave, uint16_t error_code)
91 {
92  const ec_code_msg_t *error_msg;
93 
94  for (error_msg = soe_error_codes; error_msg->code; error_msg++) {
95  if (error_msg->code == error_code) {
96  EC_SLAVE_ERR(slave, "SoE error 0x%04X: \"%s\".\n",
97  error_msg->code, error_msg->message);
98  return;
99  }
100  }
101 
102  EC_SLAVE_ERR(slave, "Unknown SoE error 0x%04X.\n", error_code);
103 }
104 
105 /*****************************************************************************/
106 
110  ec_fsm_soe_t *fsm
111  )
112 {
113  fsm->state = NULL;
114  fsm->datagram = NULL;
115  fsm->fragment_size = 0;
116 }
117 
118 /*****************************************************************************/
119 
123  ec_fsm_soe_t *fsm
124  )
125 {
126 }
127 
128 /*****************************************************************************/
129 
133  ec_fsm_soe_t *fsm,
134  ec_slave_t *slave,
135  ec_soe_request_t *request
136  )
137 {
138  fsm->slave = slave;
139  fsm->request = request;
140 
141  if (request->dir == EC_DIR_OUTPUT) {
143  } else {
145  }
146 }
147 
148 /*****************************************************************************/
149 
155  ec_fsm_soe_t *fsm,
156  ec_datagram_t *datagram
157  )
158 {
159  if (fsm->state == ec_fsm_soe_end || fsm->state == ec_fsm_soe_error)
160  return 0;
161 
162  fsm->state(fsm, datagram);
163 
164  if (fsm->state == ec_fsm_soe_end || fsm->state == ec_fsm_soe_error) {
165  fsm->datagram = NULL;
166  return 0;
167  }
168 
169  fsm->datagram = datagram;
170  return 1;
171 }
172 
173 /*****************************************************************************/
174 
180 {
181  return fsm->state == ec_fsm_soe_end;
182 }
183 
184 /*****************************************************************************/
185 
189 {
190  ec_soe_request_t *request = fsm->request;
191 
192  EC_SLAVE_ERR(fsm->slave, "");
193 
194  if (request->dir == EC_DIR_OUTPUT) {
195  printk(KERN_CONT "Writing");
196  } else {
197  printk(KERN_CONT "Reading");
198  }
199 
200  printk(KERN_CONT " IDN 0x%04X failed.\n", request->idn);
201 }
202 
203 /******************************************************************************
204  * SoE read state machine
205  *****************************************************************************/
206 
212  ec_fsm_soe_t *fsm,
213  ec_datagram_t *datagram
214  )
215 {
216  uint8_t *data;
217  ec_slave_t *slave = fsm->slave;
218  ec_master_t *master = slave->master;
219  ec_soe_request_t *request = fsm->request;
220 
221  data = ec_slave_mbox_prepare_send(slave, datagram, EC_MBOX_TYPE_SOE,
222  EC_SOE_SIZE);
223  if (IS_ERR(data)) {
224  return PTR_ERR(data);
225  }
226 
227  EC_WRITE_U8(data, OPCODE_READ_REQUEST | (request->drive_no & 0x07) << 5);
228  EC_WRITE_U8(data + 1, 1 << 6); // request value
229  EC_WRITE_U16(data + 2, request->idn);
230 
231  if (master->debug_level) {
232  EC_SLAVE_DBG(slave, 0, "SSC read request:\n");
233  ec_print_data(data, EC_SOE_SIZE);
234  }
235 
236  fsm->request->jiffies_sent = jiffies;
238 
239  return 0;
240 }
241 
242 /*****************************************************************************/
243 
247  ec_fsm_soe_t *fsm,
248  ec_datagram_t *datagram
249  )
250 {
251  ec_slave_t *slave = fsm->slave;
252  ec_soe_request_t *request = fsm->request;
253 
254  EC_SLAVE_DBG(slave, 1, "Reading IDN 0x%04X of drive %u.\n", request->idn,
255  request->drive_no);
256 
257  if (!slave->sii_image) {
258  EC_SLAVE_ERR(slave, "Slave cannot process SoE read request."
259  " SII data not available.\n");
260  fsm->state = ec_fsm_soe_error;
262  return;
263  }
264 
265  if (!(slave->sii_image->sii.mailbox_protocols & EC_MBOX_SOE)) {
266  EC_SLAVE_ERR(slave, "Slave does not support SoE!\n");
267  fsm->state = ec_fsm_soe_error;
269  return;
270  }
271 
272  request->data_size = 0;
273  fsm->retries = EC_FSM_RETRIES;
274 
275  if (ec_fsm_soe_prepare_read(fsm, datagram)) {
276  fsm->state = ec_fsm_soe_error;
278  }
279 }
280 
281 /*****************************************************************************/
282 
286  ec_fsm_soe_t *fsm,
287  ec_datagram_t *datagram
288  )
289 {
290  ec_slave_t *slave = fsm->slave;
291  unsigned long diff_ms;
292 
293  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
294  if (ec_fsm_soe_prepare_read(fsm, datagram)) {
295  fsm->state = ec_fsm_soe_error;
297  }
298  return;
299  }
300 
301  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
302  fsm->state = ec_fsm_soe_error;
303  EC_SLAVE_ERR(slave, "Failed to receive SoE read request: ");
306  return;
307  }
308 
309  diff_ms = (jiffies - fsm->request->jiffies_sent) * 1000 / HZ;
310 
311  if (fsm->datagram->working_counter != 1) {
312  if (!fsm->datagram->working_counter) {
313  if (diff_ms < EC_SOE_RESPONSE_TIMEOUT) {
314  // no response; send request datagram again
315  if (ec_fsm_soe_prepare_read(fsm, datagram)) {
316  fsm->state = ec_fsm_soe_error;
318  }
319  return;
320  }
321  }
322  fsm->state = ec_fsm_soe_error;
323  EC_SLAVE_ERR(slave, "Reception of SoE read request"
324  " failed after %lu ms: ", diff_ms);
327  return;
328  }
329 
330  fsm->jiffies_start = fsm->datagram->jiffies_sent;
331 
332  // mailbox read check is skipped if a read request is already ongoing
333  if (ec_read_mbox_locked(slave)) {
335  // the datagram is not used and marked as invalid
336  datagram->state = EC_DATAGRAM_INVALID;
337  } else {
338  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
339  fsm->retries = EC_FSM_RETRIES;
341  }
342 }
343 
344 /*****************************************************************************/
345 
349  ec_fsm_soe_t *fsm,
350  ec_datagram_t *datagram
351  )
352 {
353  ec_slave_t *slave = fsm->slave;
354 
355  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
356  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
357  return;
358  }
359 
360  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
361  fsm->state = ec_fsm_soe_error;
363  EC_SLAVE_ERR(slave, "Failed to receive SoE mailbox check datagram: ");
366  return;
367  }
368 
369  if (fsm->datagram->working_counter != 1) {
370  fsm->state = ec_fsm_soe_error;
372  EC_SLAVE_ERR(slave, "Reception of SoE mailbox check"
373  " datagram failed: ");
376  return;
377  }
378 
379  if (!ec_slave_mbox_check(fsm->datagram)) {
380  unsigned long diff_ms = 0;
381 
382  // check that data is not already received by another read request
383  if (slave->mbox_soe_data.payload_size > 0) {
386  fsm->state(fsm, datagram);
387  return;
388  }
389 
390  diff_ms = (fsm->datagram->jiffies_received - fsm->jiffies_start) *
391  1000 / HZ;
392 
393  if (diff_ms >= EC_SOE_RESPONSE_TIMEOUT) {
394  fsm->state = ec_fsm_soe_error;
396  EC_SLAVE_ERR(slave, "Timeout after %lu ms while waiting for"
397  " read response.\n", diff_ms);
399  return;
400  }
401 
402  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
403  fsm->retries = EC_FSM_RETRIES;
404  return;
405  }
406 
407  // Fetch response
408  ec_slave_mbox_prepare_fetch(slave, datagram); // can not fail.
409  fsm->retries = EC_FSM_RETRIES;
411 }
412 
413 /*****************************************************************************/
414 
418  ec_fsm_soe_t *fsm,
419  ec_datagram_t *datagram
420  )
421 {
422  ec_slave_t *slave = fsm->slave;
423 
424  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
425  ec_slave_mbox_prepare_fetch(slave, datagram); // can not fail.
426  return;
427  }
428 
429  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
430  fsm->state = ec_fsm_soe_error;
432  EC_SLAVE_ERR(slave, "Failed to receive SoE read response datagram: ");
435  return;
436  }
437 
438  if (fsm->datagram->working_counter != 1) {
439  // only an error if data has not already been read by another read request
440  if (slave->mbox_soe_data.payload_size == 0) {
441  fsm->state = ec_fsm_soe_error;
443  EC_SLAVE_ERR(slave, "Reception of SoE read response failed: ");
446  return;
447  }
448  }
451  fsm->state(fsm, datagram);
452 }
453 
454 
455 /*****************************************************************************/
456 
463  ec_fsm_soe_t *fsm,
464  ec_datagram_t *datagram
465  )
466 {
467  ec_slave_t *slave = fsm->slave;
468  ec_master_t *master = slave->master;
469  uint8_t *data, mbox_prot, header, opcode, incomplete, error_flag,
470  value_included;
471  size_t rec_size, data_size;
472  ec_soe_request_t *req = fsm->request;
473 
474  // process the data available or initiate a new mailbox read check
475  if (slave->mbox_soe_data.payload_size > 0) {
476  slave->mbox_soe_data.payload_size = 0;
477  } else {
478  // initiate a new mailbox read check if required data is not available
479  if (ec_read_mbox_locked(slave)) {
480  // await current read request and mark the datagram as invalid
481  datagram->state = EC_DATAGRAM_INVALID;
482  } else {
483  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
485  }
486  return;
487  }
488 
489  data = ec_slave_mbox_fetch(slave, &slave->mbox_soe_data, &mbox_prot, &rec_size);
490  if (IS_ERR(data)) {
491  fsm->state = ec_fsm_soe_error;
493  return;
494  }
495 
496  if (master->debug_level) {
497  EC_SLAVE_DBG(slave, 0, "SSC read response:\n");
498  ec_print_data(data, rec_size);
499  }
500 
501  if (mbox_prot != EC_MBOX_TYPE_SOE) {
502  fsm->state = ec_fsm_soe_error;
503  EC_SLAVE_ERR(slave, "Received mailbox protocol 0x%02X as response.\n",
504  mbox_prot);
506  return;
507  }
508 
509  if (rec_size < EC_SOE_SIZE) {
510  fsm->state = ec_fsm_soe_error;
511  EC_SLAVE_ERR(slave, "Received currupted SoE read response"
512  " (%zu bytes)!\n", rec_size);
513  ec_print_data(data, rec_size);
515  return;
516  }
517 
518  header = EC_READ_U8(data);
519  opcode = header & 0x7;
520  incomplete = (header >> 3) & 1;
521  error_flag = (header >> 4) & 1;
522 
523  if (opcode != OPCODE_READ_RESPONSE) {
524  EC_SLAVE_ERR(slave, "Received no read response (opcode %x).\n",
525  opcode);
526  ec_print_data(data, rec_size);
528  fsm->state = ec_fsm_soe_error;
529  return;
530  }
531 
532  if (error_flag) {
533  req->error_code = EC_READ_U16(data + rec_size - 2);
534  EC_SLAVE_ERR(slave, "Received error response:\n");
535  ec_print_soe_error(slave, req->error_code);
537  fsm->state = ec_fsm_soe_error;
538  return;
539  } else {
540  req->error_code = 0x0000;
541  }
542 
543  value_included = (EC_READ_U8(data + 1) >> 6) & 1;
544  if (!value_included) {
545  EC_SLAVE_ERR(slave, "No value included!\n");
547  fsm->state = ec_fsm_soe_error;
548  return;
549  }
550 
551  data_size = rec_size - EC_SOE_SIZE;
553  data + EC_SOE_SIZE, data_size)) {
554  fsm->state = ec_fsm_soe_error;
556  return;
557  }
558 
559  if (incomplete) {
560  EC_SLAVE_DBG(slave, 1, "SoE data incomplete. Waiting for fragment"
561  " at offset %zu.\n", req->data_size);
562  fsm->jiffies_start = fsm->datagram->jiffies_sent;
563  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
564  fsm->retries = EC_FSM_RETRIES;
566  } else {
567  if (master->debug_level) {
568  EC_SLAVE_DBG(slave, 0, "IDN data:\n");
569  ec_print_data(req->data, req->data_size);
570  }
571 
572  fsm->state = ec_fsm_soe_end; // success
573  }
574 }
575 
576 /******************************************************************************
577  * SoE write state machine
578  *****************************************************************************/
579 
583  ec_fsm_soe_t *fsm,
584  ec_datagram_t *datagram
585  )
586 {
587  ec_slave_t *slave = fsm->slave;
588  ec_master_t *master = slave->master;
589  ec_soe_request_t *req = fsm->request;
590  uint8_t incomplete, *data;
591  size_t max_fragment_size, remaining_size;
592  uint16_t fragments_left;
593 
594  remaining_size = req->data_size - fsm->offset;
595  max_fragment_size = slave->configured_rx_mailbox_size - EC_SOE_HEADER_SIZE;
596  incomplete = remaining_size > max_fragment_size;
597  fsm->fragment_size = incomplete ? max_fragment_size : remaining_size;
598  fragments_left = remaining_size / fsm->fragment_size - 1;
599  if (remaining_size % fsm->fragment_size) {
600  fragments_left++;
601  }
602 
603  data = ec_slave_mbox_prepare_send(slave, datagram, EC_MBOX_TYPE_SOE,
604  EC_SOE_SIZE + fsm->fragment_size);
605  if (IS_ERR(data)) {
606  fsm->state = ec_fsm_soe_error;
608  return;
609  }
610 
611  EC_WRITE_U8(data, OPCODE_WRITE_REQUEST | incomplete << 3 |
612  (req->drive_no & 0x07) << 5);
613  EC_WRITE_U8(data + 1, 1 << 6); // only value included
614  EC_WRITE_U16(data + 2, incomplete ? fragments_left : req->idn);
615  memcpy(data + EC_SOE_SIZE, req->data + fsm->offset, fsm->fragment_size);
616 
617  if (master->debug_level) {
618  EC_SLAVE_DBG(slave, 0, "SSC write request:\n");
620  }
621 
622  req->jiffies_sent = jiffies;
624 }
625 
626 /*****************************************************************************/
627 
631  ec_fsm_soe_t *fsm,
632  ec_datagram_t *datagram
633  )
634 {
635  ec_slave_t *slave = fsm->slave;
636  ec_soe_request_t *req = fsm->request;
637 
638  EC_SLAVE_DBG(slave, 1, "Writing IDN 0x%04X of drive %u (%zu byte).\n",
639  req->idn, req->drive_no, req->data_size);
640 
641  if (!slave->sii_image) {
642  EC_SLAVE_ERR(slave, "Slave cannot process SoE write request."
643  " SII data not available.\n");
644  fsm->state = ec_fsm_soe_error;
646  return;
647  }
648 
649  if (!(slave->sii_image->sii.mailbox_protocols & EC_MBOX_SOE)) {
650  EC_SLAVE_ERR(slave, "Slave does not support SoE!\n");
651  fsm->state = ec_fsm_soe_error;
653  return;
654  }
655 
657  EC_SLAVE_ERR(slave, "Mailbox size (%u) too small for SoE write.\n",
659  fsm->state = ec_fsm_soe_error;
661  return;
662  }
663 
664  fsm->offset = 0;
665  fsm->retries = EC_FSM_RETRIES;
666  ec_fsm_soe_write_next_fragment(fsm, datagram);
667 }
668 
669 /*****************************************************************************/
670 
674  ec_fsm_soe_t *fsm,
675  ec_datagram_t *datagram
676  )
677 {
678  ec_slave_t *slave = fsm->slave;
679  unsigned long diff_ms;
680 
681  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
682  ec_fsm_soe_write_next_fragment(fsm, datagram);
683  return;
684  }
685 
686  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
687  fsm->state = ec_fsm_soe_error;
688  EC_SLAVE_ERR(slave, "Failed to receive SoE write request: ");
691  return;
692  }
693 
694  diff_ms = (jiffies - fsm->request->jiffies_sent) * 1000 / HZ;
695 
696  if (fsm->datagram->working_counter != 1) {
697  if (!fsm->datagram->working_counter) {
698  if (diff_ms < EC_SOE_RESPONSE_TIMEOUT) {
699  // no response; send request datagram again
700  ec_fsm_soe_write_next_fragment(fsm, datagram);
701  return;
702  }
703  }
704  fsm->state = ec_fsm_soe_error;
705  EC_SLAVE_ERR(slave, "Reception of SoE write request"
706  " failed after %lu ms: ", diff_ms);
709  return;
710  }
711 
712  // fragment successfully sent
713  fsm->offset += fsm->fragment_size;
714 
715  if (fsm->offset < fsm->request->data_size) {
716  // next fragment
717  fsm->retries = EC_FSM_RETRIES;
718  ec_fsm_soe_write_next_fragment(fsm, datagram);
719  } else {
720  // all fragments sent; query response
721  fsm->jiffies_start = fsm->datagram->jiffies_sent;
722 
723  // mailbox read check is skipped if a read request is already ongoing
724  if (ec_read_mbox_locked(slave)) {
726  // the datagram is not used and marked as invalid
727  datagram->state = EC_DATAGRAM_INVALID;
728  } else {
729  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
730  fsm->retries = EC_FSM_RETRIES;
732  }
733  }
734 }
735 
736 /*****************************************************************************/
737 
741  ec_fsm_soe_t *fsm,
742  ec_datagram_t *datagram
743  )
744 {
745  ec_slave_t *slave = fsm->slave;
746 
747  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
748  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
749  return;
750  }
751 
752  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
753  fsm->state = ec_fsm_soe_error;
755  EC_SLAVE_ERR(slave, "Failed to receive SoE write request datagram: ");
758  return;
759  }
760 
761  if (fsm->datagram->working_counter != 1) {
762  fsm->state = ec_fsm_soe_error;
764  EC_SLAVE_ERR(slave, "Reception of SoE write request datagram: ");
767  return;
768  }
769 
770  if (!ec_slave_mbox_check(fsm->datagram)) {
771  unsigned long diff_ms = 0;
772 
773  // check that data is not already received by another read request
774  if (slave->mbox_soe_data.payload_size > 0) {
777  fsm->state(fsm, datagram);
778  return;
779  }
780 
781  diff_ms = (datagram->jiffies_received - fsm->jiffies_start) * 1000 / HZ;
782 
783  if (diff_ms >= EC_SOE_RESPONSE_TIMEOUT) {
784  fsm->state = ec_fsm_soe_error;
786  EC_SLAVE_ERR(slave, "Timeout after %lu ms while waiting"
787  " for write response.\n", diff_ms);
789  return;
790  }
791 
792  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
793  fsm->retries = EC_FSM_RETRIES;
794  return;
795  }
796 
797  // Fetch response
798  ec_slave_mbox_prepare_fetch(slave, datagram); // can not fail.
799  fsm->retries = EC_FSM_RETRIES;
801 }
802 
803 /*****************************************************************************/
804 
808  ec_fsm_soe_t *fsm,
809  ec_datagram_t *datagram
810  )
811 {
812  ec_slave_t *slave = fsm->slave;
813 
814  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
815  ec_slave_mbox_prepare_fetch(slave, datagram); // can not fail.
816  return; // FIXME: request again?
817  }
818 
819  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
820  fsm->state = ec_fsm_soe_error;
822  EC_SLAVE_ERR(slave, "Failed to receive SoE write"
823  " response datagram: ");
826  return;
827  }
828 
829  if (fsm->datagram->working_counter != 1) {
830  // only an error if data has not already been read by another read request
831  if (slave->mbox_soe_data.payload_size == 0) {
832  fsm->state = ec_fsm_soe_error;
834  EC_SLAVE_ERR(slave, "Reception of SoE write response failed: ");
837  return;
838  }
839  }
842  fsm->state(fsm, datagram);
843 }
844 
845 
846 /*****************************************************************************/
847 
854  ec_fsm_soe_t *fsm,
855  ec_datagram_t *datagram
856  )
857 {
858  ec_slave_t *slave = fsm->slave;
859  ec_master_t *master = slave->master;
860  ec_soe_request_t *req = fsm->request;
861  uint8_t *data, mbox_prot, opcode, error_flag;
862  uint16_t idn;
863  size_t rec_size;
864 
865  // process the data available or initiate a new mailbox read check
866  if (slave->mbox_soe_data.payload_size > 0) {
867  slave->mbox_soe_data.payload_size = 0;
868  } else {
869  // initiate a new mailbox read check if required data is not available
870  if (ec_read_mbox_locked(slave)) {
871  // await current read request and mark the datagram as invalid
872  datagram->state = EC_DATAGRAM_INVALID;
873  } else {
874  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
876  }
877  return;
878  }
879 
880  data = ec_slave_mbox_fetch(slave, &slave->mbox_soe_data, &mbox_prot, &rec_size);
881  if (IS_ERR(data)) {
882  fsm->state = ec_fsm_soe_error;
884  return;
885  }
886 
887  if (master->debug_level) {
888  EC_SLAVE_DBG(slave, 0, "SSC write response:\n");
889  ec_print_data(data, rec_size);
890  }
891 
892  if (mbox_prot != EC_MBOX_TYPE_SOE) {
893  fsm->state = ec_fsm_soe_error;
894  EC_SLAVE_ERR(slave, "Received mailbox protocol 0x%02X as response.\n",
895  mbox_prot);
897  return;
898  }
899 
900  if (rec_size < EC_SOE_SIZE) {
901  fsm->state = ec_fsm_soe_error;
902  EC_SLAVE_ERR(slave, "Received corrupted SoE write response"
903  " (%zu bytes)!\n", rec_size);
904  ec_print_data(data, rec_size);
906  return;
907  }
908 
909  opcode = EC_READ_U8(data) & 0x7;
910  if (opcode != OPCODE_WRITE_RESPONSE) {
911  EC_SLAVE_ERR(slave, "Received no write response"
912  " (opcode %x).\n", opcode);
913  ec_print_data(data, rec_size);
915  fsm->state = ec_fsm_soe_error;
916  return;
917  }
918 
919  idn = EC_READ_U16(data + 2);
920  if (idn != req->idn) {
921  EC_SLAVE_ERR(slave, "Received response for"
922  " wrong IDN 0x%04x.\n", idn);
923  ec_print_data(data, rec_size);
925  fsm->state = ec_fsm_soe_error;
926  return;
927  }
928 
929  error_flag = (EC_READ_U8(data) >> 4) & 1;
930  if (error_flag) {
931  if (rec_size < EC_SOE_SIZE + 2) {
932  EC_SLAVE_ERR(slave, "Received corrupted error response"
933  " - error flag set, but received size is %zu.\n",
934  rec_size);
935  } else {
936  req->error_code = EC_READ_U16(data + EC_SOE_SIZE);
937  EC_SLAVE_ERR(slave, "Received error response:\n");
938  ec_print_soe_error(slave, req->error_code);
939  }
940  ec_print_data(data, rec_size);
942  fsm->state = ec_fsm_soe_error;
943  } else {
944  req->error_code = 0x0000;
945  fsm->state = ec_fsm_soe_end; // success
946  }
947 }
948 
949 /*****************************************************************************/
950 
954  ec_fsm_soe_t *fsm,
955  ec_datagram_t *datagram
956  )
957 {
958 }
959 
960 /*****************************************************************************/
961 
965  ec_fsm_soe_t *fsm,
966  ec_datagram_t *datagram
967  )
968 {
969 }
970 
971 /*****************************************************************************/
#define EC_FSM_RETRIES
Number of state machine retries on datagram timeout.
Definition: globals.h:59
Finite state machines for the Sercos over EtherCAT protocol.
Definition: fsm_soe.h:51
uint8_t * ec_slave_mbox_prepare_send(const ec_slave_t *slave, ec_datagram_t *datagram, uint8_t type, size_t size)
Prepares a mailbox-send datagram.
Definition: mailbox.c:51
unsigned long jiffies_sent
Jiffies, when the datagram was sent.
Definition: datagram.h:105
unsigned long jiffies_sent
Jiffies, when the upload/download request was sent.
Definition: soe_request.h:59
size_t fragment_size
Size of the current fragment.
Definition: fsm_soe.h:60
uint8_t * data
Pointer to SDO data.
Definition: soe_request.h:53
#define EC_SLAVE_DBG(slave, level, fmt, args...)
Convenience macro for printing slave-specific debug messages to syslog.
Definition: slave.h:108
Servo-Profile over EtherCAT.
Definition: globals.h:186
void ec_fsm_soe_write_response_data(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: WRITE RESPONSE DATA.
Definition: fsm_soe.c:853
void ec_fsm_soe_end(ec_fsm_soe_t *, ec_datagram_t *)
State: END.
Definition: fsm_soe.c:964
ec_soe_request_t * request
SoE request.
Definition: fsm_soe.h:58
void ec_fsm_soe_error(ec_fsm_soe_t *, ec_datagram_t *)
State: ERROR.
Definition: fsm_soe.c:953
int ec_slave_mbox_prepare_fetch(const ec_slave_t *slave, ec_datagram_t *datagram)
Prepares a datagram to fetch mailbox data.
Definition: mailbox.c:134
uint32_t code
Code.
Definition: globals.h:314
int ec_fsm_soe_prepare_read(ec_fsm_soe_t *fsm, ec_datagram_t *datagram)
Prepare a read operation.
Definition: fsm_soe.c:211
EtherCAT datagram.
Definition: datagram.h:88
#define EC_SOE_HEADER_SIZE
SoE header size.
Definition: fsm_soe.c:59
#define EC_WRITE_U8(DATA, VAL)
Write an 8-bit unsigned value to EtherCAT data.
Definition: ecrt.h:2642
uint8_t * ec_slave_mbox_fetch(const ec_slave_t *slave, ec_mbox_data_t *response_data, uint8_t *type, size_t *size)
Processes received mailbox data.
Definition: mailbox.c:172
uint16_t error_code
SoE error code.
Definition: soe_request.h:61
void ec_fsm_soe_read_response(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: READ RESPONSE.
Definition: fsm_soe.c:417
uint16_t working_counter
Working counter.
Definition: datagram.h:100
uint8_t drive_no
Drive number.
Definition: soe_request.h:50
const char * message
Message belonging to code.
Definition: globals.h:315
void ec_print_soe_error(const ec_slave_t *slave, uint16_t error_code)
Outputs an SoE error code.
Definition: fsm_soe.c:90
void ec_fsm_soe_write_check(ec_fsm_soe_t *, ec_datagram_t *)
CoE state: WRITE CHECK.
Definition: fsm_soe.c:740
Write request.
Definition: fsm_soe.c:49
Global definitions and macros.
void ec_fsm_soe_read_check(ec_fsm_soe_t *, ec_datagram_t *)
CoE state: READ CHECK.
Definition: fsm_soe.c:348
void ec_fsm_soe_read_response_data(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: READ RESPONSE DATA.
Definition: fsm_soe.c:462
EtherCAT master structure.
EtherCAT slave.
Definition: slave.h:214
Write response.
Definition: fsm_soe.c:50
ec_sii_image_t * sii_image
Current complete SII image.
Definition: slave.h:267
Code/Message pair.
Definition: globals.h:313
void ec_fsm_soe_transfer(ec_fsm_soe_t *fsm, ec_slave_t *slave, ec_soe_request_t *request)
Starts to transfer an IDN to/from a slave.
Definition: fsm_soe.c:132
int ec_fsm_soe_success(const ec_fsm_soe_t *fsm)
Returns, if the state machine terminated with success.
Definition: fsm_soe.c:179
Read response.
Definition: fsm_soe.c:48
ec_datagram_state_t state
State.
Definition: datagram.h:101
#define EC_SOE_RESPONSE_TIMEOUT
SoE response timeout [ms].
Definition: fsm_soe.c:63
uint16_t mailbox_protocols
Supported mailbox protocols.
Definition: slave.h:172
size_t data_size
Size of SDO data.
Definition: soe_request.h:55
unsigned int debug_level
Master debug level.
Definition: master.h:310
#define EC_SLAVE_ERR(slave, fmt, args...)
Convenience macro for printing slave-specific errors to syslog.
Definition: slave.h:77
void ec_datagram_print_wc_error(const ec_datagram_t *datagram)
Evaluates the working counter of a single-cast datagram.
Definition: datagram.c:627
void(* state)(ec_fsm_soe_t *, ec_datagram_t *)
CoE state function.
Definition: fsm_soe.h:55
#define EC_WRITE_U16(DATA, VAL)
Write a 16-bit unsigned value to EtherCAT data.
Definition: ecrt.h:2659
void ec_fsm_soe_write_response(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: WRITE RESPONSE.
Definition: fsm_soe.c:807
ec_direction_t dir
Direction.
Definition: soe_request.h:56
ec_master_t * master
Master owning the slave.
Definition: slave.h:216
off_t offset
IDN data offset during fragmented write.
Definition: fsm_soe.h:59
EtherCAT CoE state machines.
int ec_read_mbox_locked(ec_slave_t *slave)
Return the current mailbox lock status and lock it if not locked.
Definition: slave.c:229
unsigned int retries
retries upon datagram timeout
Definition: fsm_soe.h:53
int ec_soe_request_append_data(ec_soe_request_t *req, const uint8_t *source, size_t size)
Copies SoE data from an external source.
Definition: soe_request.c:203
void ec_print_data(const uint8_t *, size_t)
Outputs frame contents for debugging purposes.
Definition: module.c:355
void ec_read_mbox_lock_clear(ec_slave_t *slave)
Clears the mailbox lock.
Definition: slave.c:216
void ec_fsm_soe_read_start(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: READ START.
Definition: fsm_soe.c:246
int ec_slave_mbox_prepare_check(const ec_slave_t *slave, ec_datagram_t *datagram)
Prepares a datagram for checking the mailbox state.
Definition: mailbox.c:103
#define EC_READ_U16(DATA)
Read a 16-bit unsigned value from EtherCAT data.
Definition: ecrt.h:2554
ec_datagram_t * datagram
Datagram used in the previous step.
Definition: fsm_soe.h:56
void ec_fsm_soe_read_request(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: READ REQUEST.
Definition: fsm_soe.c:285
void ec_datagram_print_state(const ec_datagram_t *datagram)
Prints the state of a datagram.
Definition: datagram.c:587
Mailbox functionality.
const ec_code_msg_t soe_error_codes[]
SoE error codes.
Definition: soe_errors.c:43
Timed out (dequeued).
Definition: datagram.h:79
unsigned long jiffies_start
Timestamp.
Definition: fsm_soe.h:57
uint16_t configured_rx_mailbox_size
Configured receive mailbox size.
Definition: slave.h:244
void ec_fsm_soe_init(ec_fsm_soe_t *fsm)
Constructor.
Definition: fsm_soe.c:109
void ec_fsm_soe_clear(ec_fsm_soe_t *fsm)
Destructor.
Definition: fsm_soe.c:122
Read request.
Definition: fsm_soe.c:47
#define EC_READ_U8(DATA)
Read an 8-bit unsigned value from EtherCAT data.
Definition: ecrt.h:2538
ec_sii_t sii
Extracted SII data.
Definition: slave.h:207
uint16_t idn
Sercos ID-Number.
Definition: soe_request.h:51
void ec_fsm_soe_write_next_fragment(ec_fsm_soe_t *fsm, ec_datagram_t *datagram)
Write next fragment.
Definition: fsm_soe.c:582
int ec_fsm_soe_exec(ec_fsm_soe_t *fsm, ec_datagram_t *datagram)
Executes the current state of the state machine.
Definition: fsm_soe.c:154
Unused and should not be queued (dequeued).
Definition: datagram.h:81
void ec_fsm_soe_print_error(ec_fsm_soe_t *fsm)
Output information about a failed SoE transfer.
Definition: fsm_soe.c:188
Values written by the master.
Definition: ecrt.h:438
Received (dequeued).
Definition: datagram.h:78
ec_slave_t * slave
slave the FSM runs on
Definition: fsm_soe.h:52
EtherCAT master.
Definition: master.h:202
#define EC_SOE_SIZE
Size of all SoE headers.
Definition: fsm_soe.c:55
unsigned long jiffies_received
Jiffies, when the datagram was received.
Definition: datagram.h:110
ec_mbox_data_t mbox_soe_data
Received mailbox data for SoE.
Definition: slave.h:293
size_t payload_size
Size of the mailbox response payload data.
Definition: datagram.h:125
int ec_slave_mbox_check(const ec_datagram_t *datagram)
Processes a mailbox state checking datagram.
Definition: mailbox.c:122
void ec_fsm_soe_write_start(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: WRITE START.
Definition: fsm_soe.c:630
Sercos-over-EtherCAT request.
Definition: soe_request.h:48
void ec_fsm_soe_write_request(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: WRITE REQUEST.
Definition: fsm_soe.c:673