134 lines
3.8 KiB
Rust
134 lines
3.8 KiB
Rust
|
#![no_std]
|
||
|
#![no_main]
|
||
|
#![feature(type_alias_impl_trait)]
|
||
|
|
||
|
use defmt::*;
|
||
|
use embassy_executor::Spawner;
|
||
|
use embassy_net::tcp::TcpSocket;
|
||
|
use embassy_net::{Ipv4Address, Stack, StackResources};
|
||
|
use embassy_stm32::eth::generic_smi::GenericSMI;
|
||
|
use embassy_stm32::eth::{Ethernet, PacketQueue};
|
||
|
use embassy_stm32::peripherals::ETH;
|
||
|
use embassy_stm32::rcc::{AHBPrescaler, APBPrescaler, Hse, HseMode, Pll, PllSource, Sysclk, VoltageScale};
|
||
|
use embassy_stm32::rng::Rng;
|
||
|
use embassy_stm32::time::Hertz;
|
||
|
use embassy_stm32::{interrupt, Config};
|
||
|
use embassy_time::{Duration, Timer};
|
||
|
use embedded_io::asynch::Write;
|
||
|
use rand_core::RngCore;
|
||
|
use static_cell::StaticCell;
|
||
|
use {defmt_rtt as _, panic_probe as _};
|
||
|
|
||
|
macro_rules! singleton {
|
||
|
($val:expr) => {{
|
||
|
type T = impl Sized;
|
||
|
static STATIC_CELL: StaticCell<T> = StaticCell::new();
|
||
|
let (x,) = STATIC_CELL.init(($val,));
|
||
|
x
|
||
|
}};
|
||
|
}
|
||
|
|
||
|
type Device = Ethernet<'static, ETH, GenericSMI>;
|
||
|
|
||
|
#[embassy_executor::task]
|
||
|
async fn net_task(stack: &'static Stack<Device>) -> ! {
|
||
|
stack.run().await
|
||
|
}
|
||
|
|
||
|
#[embassy_executor::main]
|
||
|
async fn main(spawner: Spawner) -> ! {
|
||
|
let mut config = Config::default();
|
||
|
config.rcc.hsi = None;
|
||
|
config.rcc.hsi48 = true; // needed for rng
|
||
|
config.rcc.hse = Some(Hse {
|
||
|
freq: Hertz(8_000_000),
|
||
|
mode: HseMode::BypassDigital,
|
||
|
});
|
||
|
config.rcc.pll1 = Some(Pll {
|
||
|
source: PllSource::Hse,
|
||
|
prediv: 2,
|
||
|
mul: 125,
|
||
|
divp: Some(2),
|
||
|
divq: Some(2),
|
||
|
divr: None,
|
||
|
});
|
||
|
config.rcc.ahb_pre = AHBPrescaler::NotDivided;
|
||
|
config.rcc.apb1_pre = APBPrescaler::NotDivided;
|
||
|
config.rcc.apb2_pre = APBPrescaler::NotDivided;
|
||
|
config.rcc.apb3_pre = APBPrescaler::NotDivided;
|
||
|
config.rcc.sys = Sysclk::Pll1P;
|
||
|
config.rcc.voltage_scale = VoltageScale::Scale0;
|
||
|
let p = embassy_stm32::init(config);
|
||
|
info!("Hello World!");
|
||
|
|
||
|
// Generate random seed.
|
||
|
let mut rng = Rng::new(p.RNG);
|
||
|
let mut seed = [0; 8];
|
||
|
rng.fill_bytes(&mut seed);
|
||
|
let seed = u64::from_le_bytes(seed);
|
||
|
|
||
|
let eth_int = interrupt::take!(ETH);
|
||
|
let mac_addr = [0x00, 0x00, 0xDE, 0xAD, 0xBE, 0xEF];
|
||
|
|
||
|
let device = Ethernet::new(
|
||
|
singleton!(PacketQueue::<4, 4>::new()),
|
||
|
p.ETH,
|
||
|
eth_int,
|
||
|
p.PA1,
|
||
|
p.PA2,
|
||
|
p.PC1,
|
||
|
p.PA7,
|
||
|
p.PC4,
|
||
|
p.PC5,
|
||
|
p.PG13,
|
||
|
p.PB15,
|
||
|
p.PG11,
|
||
|
GenericSMI,
|
||
|
mac_addr,
|
||
|
0,
|
||
|
);
|
||
|
|
||
|
let config = embassy_net::Config::Dhcp(Default::default());
|
||
|
//let config = embassy_net::Config::Static(embassy_net::StaticConfig {
|
||
|
// address: Ipv4Cidr::new(Ipv4Address::new(10, 42, 0, 61), 24),
|
||
|
// dns_servers: Vec::new(),
|
||
|
// gateway: Some(Ipv4Address::new(10, 42, 0, 1)),
|
||
|
//});
|
||
|
|
||
|
// Init network stack
|
||
|
let stack = &*singleton!(Stack::new(device, config, singleton!(StackResources::<2>::new()), seed));
|
||
|
|
||
|
// Launch network task
|
||
|
unwrap!(spawner.spawn(net_task(&stack)));
|
||
|
|
||
|
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(embassy_net::SmolDuration::from_secs(10)));
|
||
|
|
||
|
let remote_endpoint = (Ipv4Address::new(10, 42, 0, 1), 8000);
|
||
|
info!("connecting...");
|
||
|
let r = socket.connect(remote_endpoint).await;
|
||
|
if let Err(e) = r {
|
||
|
info!("connect error: {:?}", e);
|
||
|
Timer::after(Duration::from_secs(3)).await;
|
||
|
continue;
|
||
|
}
|
||
|
info!("connected!");
|
||
|
loop {
|
||
|
let r = socket.write_all(b"Hello\n").await;
|
||
|
if let Err(e) = r {
|
||
|
info!("write error: {:?}", e);
|
||
|
continue;
|
||
|
}
|
||
|
Timer::after(Duration::from_secs(1)).await;
|
||
|
}
|
||
|
}
|
||
|
}
|