//! Demonstrate sorting slaves into multiple slave groups. //! //! This demo is designed to be used with the following slave devices: //! //! - EK1100 (or EK1501 if using fibre) //! - EL2889 (2 bytes of outputs) //! - EL2828 (1 byte of outputs) use env_logger::Env; use ethercrab::{ error::Error, std::{ethercat_now, tx_rx_task}, Client, ClientConfig, PduStorage, SlaveGroup, Timeouts, }; use std::{ sync::Arc, time::{Duration, Instant}, }; use tokio::time::MissedTickBehavior; /// Maximum number of slaves that can be stored. This must be a power of 2 greater than 1. const MAX_SLAVES: usize = 16; /// Maximum PDU data payload size - set this to the max PDI size or higher. const MAX_PDU_DATA: usize = PduStorage::element_size(1100); /// Maximum number of EtherCAT frames that can be in flight at any one time. const MAX_FRAMES: usize = 16; static PDU_STORAGE: PduStorage = PduStorage::new(); #[derive(Default)] struct Groups { /// EL2889 and EK1100/EK1501. For EK1100, 2 items, 2 bytes of PDI for 16 output bits. The EK1501 /// has 2 bytes of its own PDI so we'll use an upper bound of 4. /// /// We'll keep the EK1100/EK1501 in here as it has no useful PDI but still needs to live /// somewhere. slow_outputs: SlaveGroup<2, 4>, /// EL2828. 1 item, 1 byte of PDI for 8 output bits. fast_outputs: SlaveGroup<1, 1>, } #[tokio::main] async fn main() -> Result<(), Error> { 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 slave device, 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 client = Client::new( pdu_loop, Timeouts { wait_loop_delay: Duration::from_millis(2), mailbox_response: Duration::from_millis(1000), ..Default::default() }, ClientConfig::default(), ); // Network TX/RX should run in a separate thread to avoid timeouts. Tokio doesn't guarantee a // separate thread is used but this is good enough for an example. If using `tokio`, make sure // the `rt-multi-thread` feature is enabled. tokio::spawn(tx_rx_task(&interface, tx, rx).expect("spawn TX/RX task")); let client = Arc::new(client); // Read configurations from slave EEPROMs and configure devices. let groups = client .init::(ethercat_now, |groups: &Groups, slave| match slave.name() { "EL2889" | "EK1100" | "EK1501" => Ok(&groups.slow_outputs), "EL2828" => Ok(&groups.fast_outputs), _ => Err(Error::UnknownSlave), }) .await .expect("Init"); let Groups { slow_outputs, fast_outputs, } = groups; let client_slow = client.clone(); let slow_task = tokio::spawn(async move { let slow_outputs = slow_outputs .into_op(&client_slow) .await .expect("PRE-OP -> OP"); let mut slow_cycle_time = tokio::time::interval(Duration::from_millis(3)); slow_cycle_time.set_missed_tick_behavior(MissedTickBehavior::Skip); let slow_duration = Duration::from_millis(250); // Only update "slow" outputs every 250ms using this instant let mut tick = Instant::now(); // EK1100 is first slave, EL2889 is second let mut el2889 = slow_outputs .slave(&client_slow, 1) .expect("EL2889 not present!"); // Set initial output state el2889.io_raw_mut().1[0] = 0x01; el2889.io_raw_mut().1[1] = 0x80; loop { slow_outputs.tx_rx(&client_slow).await.expect("TX/RX"); // Increment every output byte for every slave device by one if tick.elapsed() > slow_duration { tick = Instant::now(); let (_i, o) = el2889.io_raw_mut(); // Make a nice pattern on EL2889 LEDs o[0] = o[0].rotate_left(1); o[1] = o[1].rotate_right(1); } slow_cycle_time.tick().await; } }); let fast_task = tokio::spawn(async move { let mut fast_outputs = fast_outputs.into_op(&client).await.expect("PRE-OP -> OP"); let mut fast_cycle_time = tokio::time::interval(Duration::from_millis(5)); fast_cycle_time.set_missed_tick_behavior(MissedTickBehavior::Skip); loop { fast_outputs.tx_rx(&client).await.expect("TX/RX"); // Increment every output byte for every slave device by one for mut slave in fast_outputs.iter(&client) { let (_i, o) = slave.io_raw_mut(); for byte in o.iter_mut() { *byte = byte.wrapping_add(1); } } fast_cycle_time.tick().await; } }); let (slow, fast) = tokio::join!(slow_task, fast_task); slow.expect("slow task failed"); fast.expect("fast task failed"); Ok(()) }