//! This example shows how you can allow multiple simultaneous TCP connections, by having multiple sockets listening on the same port. //! //! Example written for the [`WIZnet W5500-EVB-Pico`](https://www.wiznet.io/product-item/w5500-evb-pico/) board. #![no_std] #![no_main] #![feature(type_alias_impl_trait)] use defmt::*; use embassy_executor::Spawner; use embassy_futures::yield_now; use embassy_net::{Stack, StackResources}; use embassy_net_w5500::*; use embassy_rp::clocks::RoscRng; use embassy_rp::gpio::{Input, Level, Output, Pull}; use embassy_rp::peripherals::{PIN_17, PIN_20, PIN_21, SPI0}; use embassy_rp::spi::{Async, Config as SpiConfig, Spi}; use embassy_time::{Delay, Duration}; use embedded_hal_async::spi::ExclusiveDevice; use embedded_io::asynch::Write; use rand::RngCore; use static_cell::make_static; use {defmt_rtt as _, panic_probe as _}; #[embassy_executor::task] async fn ethernet_task( runner: Runner< 'static, ExclusiveDevice, Output<'static, PIN_17>, Delay>, Input<'static, PIN_21>, Output<'static, PIN_20>, >, ) -> ! { runner.run().await } #[embassy_executor::task] async fn net_task(stack: &'static Stack>) -> ! { stack.run().await } #[embassy_executor::main] async fn main(spawner: Spawner) { let p = embassy_rp::init(Default::default()); let mut rng = RoscRng; let mut spi_cfg = SpiConfig::default(); spi_cfg.frequency = 50_000_000; let (miso, mosi, clk) = (p.PIN_16, p.PIN_19, p.PIN_18); let spi = Spi::new(p.SPI0, clk, mosi, miso, p.DMA_CH0, p.DMA_CH1, spi_cfg); let cs = Output::new(p.PIN_17, Level::High); let w5500_int = Input::new(p.PIN_21, Pull::Up); let w5500_reset = Output::new(p.PIN_20, Level::High); let mac_addr = [0x02, 0x00, 0x00, 0x00, 0x00, 0x00]; let state = make_static!(State::<8, 8>::new()); let (device, runner) = embassy_net_w5500::new( mac_addr, state, ExclusiveDevice::new(spi, cs, Delay), w5500_int, w5500_reset, ) .await; unwrap!(spawner.spawn(ethernet_task(runner))); // Generate random seed let seed = rng.next_u64(); // Init network stack let stack = &*make_static!(Stack::new( device, embassy_net::Config::dhcpv4(Default::default()), make_static!(StackResources::<3>::new()), seed )); // Launch network task unwrap!(spawner.spawn(net_task(&stack))); info!("Waiting for DHCP..."); let cfg = wait_for_config(stack).await; let local_addr = cfg.address.address(); info!("IP address: {:?}", local_addr); // Create two sockets listening to the same port, to handle simultaneous connections unwrap!(spawner.spawn(listen_task(&stack, 0, 1234))); unwrap!(spawner.spawn(listen_task(&stack, 1, 1234))); } #[embassy_executor::task(pool_size = 2)] async fn listen_task(stack: &'static Stack>, id: u8, port: u16) { let mut rx_buffer = [0; 4096]; let mut tx_buffer = [0; 4096]; let mut buf = [0; 4096]; loop { let mut socket = embassy_net::tcp::TcpSocket::new(stack, &mut rx_buffer, &mut tx_buffer); socket.set_timeout(Some(Duration::from_secs(10))); info!("SOCKET {}: Listening on TCP:{}...", id, port); if let Err(e) = socket.accept(port).await { warn!("accept error: {:?}", e); continue; } info!("SOCKET {}: Received connection from {:?}", id, socket.remote_endpoint()); loop { let n = match socket.read(&mut buf).await { Ok(0) => { warn!("read EOF"); break; } Ok(n) => n, Err(e) => { warn!("SOCKET {}: {:?}", id, e); break; } }; info!("SOCKET {}: rxd {}", id, core::str::from_utf8(&buf[..n]).unwrap()); if let Err(e) = socket.write_all(&buf[..n]).await { warn!("write error: {:?}", e); break; } } } } async fn wait_for_config(stack: &'static Stack>) -> embassy_net::StaticConfigV4 { loop { if let Some(config) = stack.config_v4() { return config.clone(); } yield_now().await; } }