IgH EtherCAT Master  1.5.2
fsm_eoe.c
Go to the documentation of this file.
1 /******************************************************************************
2  *
3  * $Id$
4  *
5  * Copyright (C) 2006-2014 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_eoe.h"
41 
42 /*****************************************************************************/
43 
46 #define EC_EOE_RESPONSE_TIMEOUT 3000 // [ms]
47 
48 /*****************************************************************************/
49 
55 
58 
59 /*****************************************************************************/
60 
64  ec_fsm_eoe_t *fsm
65  )
66 {
67  fsm->slave = NULL;
68  fsm->retries = 0;
69  fsm->state = NULL;
70  fsm->datagram = NULL;
71  fsm->jiffies_start = 0;
72  fsm->request = NULL;
73 }
74 
75 /*****************************************************************************/
76 
80  ec_fsm_eoe_t *fsm
81  )
82 {
83 }
84 
85 /*****************************************************************************/
86 
90  ec_fsm_eoe_t *fsm,
91  ec_slave_t *slave,
92  ec_eoe_request_t *request
93  )
94 {
95  fsm->slave = slave;
96  fsm->request = request;
98 }
99 
100 /*****************************************************************************/
101 
107  ec_fsm_eoe_t *fsm,
108  ec_datagram_t *datagram
109  )
110 {
111  if (fsm->state == ec_fsm_eoe_end || fsm->state == ec_fsm_eoe_error)
112  return 0;
113 
114  fsm->state(fsm, datagram);
115 
116  if (fsm->state == ec_fsm_eoe_end || fsm->state == ec_fsm_eoe_error) {
117  fsm->datagram = NULL;
118  return 0;
119  }
120 
121  fsm->datagram = datagram;
122  return 1;
123 }
124 
125 /*****************************************************************************/
126 
132 {
133  return fsm->state == ec_fsm_eoe_end;
134 }
135 
136 /******************************************************************************
137  * EoE set IP parameter state machine
138  *****************************************************************************/
139 
145  ec_fsm_eoe_t *fsm,
146  ec_datagram_t *datagram
147  )
148 {
149  uint8_t *data, *cur;
150  ec_slave_t *slave = fsm->slave;
151  ec_master_t *master = slave->master;
152  ec_eoe_request_t *req = fsm->request;
153 
154  // Note: based on wireshark packet filter it suggests that the EOE_INIT
155  // information is a fixed size with fixed information positions.
156  // see: packet-ecatmb.h and packet-ecatmb.c
157  // However, TwinCAT 2.1 testing also indicates that if a piece of
158  // information is missing then all subsequent items are ignored
159  // Also, if you want DHCP, then only set the mac address.
160  size_t size = 8 + // header + flags
161  ETH_ALEN + // mac address
162  4 + // ip address
163  4 + // subnet mask
164  4 + // gateway
165  4 + // dns server
166  EC_MAX_HOSTNAME_SIZE; // dns name
167 
168  data = ec_slave_mbox_prepare_send(slave, datagram, EC_MBOX_TYPE_EOE,
169  size);
170  if (IS_ERR(data)) {
171  return PTR_ERR(data);
172  }
173 
174  // zero data
175  memset(data, 0, size);
176 
177  // header
178  EC_WRITE_U8(data, EC_EOE_TYPE_INIT_REQ); // Set IP parameter req.
179  EC_WRITE_U8(data + 1, 0x00); // not used
180  EC_WRITE_U16(data + 2, 0x0000); // not used
181 
182  EC_WRITE_U32(data + 4,
183  ((req->mac_address_included != 0) << 0) |
184  ((req->ip_address_included != 0) << 1) |
185  ((req->subnet_mask_included != 0) << 2) |
186  ((req->gateway_included != 0) << 3) |
187  ((req->dns_included != 0) << 4) |
188  ((req->name_included != 0) << 5)
189  );
190 
191  cur = data + 8;
192 
193  if (req->mac_address_included) {
194  memcpy(cur, req->mac_address, ETH_ALEN);
195  }
196  cur += ETH_ALEN;
197 
198  if (req->ip_address_included) {
199  uint32_t swapped = htonl(req->ip_address);
200  memcpy(cur, &swapped, 4);
201  }
202  cur += 4;
203 
204  if (req->subnet_mask_included) {
205  uint32_t swapped = htonl(req->subnet_mask);
206  memcpy(cur, &swapped, 4);
207  }
208  cur += 4;
209 
210  if (req->gateway_included) {
211  uint32_t swapped = htonl(req->gateway);
212  memcpy(cur, &swapped, 4);
213  }
214  cur += 4;
215 
216  if (req->dns_included) {
217  uint32_t swapped = htonl(req->dns);
218  memcpy(cur, &swapped, 4);
219  }
220  cur += 4;
221 
222  if (req->name_included) {
223  memcpy(cur, req->name, EC_MAX_HOSTNAME_SIZE);
224  }
225  cur += EC_MAX_HOSTNAME_SIZE;
226 
227  if (master->debug_level) {
228  EC_SLAVE_DBG(slave, 0, "Set IP parameter request:\n");
229  ec_print_data(data, cur - data);
230  }
231 
232  fsm->request->jiffies_sent = jiffies;
233 
234  return 0;
235 }
236 
237 /*****************************************************************************/
238 
242  ec_fsm_eoe_t *fsm,
243  ec_datagram_t *datagram
244  )
245 {
246  ec_slave_t *slave = fsm->slave;
247 
248  EC_SLAVE_DBG(slave, 1, "Setting IP parameters.\n");
249 
250  if (!slave->sii_image) {
251  EC_SLAVE_ERR(slave, "Slave not ready to execute EoE FSM\n");
252  fsm->state = ec_fsm_eoe_error;
253  return;
254  }
255 
256  if (!(slave->sii_image->sii.mailbox_protocols & EC_MBOX_EOE)) {
257  EC_SLAVE_ERR(slave, "Slave does not support EoE!\n");
258  fsm->state = ec_fsm_eoe_error;
259  return;
260  }
261 
262  if (ec_fsm_eoe_prepare_set(fsm, datagram)) {
263  fsm->state = ec_fsm_eoe_error;
264  return;
265  }
266 
267  fsm->retries = EC_FSM_RETRIES;
269 }
270 
271 /*****************************************************************************/
272 
276  ec_fsm_eoe_t *fsm,
277  ec_datagram_t *datagram
278  )
279 {
280  ec_slave_t *slave = fsm->slave;
281 
282  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
283  if (ec_fsm_eoe_prepare_set(fsm, datagram)) {
284  fsm->state = ec_fsm_eoe_error;
285  }
286  return;
287  }
288 
289  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
290  fsm->state = ec_fsm_eoe_error;
291  EC_SLAVE_ERR(slave, "Failed to receive EoE set IP parameter"
292  " request: ");
294  return;
295  }
296 
297  if (fsm->datagram->working_counter != 1) {
298  unsigned long diff_ms =
299  (jiffies - fsm->request->jiffies_sent) * 1000 / HZ;
300 
301  if (!fsm->datagram->working_counter) {
302  if (diff_ms < EC_EOE_RESPONSE_TIMEOUT) {
303  // no response; send request datagram again
304  if (ec_fsm_eoe_prepare_set(fsm, datagram)) {
305  fsm->state = ec_fsm_eoe_error;
306  }
307  return;
308  }
309  }
310  fsm->state = ec_fsm_eoe_error;
311  EC_SLAVE_ERR(slave, "Reception of EoE set IP parameter request"
312  " failed after %lu ms: ", diff_ms);
314  return;
315  }
316 
317  fsm->jiffies_start = fsm->datagram->jiffies_sent;
318 
319  // mailbox read check is skipped if a read request is already ongoing
320  if (ec_read_mbox_locked(slave)) {
322  // the datagram is not used and marked as invalid
323  datagram->state = EC_DATAGRAM_INVALID;
324  } else {
325  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
326  fsm->retries = EC_FSM_RETRIES;
328  }
329 }
330 
331 /*****************************************************************************/
332 
336  ec_fsm_eoe_t *fsm,
337  ec_datagram_t *datagram
338  )
339 {
340  ec_slave_t *slave = fsm->slave;
341 
342  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
343  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
344  return;
345  }
346 
347  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
348  fsm->state = ec_fsm_eoe_error;
350  EC_SLAVE_ERR(slave, "Failed to receive EoE mailbox check datagram: ");
352  return;
353  }
354 
355  if (fsm->datagram->working_counter != 1) {
356  fsm->state = ec_fsm_eoe_error;
358  EC_SLAVE_ERR(slave, "Reception of EoE mailbox check"
359  " datagram failed: ");
361  return;
362  }
363 
364  if (!ec_slave_mbox_check(fsm->datagram)) {
365  unsigned long diff_ms;
366 
367  // check that data is not already received by another read request
368  if (slave->mbox_eoe_init_data.payload_size > 0) {
371  fsm->state(fsm, datagram);
372  return;
373  }
374 
375  diff_ms = (fsm->datagram->jiffies_received - fsm->jiffies_start) *
376  1000 / HZ;
377  if (diff_ms >= EC_EOE_RESPONSE_TIMEOUT) {
378  fsm->state = ec_fsm_eoe_error;
380  EC_SLAVE_ERR(slave, "Timeout after %lu ms while waiting for"
381  " set IP parameter response.\n", diff_ms);
382  return;
383  }
384 
385  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
386  fsm->retries = EC_FSM_RETRIES;
387  return;
388  }
389 
390  // fetch response
391  ec_slave_mbox_prepare_fetch(slave, datagram); // can not fail.
392  fsm->retries = EC_FSM_RETRIES;
394 }
395 
396 /*****************************************************************************/
397 
401  ec_fsm_eoe_t *fsm,
402  ec_datagram_t *datagram
403  )
404 {
405  ec_slave_t *slave = fsm->slave;
406 
407  if (fsm->datagram->state == EC_DATAGRAM_TIMED_OUT && fsm->retries--) {
408  ec_slave_mbox_prepare_fetch(slave, datagram); // can not fail.
409  return;
410  }
411 
412  if (fsm->datagram->state != EC_DATAGRAM_RECEIVED) {
413  fsm->state = ec_fsm_eoe_error;
415  EC_SLAVE_ERR(slave, "Failed to receive EoE read response datagram: ");
417  return;
418  }
419 
420  if (fsm->datagram->working_counter != 1) {
421  // only an error if data has not already been read by another read request
422  if (slave->mbox_eoe_init_data.payload_size == 0) {
423  fsm->state = ec_fsm_eoe_error;
425  EC_SLAVE_ERR(slave, "Reception of EoE read response failed: ");
427  return;
428  }
429  }
432  fsm->state(fsm, datagram);
433 }
434 
435 /*****************************************************************************/
436 
440  ec_fsm_eoe_t *fsm,
441  ec_datagram_t *datagram
442  )
443 {
444  ec_slave_t *slave = fsm->slave;
445  ec_master_t *master = slave->master;
446  uint8_t *data, mbox_prot, eoe_type;
447  size_t rec_size;
448  ec_eoe_request_t *req = fsm->request;
449 
450  // process the data available or initiate a new mailbox read check
451  if (slave->mbox_eoe_init_data.payload_size > 0) {
452  slave->mbox_eoe_init_data.payload_size = 0;
453  } else {
454  // initiate a new mailbox read check if required data is not available
455  if (ec_read_mbox_locked(slave)) {
456  // await current read request and mark the datagram as invalid
457  datagram->state = EC_DATAGRAM_INVALID;
458  } else {
459  ec_slave_mbox_prepare_check(slave, datagram); // can not fail.
461  }
462  return;
463  }
464 
465  data = ec_slave_mbox_fetch(slave, &slave->mbox_eoe_init_data, &mbox_prot, &rec_size);
466  if (IS_ERR(data)) {
467  fsm->state = ec_fsm_eoe_error;
468  return;
469  }
470 
471  if (master->debug_level) {
472  EC_SLAVE_DBG(slave, 0, "Set IP parameter response:\n");
473  ec_print_data(data, rec_size);
474  }
475 
476  if (mbox_prot != EC_MBOX_TYPE_EOE) {
477  fsm->state = ec_fsm_eoe_error;
478  EC_SLAVE_ERR(slave, "Received mailbox protocol 0x%02X as response.\n",
479  mbox_prot);
480  return;
481  }
482 
483  if (rec_size < 4) {
484  fsm->state = ec_fsm_eoe_error;
485  EC_SLAVE_ERR(slave, "Received currupted EoE set IP parameter response"
486  " (%zu bytes)!\n", rec_size);
487  ec_print_data(data, rec_size);
488  return;
489  }
490 
491  eoe_type = EC_READ_U8(data) & 0x0F;
492 
493  if (eoe_type != EC_EOE_TYPE_INIT_RES) {
494  EC_SLAVE_ERR(slave, "EoE Init handler received other EoE type response"
495  " (type %x). Dropping.\n", eoe_type);
496  ec_print_data(data, rec_size);
497  fsm->state = ec_fsm_eoe_error;
498  return;
499  }
500 
501  req->result = EC_READ_U16(data + 2);
502 
503  if (req->result) {
504  fsm->state = ec_fsm_eoe_error;
505  EC_SLAVE_DBG(slave, 1, "EoE set IP parameters failed with result code"
506  " 0x%04X.\n", req->result);
507  } else {
508  fsm->state = ec_fsm_eoe_end; // success
509  }
510 }
511 
512 /*****************************************************************************/
513 
517  ec_fsm_eoe_t *fsm,
518  ec_datagram_t *datagram
519  )
520 {
521 }
522 
523 /*****************************************************************************/
524 
528  ec_fsm_eoe_t *fsm,
529  ec_datagram_t *datagram
530  )
531 {
532 }
533 
534 /*****************************************************************************/
#define EC_FSM_RETRIES
Number of state machine retries on datagram timeout.
Definition: globals.h:59
int ec_fsm_eoe_exec(ec_fsm_eoe_t *fsm, ec_datagram_t *datagram)
Executes the current state of the state machine.
Definition: fsm_eoe.c:106
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
#define EC_SLAVE_DBG(slave, level, fmt, args...)
Convenience macro for printing slave-specific debug messages to syslog.
Definition: slave.h:108
ec_mbox_data_t mbox_eoe_init_data
Received mailbox data for EoE, type eoe init reponse.
Definition: slave.h:289
ec_slave_t * slave
slave the FSM runs on
Definition: fsm_eoe.h:52
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
EtherCAT datagram.
Definition: datagram.h:88
#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 working_counter
Working counter.
Definition: datagram.h:100
ec_eoe_request_t * request
EoE request.
Definition: fsm_eoe.h:58
void ec_fsm_eoe_init(ec_fsm_eoe_t *fsm)
Constructor.
Definition: fsm_eoe.c:63
unsigned long jiffies_sent
Jiffies, when the request was sent.
Definition: eoe_request.h:52
#define EC_MAX_HOSTNAME_SIZE
Maximum hostname size.
Definition: globals.h:148
EtherCAT EoE set IP parameter state machines.
void ec_fsm_eoe_set_ip_param(ec_fsm_eoe_t *fsm, ec_slave_t *slave, ec_eoe_request_t *request)
Starts to set the EoE IP partameters of a slave.
Definition: fsm_eoe.c:89
Global definitions and macros.
EtherCAT master structure.
unsigned long jiffies_start
Timestamp.
Definition: fsm_eoe.h:57
EtherCAT slave.
Definition: slave.h:214
void ec_fsm_eoe_set_ip_check(ec_fsm_eoe_t *, ec_datagram_t *)
EoE state: SET IP CHECK.
Definition: fsm_eoe.c:335
ec_sii_image_t * sii_image
Current complete SII image.
Definition: slave.h:267
ec_datagram_state_t state
State.
Definition: datagram.h:101
#define EC_WRITE_U32(DATA, VAL)
Write a 32-bit unsigned value to EtherCAT data.
Definition: ecrt.h:2676
uint16_t mailbox_protocols
Supported mailbox protocols.
Definition: slave.h:172
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 ec_fsm_eoe_set_ip_start(ec_fsm_eoe_t *, ec_datagram_t *)
EoE state: SET IP START.
Definition: fsm_eoe.c:241
#define EC_WRITE_U16(DATA, VAL)
Write a 16-bit unsigned value to EtherCAT data.
Definition: ecrt.h:2659
int ec_fsm_eoe_success(const ec_fsm_eoe_t *fsm)
Returns, if the state machine terminated with success.
Definition: fsm_eoe.c:131
EoE Init, Set IP Parameter Request.
Definition: ethernet.h:56
Ethernet over EtherCAT.
Definition: globals.h:183
ec_master_t * master
Master owning the slave.
Definition: slave.h:216
void(* state)(ec_fsm_eoe_t *, ec_datagram_t *)
EoE state function.
Definition: fsm_eoe.h:55
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
void ec_fsm_eoe_error(ec_fsm_eoe_t *, ec_datagram_t *)
State: ERROR.
Definition: fsm_eoe.c:516
Ethernet-over-EtherCAT set IP parameter request.
Definition: eoe_request.h:49
void ec_print_data(const uint8_t *, size_t)
Outputs frame contents for debugging purposes.
Definition: module.c:355
void ec_fsm_eoe_clear(ec_fsm_eoe_t *fsm)
Destructor.
Definition: fsm_eoe.c:79
void ec_read_mbox_lock_clear(ec_slave_t *slave)
Clears the mailbox lock.
Definition: slave.c:216
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
void ec_fsm_eoe_end(ec_fsm_eoe_t *, ec_datagram_t *)
State: END.
Definition: fsm_eoe.c:527
void ec_datagram_print_state(const ec_datagram_t *datagram)
Prints the state of a datagram.
Definition: datagram.c:587
Mailbox functionality.
#define EC_EOE_RESPONSE_TIMEOUT
Maximum time to wait for a set IP parameter response.
Definition: fsm_eoe.c:46
void ec_fsm_eoe_set_ip_response(ec_fsm_eoe_t *, ec_datagram_t *)
EoE state: SET IP RESPONSE.
Definition: fsm_eoe.c:400
EoE Timestamp Response.
Definition: ethernet.h:55
Timed out (dequeued).
Definition: datagram.h:79
void ec_fsm_eoe_set_ip_response_data(ec_fsm_eoe_t *, ec_datagram_t *)
EoE state: SET IP RESPONSE DATA.
Definition: fsm_eoe.c:439
#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
Unused and should not be queued (dequeued).
Definition: datagram.h:81
unsigned int retries
retries upon datagram timeout
Definition: fsm_eoe.h:53
Received (dequeued).
Definition: datagram.h:78
int ec_fsm_eoe_prepare_set(ec_fsm_eoe_t *fsm, ec_datagram_t *datagram)
Prepare a set IP parameters operation.
Definition: fsm_eoe.c:144
EtherCAT master.
Definition: master.h:202
unsigned long jiffies_received
Jiffies, when the datagram was received.
Definition: datagram.h:110
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_eoe_set_ip_request(ec_fsm_eoe_t *, ec_datagram_t *)
EoE state: SET IP REQUEST.
Definition: fsm_eoe.c:275
Finite state machines for the Ethernet over EtherCAT protocol.
Definition: fsm_eoe.h:51
ec_datagram_t * datagram
Datagram used in the previous step.
Definition: fsm_eoe.h:56