#[cfg(not(target_os = "linux"))]
fn main() {
eprintln!("This example is only supported on Linux systems");
}
#[cfg(target_os = "linux")]
fn main() -> Result<(), ethercrab::error::Error> {
use env_logger::{Env, TimestampPrecision};
use ethercrab::{
MainDevice, MainDeviceConfig, PduStorage, SubDeviceGroup, Timeouts,
error::Error,
std::{ethercat_now, tx_rx_task_io_uring},
};
use futures_lite::StreamExt;
use std::{
sync::Arc,
time::{Duration, Instant},
};
use thread_priority::{
RealtimeThreadSchedulePolicy, ThreadPriority, ThreadPriorityValue, ThreadSchedulePolicy,
};
const MAX_SUBDEVICES: usize = 16;
const MAX_PDU_DATA: usize = PduStorage::element_size(1100);
const MAX_FRAMES: usize = 16;
const INTERVAL: u64 = 100;
static PDU_STORAGE: PduStorage<MAX_FRAMES, MAX_PDU_DATA> = PduStorage::new();
#[derive(Default)]
struct Groups {
slow_outputs: SubDeviceGroup<2, 4>,
fast_outputs: SubDeviceGroup<1, 1>,
}
env_logger::Builder::from_env(Env::default().default_filter_or("info"))
.format_timestamp(Some(TimestampPrecision::Nanos))
.init();
let interface = std::env::args()
.nth(1)
.expect("Provide network interface as first argument.");
log::info!("Starting multiple groups demo...");
log::info!(
"Ensure an EK1100 or EK1501 is the first SubDevice, with an EL2828 and EL2889 following it"
);
log::info!("Run with RUST_LOG=ethercrab=debug or =trace for debug information");
let (tx, rx, pdu_loop) = PDU_STORAGE.try_split().expect("can only split once");
let core_ids = core_affinity::get_core_ids().expect("Couldn't get core IDs");
let tx_rx_core = core_ids
.first()
.copied()
.expect("At least one core is required. Are you running on a potato?");
thread_priority::ThreadBuilder::default()
.name("tx-rx-thread")
.priority(ThreadPriority::Crossplatform(
ThreadPriorityValue::try_from(49u8).unwrap(),
))
.policy(ThreadSchedulePolicy::Realtime(
RealtimeThreadSchedulePolicy::Fifo,
))
.spawn(move |_| {
core_affinity::set_for_current(tx_rx_core)
.then_some(())
.expect("Set TX/RX thread core");
tx_rx_task_io_uring(&interface, tx, rx).expect("TX/RX task");
})
.unwrap();
let maindevice = MainDevice::new(
pdu_loop,
Timeouts {
pdu: Duration::from_millis(1000),
..Timeouts::default()
},
MainDeviceConfig::default(),
);
let maindevice = Arc::new(maindevice);
let groups = smol::block_on(maindevice.init::<MAX_SUBDEVICES, _>(
ethercat_now,
Groups::default(),
|groups: &Groups, subdevice| match subdevice.name() {
"EL2889" | "EK1100" | "EK1501" => Ok(&groups.slow_outputs),
"EL2828" => Ok(&groups.fast_outputs),
_ => Err(Error::UnknownSubDevice),
},
))
.expect("Init");
let Groups {
slow_outputs,
fast_outputs,
} = groups;
let maindevice_slow = maindevice.clone();
let slow_task = smol::spawn(async move {
let slow_outputs = slow_outputs
.into_op(&maindevice_slow)
.await
.expect("PRE-OP -> OP");
let mut slow_cycle_time = smol::Timer::interval(Duration::from_micros(INTERVAL));
let slow_duration = Duration::from_millis(250);
let mut tick = Instant::now();
let el2889 = slow_outputs
.subdevice(&maindevice_slow, 1)
.expect("EL2889 not present!");
el2889.outputs_raw_mut()[0] = 0x01;
el2889.outputs_raw_mut()[1] = 0x80;
loop {
let Ok(_) = slow_outputs.tx_rx(&maindevice_slow).await else {
break;
};
if tick.elapsed() > slow_duration {
tick = Instant::now();
let el2889 = slow_outputs
.subdevice(&maindevice_slow, 1)
.expect("EL2889 not present!");
let mut o = el2889.outputs_raw_mut();
o[0] = o[0].rotate_left(1);
o[1] = o[1].rotate_right(1);
}
slow_cycle_time.next().await;
}
});
let fast_task = smol::spawn(async move {
let fast_outputs = fast_outputs
.into_op(&maindevice)
.await
.expect("PRE-OP -> OP");
let mut fast_cycle_time = smol::Timer::interval(Duration::from_micros(INTERVAL));
loop {
fast_outputs.tx_rx(&maindevice).await.expect("TX/RX");
for subdevice in fast_outputs.iter(&maindevice) {
let mut o = subdevice.outputs_raw_mut();
for byte in o.iter_mut() {
*byte = byte.wrapping_add(1);
}
}
fast_cycle_time.next().await;
}
});
smol::block_on(smol::future::race(slow_task, fast_task));
Ok(())
}