#[cfg(windows)]
#[tokio::main]
async fn main() -> Result<(), ethercrab::error::Error> {
use env_logger::Env;
use ethercrab::{
MainDevice, MainDeviceConfig, PduStorage, Timeouts,
std::{TxRxTaskConfig, ethercat_now, tx_rx_task_blocking},
};
use spin_sleep::{SpinSleeper, SpinStrategy};
use std::{
sync::{
Arc,
atomic::{AtomicBool, Ordering},
},
time::Duration,
};
use thread_priority::{ThreadPriority, ThreadPriorityOsValue, WinAPIThreadPriority};
const MAX_SUBDEVICES: usize = 16;
const MAX_PDU_DATA: usize = PduStorage::element_size(1100);
const MAX_FRAMES: usize = 16;
const PDI_LEN: usize = 64;
static PDU_STORAGE: PduStorage<MAX_FRAMES, MAX_PDU_DATA> = PduStorage::new();
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 EK1100/EK1501 demo...");
log::info!(
"Ensure an EK1100 or EK1501 is the first SubDevice, with any number of modules connected after"
);
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 = Arc::new(MainDevice::new(
pdu_loop,
Timeouts {
wait_loop_delay: Duration::ZERO,
eeprom: Duration::from_millis(50),
..Default::default()
},
MainDeviceConfig {
dc_static_sync_iterations: 1000,
..MainDeviceConfig::default()
},
));
let core_ids = core_affinity::get_core_ids().expect("Get core IDs");
let main_thread_core = core_ids[0];
let tx_rx_core = core_ids[2];
core_affinity::set_for_current(main_thread_core);
let sleeper = SpinSleeper::default().with_spin_strategy(SpinStrategy::SpinLoopHint);
let clock = quanta::Clock::new();
thread_priority::ThreadBuilder::default()
.name("tx-rx-thread")
.priority(ThreadPriority::Os(ThreadPriorityOsValue::from(
WinAPIThreadPriority::TimeCritical,
)))
.spawn(move |_| {
core_affinity::set_for_current(tx_rx_core)
.then_some(())
.expect("Set TX/RX thread core");
tx_rx_task_blocking(&interface, tx, rx, TxRxTaskConfig { spinloop: false })
.expect("TX/RX task");
})
.unwrap();
let mut group = maindevice
.init_single_group::<MAX_SUBDEVICES, PDI_LEN>(ethercat_now)
.await
.expect("Init");
log::info!("Discovered {} SubDevices", group.len());
for subdevice in group.iter(&maindevice) {
if subdevice.name() == "EL3004" {
log::info!("Found EL3004. Configuring...");
subdevice.sdo_write(0x1c12, 0, 0u8).await?;
subdevice
.sdo_write_array(0x1c13, &[0x1a00u16, 0x1a02, 0x1a04, 0x1a06])
.await?;
}
}
let mut group = group.into_op(&maindevice).await.expect("PRE-OP -> OP");
for subdevice in group.iter(&maindevice) {
let io = subdevice.io_raw();
log::info!(
"-> SubDevice {:#06x} {} inputs: {} bytes, outputs: {} bytes",
subdevice.configured_address(),
subdevice.name(),
io.inputs().len(),
io.outputs().len()
);
}
let cycle_time = Duration::from_millis(5);
let shutdown = Arc::new(AtomicBool::new(false));
signal_hook::flag::register(signal_hook::consts::SIGINT, Arc::clone(&shutdown))
.expect("Register hook");
loop {
let now = clock.now();
if shutdown.load(Ordering::Relaxed) {
log::info!("Shutting down...");
break;
}
group.tx_rx(&maindevice).await.expect("TX/RX");
for mut subdevice in group.iter(&maindevice) {
let mut o = subdevice.outputs_raw_mut();
for byte in o.iter_mut() {
*byte = byte.wrapping_add(1);
}
}
let wait = cycle_time.saturating_sub(now.elapsed());
sleeper.sleep(wait);
}
let group = group
.into_safe_op(&maindevice)
.await
.expect("OP -> SAFE-OP");
log::info!("OP -> SAFE-OP");
let group = group
.into_pre_op(&maindevice)
.await
.expect("SAFE-OP -> PRE-OP");
log::info!("SAFE-OP -> PRE-OP");
let _group = group.into_init(&maindevice).await.expect("PRE-OP -> INIT");
log::info!("PRE-OP -> INIT, shutdown complete");
Ok(())
}
#[cfg(not(windows))]
fn main() {
eprintln!(
"Windows-only - the performance changes in this example don't make sense for other OSes"
);
}