EtherCrab

Build Status Crates.io Docs.rs Matrix chat

An EtherCAT master written in pure Rust.

Crate features

For no_std targets, it is recommended to add this crate with

bash cargo add --no-default-features --features defmt

Examples

This example increments the output bytes of all detected slaves every tick. It is tested on an EK1100 with output modules but may work on other basic slave devices.

Run with e.g.

Linux

bash RUST_LOG=debug cargo run --example ek1100 --release -- eth0

Windows

ps $env:RUST_LOG="debug" ; cargo run --example ek1100 --release -- '\Device\NPF_{FF0ACEE6-E8CD-48D5-A399-619CD2340465}'

```rust use envlogger::Env; use ethercrab::{ error::Error, std::txrxtask, Client, ClientConfig, PduStorage, SlaveGroup, Timeouts, slavegroup }; use std::{sync::Arc, time::Duration}; use tokio::time::MissedTickBehavior;

/// Maximum number of slaves that can be stored. This must be a power of 2 greater than 1. const MAXSLAVES: usize = 16; /// Maximum PDU data payload size - set this to the max PDI size or higher. const MAXPDUDATA: usize = 1100; /// Maximum number of EtherCAT frames that can be in flight at any one time. const MAXFRAMES: usize = 16; /// Maximum total PDI length. const PDI_LEN: usize = 64;

static PDU_STORAGE: PduStorage = PduStorage::new();

[tokio::main]

async fn main() -> Result<(), Error> { envlogger::Builder::fromenv(Env::default().defaultfilteror("info")).init();

let interface = std::env::args()
    .nth(1)
    .expect("Provide network interface as first argument.");

log::info!("Starting EK1100 demo...");
log::info!("Ensure an EK1100 is the first slave, 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 client = Arc::new(Client::new(
    pdu_loop,
    Timeouts {
        wait_loop_delay: Duration::from_millis(2),
        mailbox_response: Duration::from_millis(1000),
        ..Default::default()
    },
    ClientConfig::default(),
));

tokio::spawn(tx_rx_task(&interface, tx, rx).expect("spawn TX/RX task"));

let mut group = client
    .init_single_group::<MAX_SLAVES, PDI_LEN>()
    .await
    .expect("Init");

log::info!("Discovered {} slaves", group.len());

for slave in group.iter(&client) {
    // Special case: if an EL3004 module is discovered, it needs some specific config during
    // init to function properly
    if slave.name() == "EL3004" {
        log::info!("Found EL3004. Configuring...");

        slave.sdo_write(0x1c12, 0, 0u8).await?;
        slave.sdo_write(0x1c13, 0, 0u8).await?;

        slave.sdo_write(0x1c13, 1, 0x1a00u16).await?;
        slave.sdo_write(0x1c13, 2, 0x1a02u16).await?;
        slave.sdo_write(0x1c13, 3, 0x1a04u16).await?;
        slave.sdo_write(0x1c13, 4, 0x1a06u16).await?;
        slave.sdo_write(0x1c13, 0, 4u8).await?;
    }
}

let mut group = group.into_op(&client).await.expect("PRE-OP -> OP");

for slave in group.iter(&client) {
    let (i, o) = slave.io_raw();

    log::info!(
        "-> Slave {:#06x} {} inputs: {} bytes, outputs: {} bytes",
        slave.configured_address(),
        slave.name(),
        i.len(),
        o.len()
    );
}

let mut tick_interval = tokio::time::interval(Duration::from_millis(5));
tick_interval.set_missed_tick_behavior(MissedTickBehavior::Skip);

loop {
    group.tx_rx(&client).await.expect("TX/RX");

    // Increment every output byte for every slave device by one
    for slave in group.iter(&client) {
        let (_i, o) = slave.io_raw();

        for byte in o.iter_mut() {
            *byte = byte.wrapping_add(1);
        }
    }

    tick_interval.tick().await;
}

} ```

Community

We're on Matrix!

Current and future features

Profiling

To profile an example:

```bash cargo build --example --profile profiling

Might need sudo sysctl kernel.perfeventparanoid=-1

Might need sudo sysctl kernel.perfeventmlock_kb=2048

sudo setcap capnetraw=pe ./target/profiling/examples/ sudo perf record ./target/profiling/examples/

Ctrl + C when you're done

sudo chown $USER perf.data samply load perf.data ```

Sponsors

GitHub Sponsors

Thank you to everyone who has donated test equipment, time or money to the EtherCrab project! Would you like to be in this list? Then please consider becoming a Github sponsor!

License

Licensed under either of

at your option.