use env_logger::Env;
use ethercrab::{
MainDevice, MainDeviceConfig, PduStorage, SubDeviceGroup, Timeouts, error::Error,
std::ethercat_now,
};
use futures_lite::StreamExt;
use std::{
sync::Arc,
time::{Duration, Instant},
};
const MAX_SUBDEVICES: usize = 16;
const MAX_PDU_DATA: usize = PduStorage::element_size(1100);
const MAX_FRAMES: usize = 16;
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>,
}
fn main() -> Result<(), Error> {
smol::block_on(async {
env_logger::Builder::from_env(Env::default().default_filter_or("info")).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 maindevice = MainDevice::new(
pdu_loop,
Timeouts {
wait_loop_delay: Duration::from_millis(2),
mailbox_response: Duration::from_millis(1000),
..Default::default()
},
MainDeviceConfig::default(),
);
#[cfg(target_os = "windows")]
std::thread::spawn(move || {
ethercrab::std::tx_rx_task_blocking(
&interface,
tx,
rx,
ethercrab::std::TxRxTaskConfig { spinloop: false },
)
.expect("TX/RX task")
});
#[cfg(not(target_os = "windows"))]
smol::spawn(ethercrab::std::tx_rx_task(&interface, tx, rx).expect("spawn TX/RX task"))
.detach();
let maindevice = Arc::new(maindevice);
let groups = 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),
},
)
.await
.expect("Init");
let Groups {
slow_outputs,
fast_outputs,
} = groups;
let maindevice_slow = maindevice.clone();
let slow_task = async {
let slow_outputs = slow_outputs
.into_op(&maindevice_slow)
.await
.expect("PRE-OP -> OP");
let mut slow_cycle_time = smol::Timer::interval(Duration::from_millis(3));
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 = 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_millis(5));
loop {
let Ok(_) = fast_outputs.tx_rx(&maindevice).await else {
break;
};
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;
}
};
let (_slow, _fast) = smol::future::zip(slow_task, fast_task).await;
Ok(())
})
}