IgH EtherCAT Master  1.6.0-rc1
fsm_soe.c
Go to the documentation of this file.
1 /******************************************************************************
2  *
3  * Copyright (C) 2006-2020 Florian Pose, Ingenieurgemeinschaft IgH
4  *
5  * This file is part of the IgH EtherCAT Master.
6  *
7  * The IgH EtherCAT Master is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU General Public License version 2, as
9  * published by the Free Software Foundation.
10  *
11  * The IgH EtherCAT Master is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
14  * Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License along
17  * with the IgH EtherCAT Master; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
19  *
20  * ---
21  *
22  * The license mentioned above concerns the source code only. Using the
23  * EtherCAT technology and brand is only permitted in compliance with the
24  * industrial property and similar rights of Beckhoff Automation GmbH.
25  *
26  *****************************************************************************/
27 
33 /*****************************************************************************/
34 
35 #include "globals.h"
36 #include "master.h"
37 #include "mailbox.h"
38 #include "fsm_soe.h"
39 
40 /*****************************************************************************/
41 
44 enum {
49 };
50 
53 #define EC_SOE_SIZE 0x04
54 
57 #define EC_SOE_HEADER_SIZE (EC_MBOX_HEADER_SIZE + EC_SOE_SIZE)
58 
61 #define EC_SOE_RESPONSE_TIMEOUT 1000
62 
63 /*****************************************************************************/
64 
70 
76 
79 
80 /*****************************************************************************/
81 
82 extern const ec_code_msg_t soe_error_codes[];
83 
84 /*****************************************************************************/
85 
88 void ec_print_soe_error(const ec_slave_t *slave, uint16_t error_code)
89 {
90  const ec_code_msg_t *error_msg;
91 
92  for (error_msg = soe_error_codes; error_msg->code; error_msg++) {
93  if (error_msg->code == error_code) {
94  EC_SLAVE_ERR(slave, "SoE error 0x%04X: \"%s\".\n",
95  error_msg->code, error_msg->message);
96  return;
97  }
98  }
99 
100  EC_SLAVE_ERR(slave, "Unknown SoE error 0x%04X.\n", error_code);
101 }
102 
103 /*****************************************************************************/
104 
108  ec_fsm_soe_t *fsm
109  )
110 {
111  fsm->state = NULL;
112  fsm->datagram = NULL;
113  fsm->fragment_size = 0;
114 }
115 
116 /*****************************************************************************/
117 
121  ec_fsm_soe_t *fsm
122  )
123 {
124 }
125 
126 /*****************************************************************************/
127 
131  ec_fsm_soe_t *fsm,
132  ec_slave_t *slave,
133  ec_soe_request_t *request
134  )
135 {
136  fsm->slave = slave;
137  fsm->request = request;
138 
139  if (request->dir == EC_DIR_OUTPUT) {
141  } else {
143  }
144 }
145 
146 /*****************************************************************************/
147 
153  ec_fsm_soe_t *fsm,
154  ec_datagram_t *datagram
155  )
156 {
157  if (fsm->state == ec_fsm_soe_end || fsm->state == ec_fsm_soe_error)
158  return 0;
159 
160  if (fsm->datagram &&
161  (fsm->datagram->state == EC_DATAGRAM_INIT ||
162  fsm->datagram->state == EC_DATAGRAM_QUEUED ||
163  fsm->datagram->state == EC_DATAGRAM_SENT)) {
164  // datagram not received yet
165  if (datagram != fsm->datagram)
166  datagram->state = EC_DATAGRAM_INVALID;
167  return 1;
168  }
169 
170  fsm->state(fsm, datagram);
171 
172  if (fsm->state == ec_fsm_soe_end || fsm->state == ec_fsm_soe_error) {
173  fsm->datagram = NULL;
174  return 0;
175  }
176 
177  fsm->datagram = datagram;
178  return 1;
179 }
180 
181 /*****************************************************************************/
182 
188 {
189  return fsm->state == ec_fsm_soe_end;
190 }
191 
192 /*****************************************************************************/
193 
197 {
198  ec_soe_request_t *request = fsm->request;
199 
200  EC_SLAVE_ERR(fsm->slave, "");
201 
202  if (request->dir == EC_DIR_OUTPUT) {
203  printk(KERN_CONT "Writing");
204  } else {
205  printk(KERN_CONT "Reading");
206  }
207 
208  printk(KERN_CONT " IDN 0x%04X failed.\n", request->idn);
209 }
210 
211 /******************************************************************************
212  * SoE read state machine
213  *****************************************************************************/
214 
220  ec_fsm_soe_t *fsm,
221  ec_datagram_t *datagram
222  )
223 {
224  uint8_t *data;
225  ec_slave_t *slave = fsm->slave;
226  ec_master_t *master = slave->master;
227  ec_soe_request_t *request = fsm->request;
228 
229  data = ec_slave_mbox_prepare_send(slave, datagram, EC_MBOX_TYPE_SOE,
230  EC_SOE_SIZE);
231  if (IS_ERR(data)) {
232  return PTR_ERR(data);
233  }
234 
235  EC_WRITE_U8(data, OPCODE_READ_REQUEST | (request->drive_no & 0x07) << 5);
236  EC_WRITE_U8(data + 1, 1 << 6); // request value
237  EC_WRITE_U16(data + 2, request->idn);
238 
239  if (master->debug_level) {
240  EC_SLAVE_DBG(slave, 0, "SSC read request:\n");
241  ec_print_data(data, EC_SOE_SIZE);
242  }
243 
244  fsm->request->jiffies_sent = jiffies;
246 
247  return 0;
248 }
249 
250 /*****************************************************************************/
251 
255  ec_fsm_soe_t *fsm,
256  ec_datagram_t *datagram
257  )
258 {
259  ec_slave_t *slave = fsm->slave;
260  ec_soe_request_t *request = fsm->request;
261 
262  EC_SLAVE_DBG(slave, 1, "Reading IDN 0x%04X of drive %u.\n", request->idn,
263  request->drive_no);
264 
265  if (!(slave->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 
623 }
624 
625 /*****************************************************************************/
626 
630  ec_fsm_soe_t *fsm,
631  ec_datagram_t *datagram
632  )
633 {
634  ec_slave_t *slave = fsm->slave;
635  ec_soe_request_t *req = fsm->request;
636 
637  EC_SLAVE_DBG(slave, 1, "Writing IDN 0x%04X of drive %u (%zu byte).\n",
638  req->idn, req->drive_no, req->data_size);
639 
640  if (!(slave->sii.mailbox_protocols & EC_MBOX_SOE)) {
641  EC_SLAVE_ERR(slave, "Slave does not support SoE!\n");
642  fsm->state = ec_fsm_soe_error;
644  return;
645  }
646 
648  EC_SLAVE_ERR(slave, "Mailbox size (%u) too small for SoE write.\n",
650  fsm->state = ec_fsm_soe_error;
652  return;
653  }
654 
655  fsm->offset = 0;
656  fsm->retries = EC_FSM_RETRIES;
657  ec_fsm_soe_write_next_fragment(fsm, datagram);
658  req->jiffies_sent = jiffies;
659 }
660 
661 /*****************************************************************************/
662 
666  ec_fsm_soe_t *fsm,
667  ec_datagram_t *datagram
668  )
669 {
670  ec_slave_t *slave = fsm->slave;
671  unsigned long diff_ms;
672 
673  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
674  ec_fsm_soe_write_next_fragment(fsm, datagram);
675  return;
676  }
677 
678  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
679  fsm->state = ec_fsm_soe_error;
680  EC_SLAVE_ERR(slave, "Failed to receive SoE write request: ");
683  return;
684  }
685 
686  diff_ms = (jiffies - fsm->request->jiffies_sent) * 1000 / HZ;
687 
688  if (fsm->datagram->working_counter != 1) {
689  if (!fsm->datagram->working_counter) {
690  if (diff_ms < EC_SOE_RESPONSE_TIMEOUT) {
691  // no response; send request datagram again
692  ec_fsm_soe_write_next_fragment(fsm, datagram);
693  return;
694  }
695  }
696  fsm->state = ec_fsm_soe_error;
697  EC_SLAVE_ERR(slave, "Reception of SoE write request"
698  " failed after %lu ms: ", diff_ms);
701  return;
702  }
703 
704  // fragment successfully sent
705  fsm->offset += fsm->fragment_size;
706 
707  if (fsm->offset < fsm->request->data_size) {
708  // next fragment
709  fsm->retries = EC_FSM_RETRIES;
710  ec_fsm_soe_write_next_fragment(fsm, datagram);
711  fsm->request->jiffies_sent = jiffies;
712  } else {
713  // all fragments sent; query response
714  fsm->jiffies_start = fsm->datagram->jiffies_sent;
715 
716  // mailbox read check is skipped if a read request is already ongoing
717  if (ec_read_mbox_locked(slave)) {
719  // the datagram is not used and marked as invalid
720  datagram->state = EC_DATAGRAM_INVALID;
721  } else {
722  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
723  fsm->retries = EC_FSM_RETRIES;
725  }
726  }
727 }
728 
729 /*****************************************************************************/
730 
734  ec_fsm_soe_t *fsm,
735  ec_datagram_t *datagram
736  )
737 {
738  ec_slave_t *slave = fsm->slave;
739 
740  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
741  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
742  return;
743  }
744 
745  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
746  fsm->state = ec_fsm_soe_error;
748  EC_SLAVE_ERR(slave, "Failed to receive SoE write request datagram: ");
751  return;
752  }
753 
754  if (fsm->datagram->working_counter != 1) {
755  fsm->state = ec_fsm_soe_error;
757  EC_SLAVE_ERR(slave, "Reception of SoE write request datagram: ");
760  return;
761  }
762 
763  if (!ec_slave_mbox_check(fsm->datagram)) {
764  unsigned long diff_ms = 0;
765 
766  // check that data is not already received by another read request
767  if (slave->mbox_soe_data.payload_size > 0) {
770  fsm->state(fsm, datagram);
771  return;
772  }
773 
774  diff_ms = (datagram->jiffies_received - fsm->jiffies_start) * 1000 / HZ;
775 
776  if (diff_ms >= EC_SOE_RESPONSE_TIMEOUT) {
777  fsm->state = ec_fsm_soe_error;
779  EC_SLAVE_ERR(slave, "Timeout after %lu ms while waiting"
780  " for write response.\n", diff_ms);
782  return;
783  }
784 
785  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
786  fsm->retries = EC_FSM_RETRIES;
787  return;
788  }
789 
790  // Fetch response
791  ec_slave_mbox_prepare_fetch(slave, datagram); // can not fail.
792  fsm->retries = EC_FSM_RETRIES;
794 }
795 
796 /*****************************************************************************/
797 
801  ec_fsm_soe_t *fsm,
802  ec_datagram_t *datagram
803  )
804 {
805  ec_slave_t *slave = fsm->slave;
806 
807  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
808  ec_slave_mbox_prepare_fetch(slave, datagram); // can not fail.
809  return; // FIXME: request again?
810  }
811 
812  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
813  fsm->state = ec_fsm_soe_error;
815  EC_SLAVE_ERR(slave, "Failed to receive SoE write"
816  " response datagram: ");
819  return;
820  }
821 
822  if (fsm->datagram->working_counter != 1) {
823  // only an error if data has not already been read by another read request
824  if (slave->mbox_soe_data.payload_size == 0) {
825  fsm->state = ec_fsm_soe_error;
827  EC_SLAVE_ERR(slave, "Reception of SoE write response failed: ");
830  return;
831  }
832  }
835  fsm->state(fsm, datagram);
836 }
837 
838 
839 /*****************************************************************************/
840 
847  ec_fsm_soe_t *fsm,
848  ec_datagram_t *datagram
849  )
850 {
851  ec_slave_t *slave = fsm->slave;
852  ec_master_t *master = slave->master;
853  ec_soe_request_t *req = fsm->request;
854  uint8_t *data, mbox_prot, opcode, error_flag;
855  uint16_t idn;
856  size_t rec_size;
857 
858  // process the data available or initiate a new mailbox read check
859  if (slave->mbox_soe_data.payload_size > 0) {
860  slave->mbox_soe_data.payload_size = 0;
861  } else {
862  // initiate a new mailbox read check if required data is not available
863  if (ec_read_mbox_locked(slave)) {
864  // await current read request and mark the datagram as invalid
865  datagram->state = EC_DATAGRAM_INVALID;
866  } else {
867  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
869  }
870  return;
871  }
872 
873  data = ec_slave_mbox_fetch(slave, &slave->mbox_soe_data, &mbox_prot, &rec_size);
874  if (IS_ERR(data)) {
875  fsm->state = ec_fsm_soe_error;
877  return;
878  }
879 
880  if (master->debug_level) {
881  EC_SLAVE_DBG(slave, 0, "SSC write response:\n");
882  ec_print_data(data, rec_size);
883  }
884 
885  if (mbox_prot != EC_MBOX_TYPE_SOE) {
886  fsm->state = ec_fsm_soe_error;
887  EC_SLAVE_ERR(slave, "Received mailbox protocol 0x%02X as response.\n",
888  mbox_prot);
890  return;
891  }
892 
893  if (rec_size < EC_SOE_SIZE) {
894  fsm->state = ec_fsm_soe_error;
895  EC_SLAVE_ERR(slave, "Received corrupted SoE write response"
896  " (%zu bytes)!\n", rec_size);
897  ec_print_data(data, rec_size);
899  return;
900  }
901 
902  opcode = EC_READ_U8(data) & 0x7;
903  if (opcode != OPCODE_WRITE_RESPONSE) {
904  EC_SLAVE_ERR(slave, "Received no write response"
905  " (opcode %x).\n", opcode);
906  ec_print_data(data, rec_size);
908  fsm->state = ec_fsm_soe_error;
909  return;
910  }
911 
912  idn = EC_READ_U16(data + 2);
913  if (idn != req->idn) {
914  EC_SLAVE_ERR(slave, "Received response for"
915  " wrong IDN 0x%04x.\n", idn);
916  ec_print_data(data, rec_size);
918  fsm->state = ec_fsm_soe_error;
919  return;
920  }
921 
922  error_flag = (EC_READ_U8(data) >> 4) & 1;
923  if (error_flag) {
924  if (rec_size < EC_SOE_SIZE + 2) {
925  EC_SLAVE_ERR(slave, "Received corrupted error response"
926  " - error flag set, but received size is %zu.\n",
927  rec_size);
928  } else {
929  req->error_code = EC_READ_U16(data + EC_SOE_SIZE);
930  EC_SLAVE_ERR(slave, "Received error response:\n");
931  ec_print_soe_error(slave, req->error_code);
932  }
933  ec_print_data(data, rec_size);
935  fsm->state = ec_fsm_soe_error;
936  } else {
937  req->error_code = 0x0000;
938  fsm->state = ec_fsm_soe_end; // success
939  }
940 }
941 
942 /*****************************************************************************/
943 
947  ec_fsm_soe_t *fsm,
948  ec_datagram_t *datagram
949  )
950 {
951 }
952 
953 /*****************************************************************************/
954 
958  ec_fsm_soe_t *fsm,
959  ec_datagram_t *datagram
960  )
961 {
962 }
963 
964 /*****************************************************************************/
#define EC_FSM_RETRIES
Number of state machine retries on datagram timeout.
Definition: globals.h:47
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
ec_sii_t sii
Extracted SII data.
Definition: slave.h:246
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:107
Servo-Profile over EtherCAT.
Definition: globals.h:154
ec_soe_request_t * request
SoE request.
Definition: fsm_soe.h:58
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:128
uint32_t code
Code.
Definition: globals.h:282
int ec_fsm_soe_prepare_read(ec_fsm_soe_t *fsm, ec_datagram_t *datagram)
Prepare a read operation.
Definition: fsm_soe.c:219
EtherCAT datagram.
Definition: datagram.h:88
#define EC_SOE_HEADER_SIZE
SoE header size.
Definition: fsm_soe.c:57
#define EC_WRITE_U8(DATA, VAL)
Write an 8-bit unsigned value to EtherCAT data.
Definition: ecrt.h:2338
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:166
uint16_t error_code
SoE error code.
Definition: soe_request.h:61
uint16_t working_counter
Working counter.
Definition: datagram.h:100
uint8_t drive_no
Drive number.
Definition: soe_request.h:50
Sent (still in the queue).
Definition: datagram.h:77
const char * message
Message belonging to code.
Definition: globals.h:283
void ec_print_soe_error(const ec_slave_t *slave, uint16_t error_code)
Outputs an SoE error code.
Definition: fsm_soe.c:88
Write request.
Definition: fsm_soe.c:47
Global definitions and macros.
EtherCAT master structure.
Initial state of a new datagram.
Definition: datagram.h:75
EtherCAT slave.
Definition: slave.h:199
void ec_fsm_soe_error(ec_fsm_soe_t *, ec_datagram_t *)
State: ERROR.
Definition: fsm_soe.c:946
Write response.
Definition: fsm_soe.c:48
Code/Message pair.
Definition: globals.h:281
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:130
int ec_fsm_soe_success(const ec_fsm_soe_t *fsm)
Returns, if the state machine terminated with success.
Definition: fsm_soe.c:187
Read response.
Definition: fsm_soe.c:46
ec_datagram_state_t state
State.
Definition: datagram.h:101
void ec_fsm_soe_read_request(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: READ REQUEST.
Definition: fsm_soe.c:285
#define EC_SOE_RESPONSE_TIMEOUT
SoE response timeout [ms].
Definition: fsm_soe.c:61
uint16_t mailbox_protocols
Supported mailbox protocols.
Definition: slave.h:170
size_t data_size
Size of SDO data.
Definition: soe_request.h:55
unsigned int debug_level
Master debug level.
Definition: master.h:285
#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:606
void ec_fsm_soe_write_start(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: WRITE START.
Definition: fsm_soe.c:629
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:2355
ec_direction_t dir
Direction.
Definition: soe_request.h:56
ec_master_t * master
Master owning the slave.
Definition: slave.h:201
void ec_fsm_soe_write_response(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: WRITE RESPONSE.
Definition: fsm_soe.c:800
void ec_fsm_soe_write_check(ec_fsm_soe_t *, ec_datagram_t *)
CoE state: WRITE CHECK.
Definition: fsm_soe.c:733
off_t offset
IDN data offset during fragmented write.
Definition: fsm_soe.h:59
void ec_fsm_soe_end(ec_fsm_soe_t *, ec_datagram_t *)
State: END.
Definition: fsm_soe.c:957
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:203
unsigned int retries
retries upon datagram timeout
Definition: fsm_soe.h:53
void ec_fsm_soe_read_check(ec_fsm_soe_t *, ec_datagram_t *)
CoE state: READ CHECK.
Definition: fsm_soe.c:348
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:345
void ec_read_mbox_lock_clear(ec_slave_t *slave)
Clears the mailbox lock.
Definition: slave.c:190
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:97
void ec_fsm_soe_read_start(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: READ START.
Definition: fsm_soe.c:254
#define EC_READ_U16(DATA)
Read a 16-bit unsigned value from EtherCAT data.
Definition: ecrt.h:2250
ec_datagram_t * datagram
Datagram used in the previous step.
Definition: fsm_soe.h:56
void ec_datagram_print_state(const ec_datagram_t *datagram)
Prints the state of a datagram.
Definition: datagram.c:566
Mailbox functionality.
void ec_fsm_soe_read_response(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: READ RESPONSE.
Definition: fsm_soe.c:417
const ec_code_msg_t soe_error_codes[]
SoE error codes.
Definition: soe_errors.c:43
void ec_fsm_soe_write_response_data(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: WRITE RESPONSE DATA.
Definition: fsm_soe.c:846
void ec_fsm_soe_read_response_data(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: READ RESPONSE DATA.
Definition: fsm_soe.c:462
Queued for sending.
Definition: datagram.h:76
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:220
void ec_fsm_soe_init(ec_fsm_soe_t *fsm)
Constructor.
Definition: fsm_soe.c:107
void ec_fsm_soe_clear(ec_fsm_soe_t *fsm)
Destructor.
Definition: fsm_soe.c:120
void ec_fsm_soe_write_request(ec_fsm_soe_t *, ec_datagram_t *)
SoE state: WRITE REQUEST.
Definition: fsm_soe.c:665
Read request.
Definition: fsm_soe.c:45
#define EC_READ_U8(DATA)
Read an 8-bit unsigned value from EtherCAT data.
Definition: ecrt.h:2234
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:152
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:196
Values written by the master.
Definition: ecrt.h:437
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:189
#define EC_SOE_SIZE
Size of all SoE headers.
Definition: fsm_soe.c:53
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:270
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:116
Sercos-over-EtherCAT request.
Definition: soe_request.h:48