EtherCAT has a feature called “distributed clocks” that seems to confuse many people on the LinuxCNC forum. Distributed clocks (when enabled) allow devices to stay synchronized to each other with very small amounts of jitter (typically a few microseconds). This allows LinuxCNC’s motion planner to tell EtherCAT servo and stepper drives where to be at a specific time around 1 millisecond, and have them almost exactly match the plan provided by the software.
EtherCAT is designed for low-latency, low-jitter control of industrial
equipment. To accomplish this, it pays very close attention to timing
and delays across the network. Running ethercat -v slaves
will show
delay times between individual EtherCAT slaves. On my system, typical
delays are around 150ns for EL* devices, 550ns for EP* devices, and
750ns for stepper and servo drives.
Individual devices can use a few different schemes for polling and communicating PDOs across the bus. For devices like digital inputs, exact timing may not matter, while for servo controls precise timing is needed to manage speeds and motion with minimal jitter. At least some servo and stepper drives will not work without distributed clocks. This includes all of the Leadshine and RTelligent drives that I’ve tested.
When running in CSP (cyclic synchronous position) mode, the motor controller expects to receive a new target position every 1 milliscond precisely. LinuxCNC can then send each axis movement plans for 1ms in the future and expect all motors to start their next segment precisely on cue, all in sync with each other.
Each slave in LCEC has its own (optional) distributed clock config, which looks like this:
<slave idx="27" type="generic" vid="00000a88" pid="0a880002" configPdos="true" name="rt-ect60">
<dcConf assignActivate="300" sync0Cycle="*1" sync0Shift="0"/>
....
</slave>
The dcConf
XML tag controls distributed clock configs for this one
slave. It takes 5 parameters, 3 of which are used in this example:
assignActivate
assignActivate
control which DC mode the device runs in, and should
generally be considered a device-specific setting. For new devices,
look in the manufacturer’s ESI file for AssignActivate
and copy the
value from there.
Common values are 300
and 700
. This is a 2 byte value, and the
higher-order byte (the 3
or 7
in these cases).
The low byte is written into 0x0980
and the high byte is written
into 0x0981
. The best documentation for these that I’ve found so
far is
https://www.sanyodenki.com/global/america/file/SANMOTION_R_3E_EtherCAT_Servo_M0011697C.pdf
It says that 0x0981
has 3 defined bits:
0: Active cycle operation (when 1, DC is in use?) 1: SYNC0 is active 2: SYNC1 is active.
So, assignActivate
values of 0x3xx
only enable SYNC0, while
0x7xx
enable SYNC0 and SYNC1.
The low order byte (0x0980
) also has 3 defined bits:
0: SYNC out unit control. 0: master, 1: slave. Should be 0 for EtherCAT? 4: Latch in Unit0 (0: master-controlled, 1: slave-controlled) 5: Latch in Unit1 (0: master-controlled, 1: slave-controlled).
The only values that I’ve seen used for assignActivate
so far are 0
(no DC), 0x300, 0x320, 0x700, and 0x720. Presumably 0x330 and 0x730
exist, and maybe 0x310 and 0x710. Those are the only defined values
for this version of the spec.
sync0Cycle
and sync1Cycle
The EtherCAT distributed clock system has 2 different timing signals
available, sync0
and sync1
. This controls the cycle time for each
signal, in units of 1ns (although EtherCAT itself may round this to
the nearest 10ns). As a shortcut, LCEC will let you say "*1"
to set this to the current cycle time, or "*X"
, where X is an
integer, to set this to an integer multiple of the current cycle time.
I’m not sure if anything other than sync0Cycle="*1"
ever makes sense
if we’re using DC, but it’s not unusual to have sync1Cycle
unset,
set to *1
, or set to a small multiple like *3
to have Sync1 run
every 3rd cycle. Exactly what this means depends on the hardware.
sync0Shift
and sync1Shift
These shift the Sync0 and Sync1 interrupts by a fixed number of ns. Presumably shifting various devices slightly could result in reduced jitter and less contention on the network, although it’s not clear that it really matters to us. Many examples seem to just use 0.
Some devices (like RTelligent stepper drives) only seem to work in
DC mode. To make configuring them less complex, the driver in
lcec_rtec.c
automatically enables DC mode if <dcConf/>
isn’t
provided. Since the DC parameters are largely just derived from
information in each device’s ESI file, this should be able to be done
safely.