Make Rust Embedded simpler
- stm32f100xx âś”
- stm32f101xx âś”
- stm32f102xx âś”
- stm32f103xx âś”
- stm32f105xx âś”
- stm32f107xx âś”
- more support comming soon
uart example
Cargo.toml file :
embassy-stm32-plus = { git = "https://github.com/lifeRobot/embassy-stm32-plus", features = ["stm32f103rc"] }
embassy-executor = { version = "0.6.3", features = ["arch-cortex-m", "executor-thread"] }
defmt-rtt = "0.4.1"
cortex-m-rt = "0.7.3"
panic-probe = { version = "0.3.2", features = ["print-defmt"] }
main.rs file :
#![no_std]
#![no_main]
use embassy_executor::Spawner;
use embassy_stm32_plus::builder::uart::uart1::Uart1Builder;
use embassy_stm32_plus::builder::uart::uart1::rx::Uart1Rx;
use embassy_stm32_plus::builder::uart::uart1::tx::Uart1Tx;
use embassy_stm32_plus::embassy_stm32;
use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::main]
async fn main(_spawner: Spawner) {
let p = embassy_stm32::init(Default::default());
// just write
/*let mut tx = Uart1TxBuilder::new(p.USART1, Uart1Tx::PA9(p.PA9))
.build_write(p.DMA1_CH4).unwrap();
tx.write(b"hello world").await.unwrap();*/
// just read
/*let mut rx = Uart1RxBuilder::new(p.USART1, Uart1Rx::PA10(p.PA10))
.build_read(p.DMA1_CH5).unwrap();
let mut bytes = [0; 1024];
rx.read(&mut bytes).await.unwrap();*/
// read and write
let mut uart = Uart1Builder::new(p.USART1, Uart1Tx::PA9(p.PA9), Uart1Rx::PA10(p.PA10))
.build_all(p.DMA1_CH4, p.DMA1_CH5).unwrap();
uart.write(b"hello world").await.unwrap();
let mut bytes = [0; 1024];
uart.read(&mut bytes).await.unwrap();
}
gpio example
Cargo.toml:
embassy-stm32-plus = { version = "0.1.4", features = ["stm32f103rc"] }
embassy-executor = { version = "0.6.0", features = ["arch-cortex-m", "executor-thread", "defmt", "integrated-timers"] }
defmt-rtt = "0.4.1"
cortex-m-rt = "0.7.3"
panic-probe = { version = "0.3.2", features = ["print-defmt"] }
#![no_std]
#![no_main]
use embassy_executor::Spawner;
use embassy_stm32_plus::embassy_stm32;
use embassy_stm32_plus::embassy_time::Timer;
use embassy_stm32_plus::traits::gpio::output::GpioOutput;
use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::main]
async fn main(_spawner: Spawner) {
// init stm32, get Peripheral
let p = embassy_stm32::init(Default::default());
// simple get output/input gpio
let mut led = p.PA8.output();
// change gpio level
loop {
led.set_high();
Timer::after_millis(300).await;
led.set_low();
Timer::after_millis(300).await;
}
}
usb example
Cargo.toml file:
embassy-stm32-plus = { version = "0.1.4", features = ["stm32f103rc"] }
embassy-executor = { version = "0.6.0", features = ["arch-cortex-m", "executor-thread", "defmt", "integrated-timers"] }
embassy-futures = { version = "0.1.1" }
defmt = "0.3.8"
defmt-rtt = "0.4.1"
cortex-m = { version = "0.7.7", features = ["inline-asm", "critical-section-single-core"] }
cortex-m-rt = "0.7.3"
panic-probe = { version = "0.3.2", features = ["print-defmt"] }
main.rs file:
#![no_std]
#![no_main]
use embassy_executor::Spawner;
use embassy_stm32_plus::embassy_stm32;
use embassy_stm32_plus::embassy_stm32::peripherals::USB;
use embassy_stm32_plus::embassy_stm32::usb::Driver;
use embassy_stm32_plus::embassy_usb::class::cdc_acm::CdcAcmClass;
use embassy_stm32_plus::embassy_usb::Config;
use embassy_stm32_plus::embassy_usb::driver::EndpointError;
use embassy_stm32_plus::traits::usb::acm_state::AcmState;
use embassy_stm32_plus::traits::usb::buf::UsbBuf;
use embassy_stm32_plus::traits::usb::UsbTrait;
use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::main]
async fn main(_spawner: Spawner) {
// rcc setting or etc., more see https://github.com/embassy-rs/embassy/blob/main/examples/stm32f3/src/bin/usb_serial.rs
let p = embassy_stm32::init(Default::default());
// build default usb device
let mut usb_buf = UsbBuf::default();
let mut acm_state = AcmState::default();
let (mut class, mut usb) = p.USB.build_cdc_acm(p.PA12, p.PA11, &mut usb_buf, &mut acm_state, Config::new(0xc0de, 0xcafe));
// usb business
let echo_fut = async {
loop {
class.wait_connection().await;
defmt::info!("Connected");
let _ = echo(&mut class).await;
defmt::info!("Disconnected");
}
};
// wait usb business
embassy_futures::join::join(echo_fut, usb.run()).await;
}
async fn echo<'d>(class: &mut CdcAcmClass<'d, Driver<'d, USB>>) -> Result<(), EndpointError> {
let mut buf = [0; 64];
loop {
let n = class.read_packet(&mut buf).await?;
let data = &buf[..n];
defmt::info!("data: {:x}", data);
class.write_packet(data).await?;
}
}
usb otg example
Cargo.toml file:
embassy-stm32-plus = { version = "0.1.4", features = ["stm32f105vc"] }
embassy-executor = { version = "0.6.0", features = ["arch-cortex-m", "executor-thread", "defmt", "integrated-timers"] }
embassy-futures = { version = "0.1.1" }
defmt = "0.3.8"
defmt-rtt = "0.4.1"
cortex-m = { version = "0.7.7", features = ["inline-asm", "critical-section-single-core"] }
cortex-m-rt = "0.7.3"
panic-probe = { version = "0.3.2", features = ["print-defmt"] }
main.rs file:
#![no_std]
#![no_main]
use embassy_executor::Spawner;
use embassy_stm32_plus::embassy_stm32;
use embassy_stm32_plus::embassy_stm32::peripherals::USB_OTG_FS;
use embassy_stm32_plus::embassy_stm32::usb_otg::Driver;
use embassy_stm32_plus::embassy_usb::class::cdc_acm::CdcAcmClass;
use embassy_stm32_plus::embassy_usb::Config;
use embassy_stm32_plus::embassy_usb::driver::EndpointError;
use embassy_stm32_plus::traits::usb::acm_state::AcmState;
use embassy_stm32_plus::traits::usb::buf::UsbBuf;
use embassy_stm32_plus::traits::usb::otg::UsbOtgTrait;
use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::main]
async fn main(_spawner: Spawner) {
// rcc setting or etc., more see https://github.com/embassy-rs/embassy/blob/main/examples/stm32f4/src/bin/usb_serial.rs
let p = embassy_stm32::init(Default::default());
// build default usb device
let mut buffer = [0; 256];
let mut usb_buf = UsbBuf::default();
let mut state = AcmState::default();
let (mut class, mut usb) = p.USB_OTG_FS.build_cdc_acm(p.PA12, p.PA11, &mut buffer, &mut usb_buf, &mut state, Config::new(0xc0de, 0xcafe));
// usb business
let echo_fut = async {
loop {
class.wait_connection().await;
defmt::info!("Connected");
let _ = echo(&mut class).await;
defmt::info!("Disconnected");
}
};
// wait usb business
embassy_futures::join::join(echo_fut, usb.run()).await;
}
async fn echo<'d>(class: &mut CdcAcmClass<'d, Driver<'d, USB_OTG_FS>>) -> Result<(), EndpointError> {
let mut buf = [0; 64];
loop {
let n = class.read_packet(&mut buf).await?;
let data = &buf[..n];
defmt::info!("data: {:x}", data);
class.write_packet(data).await?;
}
}
eth example (stm32f107xx)
Tips: currently, only stm32f107xx in the cargo crate supports ethCargo.toml file:
embassy-stm32-plus = { version = "0.1.4", features = ["stm32f107vc"] }
embassy-executor = { version = "0.6.0", features = ["arch-cortex-m", "executor-thread", "defmt", "integrated-timers"] }
embassy-net = { version = "0.5.0", features = ["dhcpv4", "tcp"] }
embedded-io-async = "0.6.1"
static_cell = "2.1.0"
defmt = "0.3.8"
defmt-rtt = "0.4.1"
cortex-m-rt = "0.7.3"
panic-probe = { version = "0.3.2", features = ["print-defmt"] }
main.rs file:
#![no_std]
#![no_main]
use embassy_executor::Spawner;
use embassy_net::{Ipv4Address, StackResources};
use embassy_net::tcp::TcpSocket;
use embassy_stm32_plus::{embassy_stm32, embassy_time};
use embassy_stm32_plus::embassy_stm32::eth::generic_smi::GenericSMI;
use embassy_stm32_plus::embassy_stm32::eth::{Ethernet, PacketQueue};
use embassy_stm32_plus::embassy_stm32::peripherals::ETH;
use embassy_stm32_plus::embassy_time::Timer;
use embassy_stm32_plus::traits::eth::Eth1;
use embedded_io_async::Write;
use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::main]
async fn main(spawner: Spawner) {
// rcc setting or etc., more see https://github.com/embassy-rs/embassy/blob/main/examples/stm32f4/src/bin/eth.rs
let p = embassy_stm32::init(Default::default());
// simple build eth
static PACKETS: StaticCell<PacketQueue<4, 4>> = StaticCell::new();
let mac_addr = [0x00, 0x00, 0xDE, 0xAD, 0xBE, 0xEF];
let eth = p.ETH.eth1(&p, PACKETS.init(PacketQueue::<4, 4>::new()), GenericSMI::new(0), mac_addr);
// Init network stack, copy from https://github.com/embassy-rs/embassy/blob/main/examples/stm32f4/src/bin/eth.rs
let config = embassy_net::Config::dhcpv4(Default::default());
static RESOURCES: StaticCell<StackResources<3>> = StaticCell::new();
// stm32f107 not support rng, random seed set default 0
let (stack, runner) = embassy_net::new(eth, config, RESOURCES.init(StackResources::new()), 0);
defmt::unwrap!(spawner.spawn(net_task(runner)));
stack.wait_config_up().await;
defmt::info!("Network task initialized");
// Then we can use it!
let mut rx_buffer = [0; 4096];
let mut tx_buffer = [0; 4096];
loop {
let mut socket = TcpSocket::new(stack, &mut rx_buffer, &mut tx_buffer);
socket.set_timeout(Some(embassy_time::Duration::from_secs(10)));
let remote_endpoint = (Ipv4Address::new(10, 42, 0, 1), 8000);
defmt::info!("connecting...");
let r = socket.connect(remote_endpoint).await;
if let Err(e) = r {
defmt::info!("connect error: {:?}", e);
Timer::after_secs(1).await;
continue;
}
defmt::info!("connected!");
let buf = [0; 1024];
loop {
let r = socket.write_all(&buf).await;
if let Err(e) = r {
defmt::info!("write error: {:?}", e);
break;
}
Timer::after_secs(1).await;
}
}
}
#[embassy_executor::task]
async fn net_task(mut runner: embassy_net::Runner<'static, Ethernet<'static, ETH, GenericSMI>>) -> ! {
runner.run().await
}
eth w5500 example
Cargo.toml file:
embassy-stm32-plus = { git = "https://github.com/lifeRobot/embassy-stm32-plus", features = ["stm32f107vc", "exti"] }
embassy-executor = { version = "0.6.0", features = ["arch-cortex-m", "executor-thread", "defmt", "integrated-timers"] }
embassy-net = { version = "0.5.0", features = ["dhcpv4", "tcp"] }
embassy-net-wiznet = "0.1.0"
embedded-hal-bus = { version = "0.2.0", features = ["async"] }
embedded-io-async = "0.6.1"
static_cell = "2.1.0"
defmt = "0.3.10"
defmt-rtt = "0.4.1"
cortex-m-rt = "0.7.3"
panic-probe = { version = "0.3.2", features = ["print-defmt"] }
#![no_std]
#![no_main]
use embassy_executor::Spawner;
use embassy_net::{Ipv4Address, StackResources};
use embassy_net::tcp::TcpSocket;
use embassy_net_wiznet::{Device, Runner, State};
use embassy_net_wiznet::chip::W5500;
use embassy_stm32_plus::embassy_stm32;
use embassy_stm32_plus::embassy_stm32::exti::ExtiInput;
use embassy_stm32_plus::embassy_stm32::gpio::{Level, Output, Speed};
use embassy_stm32_plus::embassy_stm32::peripherals::{DMA1_CH2, DMA1_CH3, PA4, PB0, PB1, SPI1};
use embassy_stm32_plus::embassy_stm32::spi::Spi;
use embassy_stm32_plus::embassy_time::{Delay, Duration, Timer};
use embassy_stm32_plus::traits::gpio::input::GpioInput;
use embassy_stm32_plus::traits::gpio::output::GpioOutput;
use embassy_stm32_plus::traits::spi::SpiDmaTrait;
use embedded_hal_bus::spi::ExclusiveDevice;
use embedded_io_async::Write;
use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::main]
async fn main(spawner: Spawner) {
// rcc setting or etc., more see https://github.com/embassy-rs/embassy/blob/main/examples/stm32f4/src/bin/eth.rs
let p = embassy_stm32::init(Default::default());
let spi = p.SPI1.build_with_dma(p.PA5, p.PA7, p.PA6, p.DMA1_CH3, p.DMA1_CH2);
let cs = p.PA4.output_with_level_speed(Level::High, Speed::VeryHigh);
let spi = defmt::unwrap!(ExclusiveDevice::new(spi,cs,Delay));
let w5500_int = ExtiInput::new(p.PB0.input(), p.EXTI0);
let w5500_reset = p.PB1.output_with_level_speed(Level::High, Speed::VeryHigh);
let mac_addr = [0x02, 234, 3, 4, 82, 231];
static STATE: StaticCell<State<2, 2>> = StaticCell::new();
let state = STATE.init(State::<2, 2>::new());
let (device, runner) = embassy_net_wiznet::new(mac_addr, state, spi, w5500_int, w5500_reset)
.await;
defmt::unwrap!(spawner.spawn(ethernet_task(runner)));
let config = embassy_net::Config::dhcpv4(Default::default());
static RESOURCES: StaticCell<StackResources<3>> = StaticCell::new();
let (stack, runner) = embassy_net::new(device, config, RESOURCES.init(StackResources::new()), 0);
defmt::unwrap!(spawner.spawn(net_task(runner)));
stack.wait_config_up().await;
defmt::info!("Network task initialized");
// Then we can use it!
let mut rx_buffer = [0; 1024];
let mut tx_buffer = [0; 1024];
loop {
let mut socket = TcpSocket::new(stack, &mut rx_buffer, &mut tx_buffer);
socket.set_timeout(Some(Duration::from_secs(10)));
let remote_endpoint = (Ipv4Address::new(10, 42, 0, 1), 8000);
defmt::info!("connecting...");
let r = socket.connect(remote_endpoint).await;
if let Err(e) = r {
defmt::info!("connect error: {:?}", e);
Timer::after_secs(1).await;
continue;
}
defmt::info!("connected!");
let buf = [0; 1024];
loop {
let r = socket.write_all(&buf).await;
if let Err(e) = r {
defmt::info!("write error: {:?}", e);
break;
}
Timer::after_secs(1).await;
}
}
}
#[allow(clippy::type_complexity)]
#[embassy_executor::task]
async fn ethernet_task(
runner:
Runner<'static, W5500,
ExclusiveDevice<Spi<'static, SPI1, DMA1_CH3, DMA1_CH2>, Output<'static, PA4>, Delay>,
ExtiInput<'static, PB0>,
Output<'static, PB1>>) -> ! {
runner.run().await
}
#[embassy_executor::task]
async fn net_task(mut runner: embassy_net::Runner<'static, Device<'static>>) -> ! {
runner.run().await
}