2023-04-06 18:53:51 +02:00
|
|
|
#![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;
|
2023-05-25 00:29:56 +02:00
|
|
|
use embassy_stm32::{bind_interrupts, eth, Config};
|
2023-04-06 18:53:51 +02:00
|
|
|
use embassy_time::{Duration, Timer};
|
|
|
|
use embedded_io::asynch::Write;
|
|
|
|
use rand_core::RngCore;
|
2023-06-01 01:32:11 +02:00
|
|
|
use static_cell::make_static;
|
2023-04-06 18:53:51 +02:00
|
|
|
use {defmt_rtt as _, panic_probe as _};
|
2023-05-25 00:29:56 +02:00
|
|
|
bind_interrupts!(struct Irqs {
|
|
|
|
ETH => eth::InterruptHandler;
|
|
|
|
});
|
|
|
|
|
2023-04-06 18:53:51 +02:00
|
|
|
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 mac_addr = [0x00, 0x00, 0xDE, 0xAD, 0xBE, 0xEF];
|
|
|
|
|
|
|
|
let device = Ethernet::new(
|
2023-06-01 01:32:11 +02:00
|
|
|
make_static!(PacketQueue::<4, 4>::new()),
|
2023-04-06 18:53:51 +02:00
|
|
|
p.ETH,
|
2023-05-25 00:29:56 +02:00
|
|
|
Irqs,
|
2023-04-06 18:53:51 +02:00
|
|
|
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());
|
2023-06-05 14:57:17 +02:00
|
|
|
//let config = embassy_net::Config::StaticV4(embassy_net::StaticConfigV4 {
|
2023-04-06 18:53:51 +02:00
|
|
|
// 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
|
2023-06-01 01:32:11 +02:00
|
|
|
let stack = &*make_static!(Stack::new(
|
|
|
|
device,
|
|
|
|
config,
|
|
|
|
make_static!(StackResources::<2>::new()),
|
|
|
|
seed
|
|
|
|
));
|
2023-04-06 18:53:51 +02:00
|
|
|
|
|
|
|
// 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);
|
|
|
|
|
2023-05-15 00:38:58 +02:00
|
|
|
socket.set_timeout(Some(embassy_time::Duration::from_secs(10)));
|
2023-04-06 18:53:51 +02:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|