Compare commits
	
		
			8 Commits
		
	
	
		
			comment-me
			...
			net-driver
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 3cbc687424 | ||
|  | 51708c8ed1 | ||
|  | 10f08445e4 | ||
|  | f24a1b62bb | ||
|  | bbd12c9372 | ||
|  | d94b9fe6fb | ||
|  | b478640463 | ||
|  | 846f2fc6e4 | 
							
								
								
									
										1
									
								
								ci.sh
									
									
									
									
									
								
							
							
						
						
									
										1
									
								
								ci.sh
									
									
									
									
									
								
							| @@ -204,6 +204,7 @@ cargo batch  \ | |||||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32f207zg --out-dir out/tests/stm32f207zg \ |     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32f207zg --out-dir out/tests/stm32f207zg \ | ||||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f303ze --out-dir out/tests/stm32f303ze \ |     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f303ze --out-dir out/tests/stm32f303ze \ | ||||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l496zg --out-dir out/tests/stm32l496zg \ |     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l496zg --out-dir out/tests/stm32l496zg \ | ||||||
|  |     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32wl55jc --out-dir out/tests/stm32wl55jc \ | ||||||
|     --- build --release --manifest-path tests/rp/Cargo.toml --target thumbv6m-none-eabi --out-dir out/tests/rpi-pico \ |     --- build --release --manifest-path tests/rp/Cargo.toml --target thumbv6m-none-eabi --out-dir out/tests/rpi-pico \ | ||||||
|     --- build --release --manifest-path tests/nrf/Cargo.toml --target thumbv7em-none-eabi --out-dir out/tests/nrf52840-dk \ |     --- build --release --manifest-path tests/nrf/Cargo.toml --target thumbv7em-none-eabi --out-dir out/tests/nrf52840-dk \ | ||||||
|     --- build --release --manifest-path tests/riscv32/Cargo.toml --target riscv32imac-unknown-none-elf \ |     --- build --release --manifest-path tests/riscv32/Cargo.toml --target riscv32imac-unknown-none-elf \ | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| use core::cmp::{max, min}; | use core::cmp::{max, min}; | ||||||
|  |  | ||||||
| use ch::driver::LinkState; |  | ||||||
| use embassy_net_driver_channel as ch; | use embassy_net_driver_channel as ch; | ||||||
|  | use embassy_net_driver_channel::driver::{HardwareAddress, LinkState}; | ||||||
| use embassy_time::Timer; | use embassy_time::Timer; | ||||||
|  |  | ||||||
| pub use crate::bus::SpiBusCyw43; | pub use crate::bus::SpiBusCyw43; | ||||||
| @@ -133,7 +133,7 @@ impl<'a> Control<'a> { | |||||||
|  |  | ||||||
|         Timer::after_millis(100).await; |         Timer::after_millis(100).await; | ||||||
|  |  | ||||||
|         self.state_ch.set_ethernet_address(mac_addr); |         self.state_ch.set_hardware_address(HardwareAddress::Ethernet(mac_addr)); | ||||||
|  |  | ||||||
|         debug!("INIT DONE"); |         debug!("INIT DONE"); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -48,7 +48,7 @@ The `Spawner` is the way the main application spawns other tasks. The `Periphera | |||||||
| include::example$basic/src/main.rs[lines="22..-1"] | include::example$basic/src/main.rs[lines="22..-1"] | ||||||
| ---- | ---- | ||||||
|  |  | ||||||
| What happens when the `blinker` task has been spawned and main returns? Well, the main entry point is actually just like any other task, except that you can only have one and it takes some specific type arguments. The magic lies within the `#[embassy::main]` macro. The macro does the following: | What happens when the `blinker` task has been spawned and main returns? Well, the main entry point is actually just like any other task, except that you can only have one and it takes some specific type arguments. The magic lies within the `#[embassy_executor::main]` macro. The macro does the following: | ||||||
|  |  | ||||||
| . Creates an Embassy Executor | . Creates an Embassy Executor | ||||||
| . Initializes the microcontroller HAL to get the `Peripherals` | . Initializes the microcontroller HAL to get the `Peripherals` | ||||||
|   | |||||||
| @@ -5,14 +5,12 @@ All notable changes to this project will be documented in this file. | |||||||
| The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | ||||||
| and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | ||||||
|  |  | ||||||
| ## 0.2.0 - 2023-10-15 | ## 0.2.0 - 2023-10-18 | ||||||
|  |  | ||||||
| - Update embassy-net-driver | - Update `embassy-net-driver` to v0.2 | ||||||
| - `Runner::new` now takes an `embassy_net_driver::HardwareAddress` parameter | - `Runner::new` now takes an `embassy_net_driver::HardwareAddress` parameter. | ||||||
| - Added `Runner::set_ieee802154_address`, `Runner::ieee802154_address` | - `Runner::set_ethernet_address` is now `set_hardware_address`. | ||||||
|  |  | ||||||
| ## 0.1.0 - 2023-06-29 | ## 0.1.0 - 2023-06-29 | ||||||
|  |  | ||||||
| - First release | - First release | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,7 +7,9 @@ The `embassy-net-driver` trait is polling-based. To implement it, you must write | |||||||
| hand, and hook up the `Waker`s provided by `embassy-net` to the right interrupt handlers so that `embassy-net` | hand, and hook up the `Waker`s provided by `embassy-net` to the right interrupt handlers so that `embassy-net` | ||||||
| knows when to poll your driver again to make more progress. | knows when to poll your driver again to make more progress. | ||||||
|  |  | ||||||
| With `embassy-net-driver-channel` | With `embassy-net-driver-channel` you get a "channel-like" interface instead, where you can send/receive packets | ||||||
|  | to/from embassy-net. The intended usage is to spawn a "driver task" in the background that does this, passing | ||||||
|  | packets between the hardware and the channel. | ||||||
|  |  | ||||||
| ## A note about deadlocks | ## A note about deadlocks | ||||||
|  |  | ||||||
| @@ -41,7 +43,7 @@ However, this code has a latent deadlock bug. The symptom is it can hang at `rx_ | |||||||
|  |  | ||||||
| The reason is that, under load, both the TX and RX queues can get full at the same time. When this happens, the `embassy-net` task stalls trying to send because the TX queue is full, therefore it stops processing packets in the RX queue. Your driver task also stalls because the RX queue is full, therefore it stops processing packets in the TX queue. | The reason is that, under load, both the TX and RX queues can get full at the same time. When this happens, the `embassy-net` task stalls trying to send because the TX queue is full, therefore it stops processing packets in the RX queue. Your driver task also stalls because the RX queue is full, therefore it stops processing packets in the TX queue. | ||||||
|  |  | ||||||
| The fix is to make sure to always service the TX queue while you're waiting for space to become available in the TX queue. For example, select on either "tx_chan.tx_buf() available" or "INT is low AND rx_chan.rx_buf() available": | The fix is to make sure to always service the TX queue while you're waiting for space to become available in the RX queue. For example, select on either "tx_chan.tx_buf() available" or "INT is low AND rx_chan.rx_buf() available": | ||||||
|  |  | ||||||
| ```rust,ignore | ```rust,ignore | ||||||
| loop { | loop { | ||||||
| @@ -79,12 +81,10 @@ These `embassy-net` drivers are implemented using this crate. You can look at th | |||||||
| - [`embassy-net-wiznet`](https://github.com/embassy-rs/embassy/tree/main/embassy-net-wiznet) for Wiznet SPI Ethernet MAC+PHY chips. | - [`embassy-net-wiznet`](https://github.com/embassy-rs/embassy/tree/main/embassy-net-wiznet) for Wiznet SPI Ethernet MAC+PHY chips. | ||||||
| - [`embassy-net-esp-hosted`](https://github.com/embassy-rs/embassy/tree/main/embassy-net-esp-hosted) for using ESP32 chips with the [`esp-hosted`](https://github.com/espressif/esp-hosted) firmware as WiFi adapters for another non-ESP32 MCU. | - [`embassy-net-esp-hosted`](https://github.com/embassy-rs/embassy/tree/main/embassy-net-esp-hosted) for using ESP32 chips with the [`esp-hosted`](https://github.com/espressif/esp-hosted) firmware as WiFi adapters for another non-ESP32 MCU. | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Interoperability | ## Interoperability | ||||||
|  |  | ||||||
| This crate can run on any executor. | This crate can run on any executor. | ||||||
|  |  | ||||||
|  |  | ||||||
| ## License | ## License | ||||||
|  |  | ||||||
| This work is licensed under either of | This work is licensed under either of | ||||||
|   | |||||||
| @@ -8,9 +8,8 @@ use core::cell::RefCell; | |||||||
| use core::mem::MaybeUninit; | use core::mem::MaybeUninit; | ||||||
| use core::task::{Context, Poll}; | use core::task::{Context, Poll}; | ||||||
|  |  | ||||||
| use driver::HardwareAddress; |  | ||||||
| pub use embassy_net_driver as driver; | pub use embassy_net_driver as driver; | ||||||
| use embassy_net_driver::{Capabilities, LinkState, Medium}; | use embassy_net_driver::{Capabilities, LinkState}; | ||||||
| use embassy_sync::blocking_mutex::raw::NoopRawMutex; | use embassy_sync::blocking_mutex::raw::NoopRawMutex; | ||||||
| use embassy_sync::blocking_mutex::Mutex; | use embassy_sync::blocking_mutex::Mutex; | ||||||
| use embassy_sync::waitqueue::WakerRegistration; | use embassy_sync::waitqueue::WakerRegistration; | ||||||
| @@ -161,18 +160,10 @@ impl<'d> StateRunner<'d> { | |||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn set_ethernet_address(&self, address: [u8; 6]) { |     pub fn set_hardware_address(&self, address: driver::HardwareAddress) { | ||||||
|         self.shared.lock(|s| { |         self.shared.lock(|s| { | ||||||
|             let s = &mut *s.borrow_mut(); |             let s = &mut *s.borrow_mut(); | ||||||
|             s.hardware_address = driver::HardwareAddress::Ethernet(address); |             s.hardware_address = address; | ||||||
|             s.waker.wake(); |  | ||||||
|         }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn set_ieee802154_address(&self, address: [u8; 8]) { |  | ||||||
|         self.shared.lock(|s| { |  | ||||||
|             let s = &mut *s.borrow_mut(); |  | ||||||
|             s.hardware_address = driver::HardwareAddress::Ieee802154(address); |  | ||||||
|             s.waker.wake(); |             s.waker.wake(); | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
| @@ -232,11 +223,6 @@ pub fn new<'d, const MTU: usize, const N_RX: usize, const N_TX: usize>( | |||||||
| ) -> (Runner<'d, MTU>, Device<'d, MTU>) { | ) -> (Runner<'d, MTU>, Device<'d, MTU>) { | ||||||
|     let mut caps = Capabilities::default(); |     let mut caps = Capabilities::default(); | ||||||
|     caps.max_transmission_unit = MTU; |     caps.max_transmission_unit = MTU; | ||||||
|     caps.medium = match &hardware_address { |  | ||||||
|         HardwareAddress::Ethernet(_) => Medium::Ethernet, |  | ||||||
|         HardwareAddress::Ieee802154(_) => Medium::Ieee802154, |  | ||||||
|         HardwareAddress::Ip => Medium::Ip, |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     // safety: this is a self-referential struct, however: |     // safety: this is a self-referential struct, however: | ||||||
|     // - it can't move while the `'d` borrow is active. |     // - it can't move while the `'d` borrow is active. | ||||||
|   | |||||||
| @@ -5,13 +5,13 @@ All notable changes to this project will be documented in this file. | |||||||
| The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | ||||||
| and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | ||||||
|  |  | ||||||
| ## 0.2.0 - 2023-10-15 | ## 0.2.0 - 2023-10-18 | ||||||
|  |  | ||||||
| - Added `Driver::ieee802154_address` | - Added support for IEEE 802.15.4 mediums. | ||||||
| - Added `Medium::Ieee802154` | - Added `Driver::hardware_address()`, `HardwareAddress`. | ||||||
|  | - Removed `Medium` enum. The medium is deduced out of the hardware address. | ||||||
|  | - Removed `Driver::ethernet_address()`. Replacement is `hardware_address()`. | ||||||
|  |  | ||||||
| ## 0.1.0 - 2023-06-29 | ## 0.1.0 - 2023-06-29 | ||||||
|  |  | ||||||
| - First release | - First release | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,12 +7,23 @@ use core::task::Context; | |||||||
| /// Representation of an hardware address, such as an Ethernet address or an IEEE802.15.4 address. | /// Representation of an hardware address, such as an Ethernet address or an IEEE802.15.4 address. | ||||||
| #[derive(Debug, Clone, Copy, PartialEq, Eq)] | #[derive(Debug, Clone, Copy, PartialEq, Eq)] | ||||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||||
|  | #[non_exhaustive] | ||||||
| pub enum HardwareAddress { | pub enum HardwareAddress { | ||||||
|     /// A six-octet Ethernet address |     /// Ethernet medium, with a A six-octet Ethernet address. | ||||||
|  |     /// | ||||||
|  |     /// Devices of this type send and receive Ethernet frames, | ||||||
|  |     /// and interfaces using it must do neighbor discovery via ARP or NDISC. | ||||||
|  |     /// | ||||||
|  |     /// Examples of devices of this type are Ethernet, WiFi (802.11), Linux `tap`, and VPNs in tap (layer 2) mode. | ||||||
|     Ethernet([u8; 6]), |     Ethernet([u8; 6]), | ||||||
|     /// An eight-octet IEEE802.15.4 address |     /// 6LoWPAN over IEEE802.15.4, with an eight-octet address. | ||||||
|     Ieee802154([u8; 8]), |     Ieee802154([u8; 8]), | ||||||
|     /// Indicates that a Driver is IP-native, and has no hardware address |     /// Indicates that a Driver is IP-native, and has no hardware address. | ||||||
|  |     /// | ||||||
|  |     /// Devices of this type send and receive IP frames, without an | ||||||
|  |     /// Ethernet header. MAC addresses are not used, and no neighbor discovery (ARP, NDISC) is done. | ||||||
|  |     /// | ||||||
|  |     /// Examples of devices of this type are the Linux `tun`, PPP interfaces, VPNs in tun (layer 3) mode. | ||||||
|     Ip, |     Ip, | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -64,6 +75,10 @@ pub trait Driver { | |||||||
|     fn capabilities(&self) -> Capabilities; |     fn capabilities(&self) -> Capabilities; | ||||||
|  |  | ||||||
|     /// Get the device's hardware address. |     /// Get the device's hardware address. | ||||||
|  |     /// | ||||||
|  |     /// The returned hardware address also determines the "medium" of this driver. This indicates | ||||||
|  |     /// what kind of packet the sent/received bytes are, and determines some behaviors of | ||||||
|  |     /// the interface. For example, ARP/NDISC address resolution is only done for Ethernet mediums. | ||||||
|     fn hardware_address(&self) -> HardwareAddress; |     fn hardware_address(&self) -> HardwareAddress; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -124,13 +139,6 @@ pub trait TxToken { | |||||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||||
| #[non_exhaustive] | #[non_exhaustive] | ||||||
| pub struct Capabilities { | pub struct Capabilities { | ||||||
|     /// Medium of the device. |  | ||||||
|     /// |  | ||||||
|     /// This indicates what kind of packet the sent/received bytes are, and determines |  | ||||||
|     /// some behaviors of Interface. For example, ARP/NDISC address resolution is only done |  | ||||||
|     /// for Ethernet mediums. |  | ||||||
|     pub medium: Medium, |  | ||||||
|  |  | ||||||
|     /// Maximum transmission unit. |     /// Maximum transmission unit. | ||||||
|     /// |     /// | ||||||
|     /// The network device is unable to send or receive frames larger than the value returned |     /// The network device is unable to send or receive frames larger than the value returned | ||||||
| @@ -161,32 +169,6 @@ pub struct Capabilities { | |||||||
|     pub checksum: ChecksumCapabilities, |     pub checksum: ChecksumCapabilities, | ||||||
| } | } | ||||||
|  |  | ||||||
| /// Type of medium of a device. |  | ||||||
| #[derive(Debug, Eq, PartialEq, Copy, Clone)] |  | ||||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] |  | ||||||
| pub enum Medium { |  | ||||||
|     /// Ethernet medium. Devices of this type send and receive Ethernet frames, |  | ||||||
|     /// and interfaces using it must do neighbor discovery via ARP or NDISC. |  | ||||||
|     /// |  | ||||||
|     /// Examples of devices of this type are Ethernet, WiFi (802.11), Linux `tap`, and VPNs in tap (layer 2) mode. |  | ||||||
|     Ethernet, |  | ||||||
|  |  | ||||||
|     /// IP medium. Devices of this type send and receive IP frames, without an |  | ||||||
|     /// Ethernet header. MAC addresses are not used, and no neighbor discovery (ARP, NDISC) is done. |  | ||||||
|     /// |  | ||||||
|     /// Examples of devices of this type are the Linux `tun`, PPP interfaces, VPNs in tun (layer 3) mode. |  | ||||||
|     Ip, |  | ||||||
|  |  | ||||||
|     /// IEEE 802_15_4 medium |  | ||||||
|     Ieee802154, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl Default for Medium { |  | ||||||
|     fn default() -> Medium { |  | ||||||
|         Medium::Ethernet |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /// A description of checksum behavior for every supported protocol. | /// A description of checksum behavior for every supported protocol. | ||||||
| #[derive(Debug, Clone, Default)] | #[derive(Debug, Clone, Default)] | ||||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ mod traits; | |||||||
| use core::cmp; | use core::cmp; | ||||||
| use core::convert::TryInto; | use core::convert::TryInto; | ||||||
|  |  | ||||||
| use embassy_net_driver::{Capabilities, HardwareAddress, LinkState, Medium}; | use embassy_net_driver::{Capabilities, HardwareAddress, LinkState}; | ||||||
| use embassy_time::Duration; | use embassy_time::Duration; | ||||||
| use embedded_hal::digital::OutputPin; | use embedded_hal::digital::OutputPin; | ||||||
| use embedded_hal::spi::{Operation, SpiDevice}; | use embedded_hal::spi::{Operation, SpiDevice}; | ||||||
| @@ -671,7 +671,6 @@ where | |||||||
|     fn capabilities(&self) -> Capabilities { |     fn capabilities(&self) -> Capabilities { | ||||||
|         let mut caps = Capabilities::default(); |         let mut caps = Capabilities::default(); | ||||||
|         caps.max_transmission_unit = MTU; |         caps.max_transmission_unit = MTU; | ||||||
|         caps.medium = Medium::Ethernet; |  | ||||||
|         caps |         caps | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| use ch::driver::LinkState; |  | ||||||
| use embassy_net_driver_channel as ch; | use embassy_net_driver_channel as ch; | ||||||
|  | use embassy_net_driver_channel::driver::{HardwareAddress, LinkState}; | ||||||
| use heapless::String; | use heapless::String; | ||||||
|  |  | ||||||
| use crate::ioctl::Shared; | use crate::ioctl::Shared; | ||||||
| @@ -77,7 +77,7 @@ impl<'a> Control<'a> { | |||||||
|  |  | ||||||
|         let mac_addr = self.get_mac_addr().await?; |         let mac_addr = self.get_mac_addr().await?; | ||||||
|         debug!("mac addr: {:02x}", mac_addr); |         debug!("mac addr: {:02x}", mac_addr); | ||||||
|         self.state_ch.set_ethernet_address(mac_addr); |         self.state_ch.set_hardware_address(HardwareAddress::Ethernet(mac_addr)); | ||||||
|  |  | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ All notable changes to this project will be documented in this file. | |||||||
| The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | ||||||
| and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | ||||||
|  |  | ||||||
| ## 0.2.0 - 2023-10-15 | ## 0.2.0 - 2023-10-18 | ||||||
|  |  | ||||||
| - Re-export `smoltcp::wire::IpEndpoint` | - Re-export `smoltcp::wire::IpEndpoint` | ||||||
| - Add poll functions on UdpSocket | - Add poll functions on UdpSocket | ||||||
| @@ -27,5 +27,3 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 | |||||||
| ## 0.1.0 - 2023-06-29 | ## 0.1.0 - 2023-06-29 | ||||||
|  |  | ||||||
| - First release | - First release | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| use core::task::Context; | use core::task::Context; | ||||||
|  |  | ||||||
| use embassy_net_driver::{Capabilities, Checksum, Driver, Medium, RxToken, TxToken}; | use embassy_net_driver::{Capabilities, Checksum, Driver, RxToken, TxToken}; | ||||||
| use smoltcp::phy; | use smoltcp::phy::{self, Medium}; | ||||||
| use smoltcp::time::Instant; | use smoltcp::time::Instant; | ||||||
|  |  | ||||||
| pub(crate) struct DriverAdapter<'d, 'c, T> | pub(crate) struct DriverAdapter<'d, 'c, T> | ||||||
| @@ -11,6 +11,7 @@ where | |||||||
|     // must be Some when actually using this to rx/tx |     // must be Some when actually using this to rx/tx | ||||||
|     pub cx: Option<&'d mut Context<'c>>, |     pub cx: Option<&'d mut Context<'c>>, | ||||||
|     pub inner: &'d mut T, |     pub inner: &'d mut T, | ||||||
|  |     pub medium: Medium, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'d, 'c, T> phy::Device for DriverAdapter<'d, 'c, T> | impl<'d, 'c, T> phy::Device for DriverAdapter<'d, 'c, T> | ||||||
| @@ -46,19 +47,7 @@ where | |||||||
|  |  | ||||||
|         smolcaps.max_transmission_unit = caps.max_transmission_unit; |         smolcaps.max_transmission_unit = caps.max_transmission_unit; | ||||||
|         smolcaps.max_burst_size = caps.max_burst_size; |         smolcaps.max_burst_size = caps.max_burst_size; | ||||||
|         smolcaps.medium = match caps.medium { |         smolcaps.medium = self.medium; | ||||||
|             #[cfg(feature = "medium-ethernet")] |  | ||||||
|             Medium::Ethernet => phy::Medium::Ethernet, |  | ||||||
|             #[cfg(feature = "medium-ip")] |  | ||||||
|             Medium::Ip => phy::Medium::Ip, |  | ||||||
|             #[cfg(feature = "medium-ieee802154")] |  | ||||||
|             Medium::Ieee802154 => phy::Medium::Ieee802154, |  | ||||||
|             #[allow(unreachable_patterns)] |  | ||||||
|             _ => panic!( |  | ||||||
|                 "Unsupported medium {:?}. Make sure to enable it in embassy-net's Cargo features.", |  | ||||||
|                 caps.medium |  | ||||||
|             ), |  | ||||||
|         }; |  | ||||||
|         smolcaps.checksum.ipv4 = convert(caps.checksum.ipv4); |         smolcaps.checksum.ipv4 = convert(caps.checksum.ipv4); | ||||||
|         smolcaps.checksum.tcp = convert(caps.checksum.tcp); |         smolcaps.checksum.tcp = convert(caps.checksum.tcp); | ||||||
|         smolcaps.checksum.udp = convert(caps.checksum.udp); |         smolcaps.checksum.udp = convert(caps.checksum.udp); | ||||||
|   | |||||||
| @@ -33,6 +33,7 @@ use heapless::Vec; | |||||||
| pub use smoltcp::iface::MulticastError; | pub use smoltcp::iface::MulticastError; | ||||||
| #[allow(unused_imports)] | #[allow(unused_imports)] | ||||||
| use smoltcp::iface::{Interface, SocketHandle, SocketSet, SocketStorage}; | use smoltcp::iface::{Interface, SocketHandle, SocketSet, SocketStorage}; | ||||||
|  | use smoltcp::phy::Medium; | ||||||
| #[cfg(feature = "dhcpv4")] | #[cfg(feature = "dhcpv4")] | ||||||
| use smoltcp::socket::dhcpv4::{self, RetryConfig}; | use smoltcp::socket::dhcpv4::{self, RetryConfig}; | ||||||
| #[cfg(feature = "medium-ethernet")] | #[cfg(feature = "medium-ethernet")] | ||||||
| @@ -264,14 +265,17 @@ pub(crate) struct SocketStack { | |||||||
|     next_local_port: u16, |     next_local_port: u16, | ||||||
| } | } | ||||||
|  |  | ||||||
| fn to_smoltcp_hardware_address(addr: driver::HardwareAddress) -> HardwareAddress { | fn to_smoltcp_hardware_address(addr: driver::HardwareAddress) -> (HardwareAddress, Medium) { | ||||||
|     match addr { |     match addr { | ||||||
|         #[cfg(feature = "medium-ethernet")] |         #[cfg(feature = "medium-ethernet")] | ||||||
|         driver::HardwareAddress::Ethernet(eth) => HardwareAddress::Ethernet(EthernetAddress(eth)), |         driver::HardwareAddress::Ethernet(eth) => (HardwareAddress::Ethernet(EthernetAddress(eth)), Medium::Ethernet), | ||||||
|         #[cfg(feature = "medium-ieee802154")] |         #[cfg(feature = "medium-ieee802154")] | ||||||
|         driver::HardwareAddress::Ieee802154(ieee) => HardwareAddress::Ieee802154(Ieee802154Address::Extended(ieee)), |         driver::HardwareAddress::Ieee802154(ieee) => ( | ||||||
|  |             HardwareAddress::Ieee802154(Ieee802154Address::Extended(ieee)), | ||||||
|  |             Medium::Ieee802154, | ||||||
|  |         ), | ||||||
|         #[cfg(feature = "medium-ip")] |         #[cfg(feature = "medium-ip")] | ||||||
|         driver::HardwareAddress::Ip => HardwareAddress::Ip, |         driver::HardwareAddress::Ip => (HardwareAddress::Ip, Medium::Ip), | ||||||
|  |  | ||||||
|         #[allow(unreachable_patterns)] |         #[allow(unreachable_patterns)] | ||||||
|         _ => panic!( |         _ => panic!( | ||||||
| @@ -289,7 +293,8 @@ impl<D: Driver> Stack<D> { | |||||||
|         resources: &'static mut StackResources<SOCK>, |         resources: &'static mut StackResources<SOCK>, | ||||||
|         random_seed: u64, |         random_seed: u64, | ||||||
|     ) -> Self { |     ) -> Self { | ||||||
|         let mut iface_cfg = smoltcp::iface::Config::new(to_smoltcp_hardware_address(device.hardware_address())); |         let (hardware_addr, medium) = to_smoltcp_hardware_address(device.hardware_address()); | ||||||
|  |         let mut iface_cfg = smoltcp::iface::Config::new(hardware_addr); | ||||||
|         iface_cfg.random_seed = random_seed; |         iface_cfg.random_seed = random_seed; | ||||||
|  |  | ||||||
|         let iface = Interface::new( |         let iface = Interface::new( | ||||||
| @@ -297,6 +302,7 @@ impl<D: Driver> Stack<D> { | |||||||
|             &mut DriverAdapter { |             &mut DriverAdapter { | ||||||
|                 inner: &mut device, |                 inner: &mut device, | ||||||
|                 cx: None, |                 cx: None, | ||||||
|  |                 medium, | ||||||
|             }, |             }, | ||||||
|             instant_to_smoltcp(Instant::now()), |             instant_to_smoltcp(Instant::now()), | ||||||
|         ); |         ); | ||||||
| @@ -356,7 +362,7 @@ impl<D: Driver> Stack<D> { | |||||||
|  |  | ||||||
|     /// Get the hardware address of the network interface. |     /// Get the hardware address of the network interface. | ||||||
|     pub fn hardware_address(&self) -> HardwareAddress { |     pub fn hardware_address(&self) -> HardwareAddress { | ||||||
|         self.with(|_s, i| to_smoltcp_hardware_address(i.device.hardware_address())) |         self.with(|_s, i| to_smoltcp_hardware_address(i.device.hardware_address()).0) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /// Get whether the link is up. |     /// Get whether the link is up. | ||||||
| @@ -812,18 +818,28 @@ impl<D: Driver> Inner<D> { | |||||||
|     fn poll(&mut self, cx: &mut Context<'_>, s: &mut SocketStack) { |     fn poll(&mut self, cx: &mut Context<'_>, s: &mut SocketStack) { | ||||||
|         s.waker.register(cx.waker()); |         s.waker.register(cx.waker()); | ||||||
|  |  | ||||||
|  |         let (_hardware_addr, medium) = to_smoltcp_hardware_address(self.device.hardware_address()); | ||||||
|  |  | ||||||
|         #[cfg(any(feature = "medium-ethernet", feature = "medium-ieee802154"))] |         #[cfg(any(feature = "medium-ethernet", feature = "medium-ieee802154"))] | ||||||
|         if self.device.capabilities().medium == embassy_net_driver::Medium::Ethernet |  | ||||||
|             || self.device.capabilities().medium == embassy_net_driver::Medium::Ieee802154 |  | ||||||
|         { |         { | ||||||
|             s.iface |             let do_set = match medium { | ||||||
|                 .set_hardware_addr(to_smoltcp_hardware_address(self.device.hardware_address())); |                 #[cfg(feature = "medium-ethernet")] | ||||||
|  |                 Medium::Ethernet => true, | ||||||
|  |                 #[cfg(feature = "medium-ieee802154")] | ||||||
|  |                 Medium::Ieee802154 => true, | ||||||
|  |                 #[allow(unreachable_patterns)] | ||||||
|  |                 _ => false, | ||||||
|  |             }; | ||||||
|  |             if do_set { | ||||||
|  |                 s.iface.set_hardware_addr(_hardware_addr); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         let timestamp = instant_to_smoltcp(Instant::now()); |         let timestamp = instant_to_smoltcp(Instant::now()); | ||||||
|         let mut smoldev = DriverAdapter { |         let mut smoldev = DriverAdapter { | ||||||
|             cx: Some(cx), |             cx: Some(cx), | ||||||
|             inner: &mut self.device, |             inner: &mut self.device, | ||||||
|  |             medium, | ||||||
|         }; |         }; | ||||||
|         s.iface.poll(timestamp, &mut smoldev, &mut s.sockets); |         s.iface.poll(timestamp, &mut smoldev, &mut s.sockets); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ | |||||||
|  |  | ||||||
| use core::task::Context; | use core::task::Context; | ||||||
|  |  | ||||||
| use embassy_net_driver::{Capabilities, HardwareAddress, LinkState, Medium}; | use embassy_net_driver::{Capabilities, HardwareAddress, LinkState}; | ||||||
| use embassy_sync::blocking_mutex::raw::CriticalSectionRawMutex; | use embassy_sync::blocking_mutex::raw::CriticalSectionRawMutex; | ||||||
| use embassy_sync::channel::Channel; | use embassy_sync::channel::Channel; | ||||||
|  |  | ||||||
| @@ -60,24 +60,15 @@ impl<'d> embassy_net_driver::Driver for Driver<'d> { | |||||||
|         let mut caps = Capabilities::default(); |         let mut caps = Capabilities::default(); | ||||||
|         caps.max_transmission_unit = MTU; |         caps.max_transmission_unit = MTU; | ||||||
|         // caps.max_burst_size = Some(self.tx.len()); |         // caps.max_burst_size = Some(self.tx.len()); | ||||||
|  |  | ||||||
|         caps.medium = Medium::Ieee802154; |  | ||||||
|         caps |         caps | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fn link_state(&mut self, _cx: &mut Context) -> LinkState { |     fn link_state(&mut self, _cx: &mut Context) -> LinkState { | ||||||
|         //        if self.phy.poll_link(&mut self.station_management, cx) { |  | ||||||
|         //            LinkState::Up |  | ||||||
|         //        } else { |  | ||||||
|         //            LinkState::Down |  | ||||||
|         //        } |  | ||||||
|  |  | ||||||
|         LinkState::Down |         LinkState::Down | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fn hardware_address(&self) -> HardwareAddress { |     fn hardware_address(&self) -> HardwareAddress { | ||||||
|         // self.mac_addr |         // self.mac_addr | ||||||
|  |  | ||||||
|         HardwareAddress::Ieee802154([0; 8]) |         HardwareAddress::Ieee802154([0; 8]) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -58,7 +58,7 @@ rand_core = "0.6.3" | |||||||
| sdio-host = "0.5.0" | sdio-host = "0.5.0" | ||||||
| embedded-sdmmc = { git = "https://github.com/embassy-rs/embedded-sdmmc-rs", rev = "a4f293d3a6f72158385f79c98634cb8a14d0d2fc", optional = true } | embedded-sdmmc = { git = "https://github.com/embassy-rs/embedded-sdmmc-rs", rev = "a4f293d3a6f72158385f79c98634cb8a14d0d2fc", optional = true } | ||||||
| critical-section = "1.1" | critical-section = "1.1" | ||||||
| stm32-metapac = { git = "https://github.com/embassy-rs/stm32-data-generated", tag = "stm32-data-c20cbde88fdfaef4645361d09df0cb63a4dc6462" } | stm32-metapac = { git = "https://github.com/embassy-rs/stm32-data-generated", tag = "stm32-data-6f7449303bf8af60a63704d35df9af46006c6148" } | ||||||
| vcell = "0.1.3" | vcell = "0.1.3" | ||||||
| bxcan = "0.7.0" | bxcan = "0.7.0" | ||||||
| nb = "1.0.0" | nb = "1.0.0" | ||||||
| @@ -76,7 +76,7 @@ critical-section = { version = "1.1", features = ["std"] } | |||||||
| [build-dependencies] | [build-dependencies] | ||||||
| proc-macro2 = "1.0.36" | proc-macro2 = "1.0.36" | ||||||
| quote = "1.0.15" | quote = "1.0.15" | ||||||
| stm32-metapac = { git = "https://github.com/embassy-rs/stm32-data-generated", tag = "stm32-data-c20cbde88fdfaef4645361d09df0cb63a4dc6462", default-features = false, features = ["metadata"]} | stm32-metapac = { git = "https://github.com/embassy-rs/stm32-data-generated", tag = "stm32-data-6f7449303bf8af60a63704d35df9af46006c6148", default-features = false, features = ["metadata"]} | ||||||
|  |  | ||||||
|  |  | ||||||
| [features] | [features] | ||||||
|   | |||||||
| @@ -127,7 +127,7 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     if config.usb_pll { |     if config.usb_pll { | ||||||
|         RCC.cfgr3().modify(|w| w.set_usbsw(Usbsw::PLLCLK)); |         RCC.cfgr3().modify(|w| w.set_usbsw(Usbsw::PLL1_P)); | ||||||
|     } |     } | ||||||
|     // TODO: Option to use CRS (Clock Recovery) |     // TODO: Option to use CRS (Clock Recovery) | ||||||
|  |  | ||||||
| @@ -140,7 +140,7 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|         RCC.cfgr().modify(|w| { |         RCC.cfgr().modify(|w| { | ||||||
|             w.set_ppre(Ppre::from_bits(ppre_bits)); |             w.set_ppre(Ppre::from_bits(ppre_bits)); | ||||||
|             w.set_hpre(Hpre::from_bits(hpre_bits)); |             w.set_hpre(Hpre::from_bits(hpre_bits)); | ||||||
|             w.set_sw(Sw::PLL) |             w.set_sw(Sw::PLL1_P) | ||||||
|         }); |         }); | ||||||
|     } else { |     } else { | ||||||
|         RCC.cfgr().modify(|w| { |         RCC.cfgr().modify(|w| { | ||||||
|   | |||||||
| @@ -169,7 +169,14 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|         #[cfg(not(rcc_f100))] |         #[cfg(not(rcc_f100))] | ||||||
|         w.set_usbpre(Usbpre::from_bits(usbpre as u8)); |         w.set_usbpre(Usbpre::from_bits(usbpre as u8)); | ||||||
|         w.set_sw(if pllmul_bits.is_some() { |         w.set_sw(if pllmul_bits.is_some() { | ||||||
|             Sw::PLL |             #[cfg(not(rcc_f1cl))] | ||||||
|  |             { | ||||||
|  |                 Sw::PLL1_P | ||||||
|  |             } | ||||||
|  |             #[cfg(rcc_f1cl)] | ||||||
|  |             { | ||||||
|  |                 Sw::PLL | ||||||
|  |             } | ||||||
|         } else if config.hse.is_some() { |         } else if config.hse.is_some() { | ||||||
|             Sw::HSE |             Sw::HSE | ||||||
|         } else { |         } else { | ||||||
|   | |||||||
| @@ -256,7 +256,7 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|         ClockSrc::PLL => { |         ClockSrc::PLL => { | ||||||
|             RCC.cr().modify(|w| w.set_pllon(true)); |             RCC.cr().modify(|w| w.set_pllon(true)); | ||||||
|             while !RCC.cr().read().pllrdy() {} |             while !RCC.cr().read().pllrdy() {} | ||||||
|             (pll_clocks.main_freq, Sw::PLL) |             (pll_clocks.main_freq, Sw::PLL1_P) | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
|     // RM0033 Figure 9. Clock tree suggests max SYSCLK/HCLK is 168 MHz, but datasheet specifies PLL |     // RM0033 Figure 9. Clock tree suggests max SYSCLK/HCLK is 168 MHz, but datasheet specifies PLL | ||||||
|   | |||||||
| @@ -214,7 +214,7 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|     // CFGR has been written before (PLL, PLL48, clock divider) don't overwrite these settings |     // CFGR has been written before (PLL, PLL48, clock divider) don't overwrite these settings | ||||||
|     RCC.cfgr().modify(|w| { |     RCC.cfgr().modify(|w| { | ||||||
|         w.set_sw(match (pll_config, config.hse) { |         w.set_sw(match (pll_config, config.hse) { | ||||||
|             (Some(_), _) => Sw::PLL, |             (Some(_), _) => Sw::PLL1_P, | ||||||
|             (None, Some(_)) => Sw::HSE, |             (None, Some(_)) => Sw::HSE, | ||||||
|             (None, None) => Sw::HSI, |             (None, None) => Sw::HSI, | ||||||
|         }) |         }) | ||||||
| @@ -271,7 +271,7 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|             pll_config.unwrap(); |             pll_config.unwrap(); | ||||||
|             assert!((pclk2 == sysclk) || (pclk2 * 2u32 == sysclk)); |             assert!((pclk2 == sysclk) || (pclk2 * 2u32 == sysclk)); | ||||||
|  |  | ||||||
|             RCC.cfgr3().modify(|w| w.set_hrtim1sw(Timsw::PLL)); |             RCC.cfgr3().modify(|w| w.set_hrtim1sw(Timsw::PLL1_P)); | ||||||
|  |  | ||||||
|             Some(sysclk * 2u32) |             Some(sysclk * 2u32) | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -328,7 +328,7 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|  |  | ||||||
|     RCC.cfgr().modify(|w| { |     RCC.cfgr().modify(|w| { | ||||||
|         w.set_sw(if sysclk_on_pll { |         w.set_sw(if sysclk_on_pll { | ||||||
|             Sw::PLL |             Sw::PLL1_P | ||||||
|         } else if config.hse.is_some() { |         } else if config.hse.is_some() { | ||||||
|             Sw::HSE |             Sw::HSE | ||||||
|         } else { |         } else { | ||||||
|   | |||||||
| @@ -247,7 +247,7 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|  |  | ||||||
|     RCC.cfgr().modify(|w| { |     RCC.cfgr().modify(|w| { | ||||||
|         w.set_sw(if sysclk_on_pll { |         w.set_sw(if sysclk_on_pll { | ||||||
|             Sw::PLL |             Sw::PLL1_P | ||||||
|         } else if config.hse.is_some() { |         } else if config.hse.is_some() { | ||||||
|             Sw::HSE |             Sw::HSE | ||||||
|         } else { |         } else { | ||||||
|   | |||||||
| @@ -131,7 +131,7 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|             RCC.cr().modify(|w| w.set_pllon(true)); |             RCC.cr().modify(|w| w.set_pllon(true)); | ||||||
|             while !RCC.cr().read().pllrdy() {} |             while !RCC.cr().read().pllrdy() {} | ||||||
|  |  | ||||||
|             (freq, Sw::PLL) |             (freq, Sw::PLL1_P) | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -187,12 +187,12 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|  |  | ||||||
|     let sys_clk = match config.mux { |     let sys_clk = match config.mux { | ||||||
|         ClockSrc::HSE => hse.unwrap(), |         ClockSrc::HSE => hse.unwrap(), | ||||||
|         #[cfg(rcc_l5)] |  | ||||||
|         ClockSrc::HSI16 => hsi16.unwrap(), |  | ||||||
|         #[cfg(not(rcc_l5))] |  | ||||||
|         ClockSrc::HSI => hsi16.unwrap(), |         ClockSrc::HSI => hsi16.unwrap(), | ||||||
|         ClockSrc::MSI => msi.unwrap(), |         ClockSrc::MSI => msi.unwrap(), | ||||||
|         ClockSrc::PLL => pll._r.unwrap(), |         #[cfg(rcc_l4)] | ||||||
|  |         ClockSrc::PLL1_P => pll._r.unwrap(), | ||||||
|  |         #[cfg(not(rcc_l4))] | ||||||
|  |         ClockSrc::PLL1_R => pll._r.unwrap(), | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     #[cfg(stm32l4)] |     #[cfg(stm32l4)] | ||||||
| @@ -203,9 +203,6 @@ pub(crate) unsafe fn init(config: Config) { | |||||||
|         Clk48Src::HSI48 => hsi48, |         Clk48Src::HSI48 => hsi48, | ||||||
|         Clk48Src::MSI => msi, |         Clk48Src::MSI => msi, | ||||||
|         Clk48Src::PLLSAI1_Q => pllsai1._q, |         Clk48Src::PLLSAI1_Q => pllsai1._q, | ||||||
|         #[cfg(rcc_l5)] |  | ||||||
|         Clk48Src::PLL_Q => pll._q, |  | ||||||
|         #[cfg(not(rcc_l5))] |  | ||||||
|         Clk48Src::PLL1_Q => pll._q, |         Clk48Src::PLL1_Q => pll._q, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
| @@ -363,9 +360,6 @@ fn init_pll(instance: PllInstance, config: Option<Pll>, input: &PllInput) -> Pll | |||||||
|     let pll_src = match pll.source { |     let pll_src = match pll.source { | ||||||
|         PLLSource::NONE => panic!("must not select PLL source as NONE"), |         PLLSource::NONE => panic!("must not select PLL source as NONE"), | ||||||
|         PLLSource::HSE => input.hse, |         PLLSource::HSE => input.hse, | ||||||
|         #[cfg(rcc_l5)] |  | ||||||
|         PLLSource::HSI16 => input.hsi16, |  | ||||||
|         #[cfg(not(rcc_l5))] |  | ||||||
|         PLLSource::HSI => input.hsi16, |         PLLSource::HSI => input.hsi16, | ||||||
|         PLLSource::MSI => input.msi, |         PLLSource::MSI => input.msi, | ||||||
|     }; |     }; | ||||||
|   | |||||||
| @@ -1,12 +1,7 @@ | |||||||
| MEMORY | MEMORY | ||||||
| { | { | ||||||
|   /* NOTE 1 K = 1 KiBi = 1024 bytes */ |   /* NOTE 1 K = 1 KiBi = 1024 bytes */ | ||||||
|  |   /* These values correspond to the NRF52840 with Softdevices S140 7.0.1 */ | ||||||
|   FLASH : ORIGIN = 0x00000000, LENGTH = 1024K |   FLASH : ORIGIN = 0x00000000, LENGTH = 1024K | ||||||
|   RAM : ORIGIN = 0x20000000, LENGTH = 256K |   RAM : ORIGIN = 0x20000000, LENGTH = 256K | ||||||
|  |  | ||||||
|   /* These values correspond to the NRF52840 with Softdevices S140 7.3.0 */ |  | ||||||
|   /* |  | ||||||
|      FLASH : ORIGIN = 0x00027000, LENGTH = 868K |  | ||||||
|      RAM : ORIGIN = 0x20020000, LENGTH = 128K |  | ||||||
|   */ |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,12 +1,7 @@ | |||||||
| MEMORY | MEMORY | ||||||
| { | { | ||||||
|   /* NOTE 1 K = 1 KiBi = 1024 bytes */ |   /* NOTE 1 K = 1 KiBi = 1024 bytes */ | ||||||
|  |   /* These values correspond to the NRF52840 with Softdevices S140 7.0.1 */ | ||||||
|   FLASH : ORIGIN = 0x00000000, LENGTH = 1024K |   FLASH : ORIGIN = 0x00000000, LENGTH = 1024K | ||||||
|   RAM : ORIGIN = 0x20000000, LENGTH = 256K |   RAM : ORIGIN = 0x20000000, LENGTH = 256K | ||||||
|  |  | ||||||
|   /* These values correspond to the NRF52840 with Softdevices S140 7.3.0 */ |  | ||||||
|   /* |  | ||||||
|      FLASH : ORIGIN = 0x00027000, LENGTH = 868K |  | ||||||
|      RAM : ORIGIN = 0x20020000, LENGTH = 128K |  | ||||||
|   */ |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ bind_interrupts!(struct Irqs { | |||||||
| #[embassy_executor::main] | #[embassy_executor::main] | ||||||
| async fn main(_spawner: Spawner) { | async fn main(_spawner: Spawner) { | ||||||
|     let mut config = Config::default(); |     let mut config = Config::default(); | ||||||
|     config.rcc.mux = ClockSrc::PLL; |     config.rcc.mux = ClockSrc::PLL1_R; | ||||||
|     config.rcc.hsi16 = true; |     config.rcc.hsi16 = true; | ||||||
|     config.rcc.pll = Some(Pll { |     config.rcc.pll = Some(Pll { | ||||||
|         source: PLLSource::HSI, |         source: PLLSource::HSI, | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ use {defmt_rtt as _, panic_probe as _}; | |||||||
| #[embassy_executor::main] | #[embassy_executor::main] | ||||||
| async fn main(_spawner: Spawner) { | async fn main(_spawner: Spawner) { | ||||||
|     let mut config = Config::default(); |     let mut config = Config::default(); | ||||||
|     config.rcc.mux = ClockSrc::PLL; |     config.rcc.mux = ClockSrc::PLL1_R; | ||||||
|     config.rcc.hse = Some(Hertz::mhz(8)); |     config.rcc.hse = Some(Hertz::mhz(8)); | ||||||
|     config.rcc.pll = Some(Pll { |     config.rcc.pll = Some(Pll { | ||||||
|         source: PLLSource::HSE, |         source: PLLSource::HSE, | ||||||
|   | |||||||
| @@ -77,7 +77,7 @@ async fn main(spawner: Spawner) { | |||||||
|  |  | ||||||
|     // 80Mhz clock (Source: 8 / SrcDiv: 1 * PLLMul 20 / ClkDiv 2) |     // 80Mhz clock (Source: 8 / SrcDiv: 1 * PLLMul 20 / ClkDiv 2) | ||||||
|     // 80MHz highest frequency for flash 0 wait. |     // 80MHz highest frequency for flash 0 wait. | ||||||
|     config.rcc.mux = ClockSrc::PLL; |     config.rcc.mux = ClockSrc::PLL1_R; | ||||||
|     config.rcc.hse = Some(Hertz::mhz(8)); |     config.rcc.hse = Some(Hertz::mhz(8)); | ||||||
|     config.rcc.pll = Some(Pll { |     config.rcc.pll = Some(Pll { | ||||||
|         source: PLLSource::HSE, |         source: PLLSource::HSE, | ||||||
|   | |||||||
| @@ -24,7 +24,7 @@ async fn main(_spawner: Spawner) { | |||||||
|  |  | ||||||
|     let mut config = Config::default(); |     let mut config = Config::default(); | ||||||
|     config.rcc.hsi48 = true; |     config.rcc.hsi48 = true; | ||||||
|     config.rcc.mux = ClockSrc::PLL; |     config.rcc.mux = ClockSrc::PLL1_R; | ||||||
|     config.rcc.hsi16 = true; |     config.rcc.hsi16 = true; | ||||||
|     config.rcc.pll = Some(Pll { |     config.rcc.pll = Some(Pll { | ||||||
|         source: PLLSource::HSI, |         source: PLLSource::HSI, | ||||||
|   | |||||||
| @@ -17,10 +17,10 @@ bind_interrupts!(struct Irqs { | |||||||
| async fn main(_spawner: Spawner) { | async fn main(_spawner: Spawner) { | ||||||
|     let mut config = Config::default(); |     let mut config = Config::default(); | ||||||
|     config.rcc.hsi16 = true; |     config.rcc.hsi16 = true; | ||||||
|     config.rcc.mux = ClockSrc::PLL; |     config.rcc.mux = ClockSrc::PLL1_R; | ||||||
|     config.rcc.pll = Some(Pll { |     config.rcc.pll = Some(Pll { | ||||||
|         // 64Mhz clock (16 / 1 * 8 / 2) |         // 64Mhz clock (16 / 1 * 8 / 2) | ||||||
|         source: PLLSource::HSI16, |         source: PLLSource::HSI, | ||||||
|         prediv: PllPreDiv::DIV1, |         prediv: PllPreDiv::DIV1, | ||||||
|         mul: PllMul::MUL8, |         mul: PllMul::MUL8, | ||||||
|         divp: None, |         divp: None, | ||||||
|   | |||||||
| @@ -46,10 +46,10 @@ async fn net_task(stack: &'static Stack<Device<'static, MTU>>) -> ! { | |||||||
| async fn main(spawner: Spawner) { | async fn main(spawner: Spawner) { | ||||||
|     let mut config = Config::default(); |     let mut config = Config::default(); | ||||||
|     config.rcc.hsi16 = true; |     config.rcc.hsi16 = true; | ||||||
|     config.rcc.mux = ClockSrc::PLL; |     config.rcc.mux = ClockSrc::PLL1_R; | ||||||
|     config.rcc.pll = Some(Pll { |     config.rcc.pll = Some(Pll { | ||||||
|         // 80Mhz clock (16 / 1 * 10 / 2) |         // 80Mhz clock (16 / 1 * 10 / 2) | ||||||
|         source: PLLSource::HSI16, |         source: PLLSource::HSI, | ||||||
|         prediv: PllPreDiv::DIV1, |         prediv: PllPreDiv::DIV1, | ||||||
|         mul: PllMul::MUL10, |         mul: PllMul::MUL10, | ||||||
|         divp: None, |         divp: None, | ||||||
|   | |||||||
| @@ -23,10 +23,10 @@ bind_interrupts!(struct Irqs { | |||||||
| async fn main(_spawner: Spawner) { | async fn main(_spawner: Spawner) { | ||||||
|     let mut config = Config::default(); |     let mut config = Config::default(); | ||||||
|     config.rcc.hsi16 = true; |     config.rcc.hsi16 = true; | ||||||
|     config.rcc.mux = ClockSrc::PLL; |     config.rcc.mux = ClockSrc::PLL1_R; | ||||||
|     config.rcc.pll = Some(Pll { |     config.rcc.pll = Some(Pll { | ||||||
|         // 80Mhz clock (16 / 1 * 10 / 2) |         // 80Mhz clock (16 / 1 * 10 / 2) | ||||||
|         source: PLLSource::HSI16, |         source: PLLSource::HSI, | ||||||
|         prediv: PllPreDiv::DIV1, |         prediv: PllPreDiv::DIV1, | ||||||
|         mul: PllMul::MUL10, |         mul: PllMul::MUL10, | ||||||
|         divp: None, |         divp: None, | ||||||
|   | |||||||
| @@ -21,10 +21,10 @@ bind_interrupts!(struct Irqs { | |||||||
| async fn main(_spawner: Spawner) { | async fn main(_spawner: Spawner) { | ||||||
|     let mut config = Config::default(); |     let mut config = Config::default(); | ||||||
|     config.rcc.hsi16 = true; |     config.rcc.hsi16 = true; | ||||||
|     config.rcc.mux = ClockSrc::PLL; |     config.rcc.mux = ClockSrc::PLL1_R; | ||||||
|     config.rcc.pll = Some(Pll { |     config.rcc.pll = Some(Pll { | ||||||
|         // 80Mhz clock (16 / 1 * 10 / 2) |         // 80Mhz clock (16 / 1 * 10 / 2) | ||||||
|         source: PLLSource::HSI16, |         source: PLLSource::HSI, | ||||||
|         prediv: PllPreDiv::DIV1, |         prediv: PllPreDiv::DIV1, | ||||||
|         mul: PllMul::MUL10, |         mul: PllMul::MUL10, | ||||||
|         divp: None, |         divp: None, | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ | |||||||
|  |  | ||||||
| use defmt::*; | use defmt::*; | ||||||
| use embassy_executor::Spawner; | use embassy_executor::Spawner; | ||||||
|  | use embassy_stm32::rcc::{ClockSrc, MSIRange}; | ||||||
| use embassy_stm32::rng::{self, Rng}; | use embassy_stm32::rng::{self, Rng}; | ||||||
| use embassy_stm32::{bind_interrupts, pac, peripherals}; | use embassy_stm32::{bind_interrupts, pac, peripherals}; | ||||||
| use {defmt_rtt as _, panic_probe as _}; | use {defmt_rtt as _, panic_probe as _}; | ||||||
| @@ -15,12 +16,10 @@ bind_interrupts!(struct Irqs{ | |||||||
| #[embassy_executor::main] | #[embassy_executor::main] | ||||||
| async fn main(_spawner: Spawner) { | async fn main(_spawner: Spawner) { | ||||||
|     let mut config = embassy_stm32::Config::default(); |     let mut config = embassy_stm32::Config::default(); | ||||||
|     config.rcc.mux = embassy_stm32::rcc::ClockSrc::HSE; |     config.rcc.mux = ClockSrc::MSI(MSIRange::RANGE32M); | ||||||
|  |  | ||||||
|     let p = embassy_stm32::init(config); |     let p = embassy_stm32::init(config); | ||||||
|     pac::RCC.ccipr().modify(|w| { |  | ||||||
|         w.set_rngsel(0b01); |     pac::RCC.ccipr().modify(|w| w.set_rngsel(0b11)); // msi | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     info!("Hello World!"); |     info!("Hello World!"); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -24,6 +24,7 @@ stm32f767zi = ["embassy-stm32/stm32f767zi", "chrono", "not-gpdma", "eth", "rng"] | |||||||
| stm32f207zg = ["embassy-stm32/stm32f207zg", "chrono", "not-gpdma", "eth", "rng"] | stm32f207zg = ["embassy-stm32/stm32f207zg", "chrono", "not-gpdma", "eth", "rng"] | ||||||
| stm32f303ze = ["embassy-stm32/stm32f303ze", "chrono", "not-gpdma"] | stm32f303ze = ["embassy-stm32/stm32f303ze", "chrono", "not-gpdma"] | ||||||
| stm32l496zg = ["embassy-stm32/stm32l496zg", "not-gpdma", "rng"] | stm32l496zg = ["embassy-stm32/stm32l496zg", "not-gpdma", "rng"] | ||||||
|  | stm32wl55jc = ["embassy-stm32/stm32wl55jc-cm4", "not-gpdma", "rng", "chrono"] | ||||||
|  |  | ||||||
| eth = [] | eth = [] | ||||||
| rng = [] | rng = [] | ||||||
|   | |||||||
| @@ -8,12 +8,14 @@ fn main() -> Result<(), Box<dyn Error>> { | |||||||
|     println!("cargo:rustc-link-search={}", out.display()); |     println!("cargo:rustc-link-search={}", out.display()); | ||||||
|     println!("cargo:rustc-link-arg-bins=--nmagic"); |     println!("cargo:rustc-link-arg-bins=--nmagic"); | ||||||
|  |  | ||||||
|     // too little RAM to run from RAM. |  | ||||||
|     if cfg!(any( |     if cfg!(any( | ||||||
|  |         // too little RAM to run from RAM. | ||||||
|         feature = "stm32f103c8", |         feature = "stm32f103c8", | ||||||
|         feature = "stm32c031c6", |         feature = "stm32c031c6", | ||||||
|         feature = "stm32wb55rg", |         feature = "stm32wb55rg", | ||||||
|         feature = "stm32l073rz", |         feature = "stm32l073rz", | ||||||
|  |         // wrong ram size in stm32-data | ||||||
|  |         feature = "stm32wl55jc", | ||||||
|     )) { |     )) { | ||||||
|         println!("cargo:rustc-link-arg-bins=-Tlink.x"); |         println!("cargo:rustc-link-arg-bins=-Tlink.x"); | ||||||
|         println!("cargo:rerun-if-changed=link.x"); |         println!("cargo:rerun-if-changed=link.x"); | ||||||
|   | |||||||
| @@ -42,6 +42,8 @@ teleprobe_meta::target!(b"nucleo-stm32f207zg"); | |||||||
| teleprobe_meta::target!(b"nucleo-stm32f303ze"); | teleprobe_meta::target!(b"nucleo-stm32f303ze"); | ||||||
| #[cfg(feature = "stm32l496zg")] | #[cfg(feature = "stm32l496zg")] | ||||||
| teleprobe_meta::target!(b"nucleo-stm32l496zg"); | teleprobe_meta::target!(b"nucleo-stm32l496zg"); | ||||||
|  | #[cfg(feature = "stm32wl55jc")] | ||||||
|  | teleprobe_meta::target!(b"nucleo-stm32wl55jc"); | ||||||
|  |  | ||||||
| macro_rules! define_peris { | macro_rules! define_peris { | ||||||
|     ($($name:ident = $peri:ident,)* $(@irq $irq_name:ident = $irq_code:tt,)*) => { |     ($($name:ident = $peri:ident,)* $(@irq $irq_name:ident = $irq_code:tt,)*) => { | ||||||
| @@ -181,6 +183,12 @@ define_peris!( | |||||||
|     SPI = SPI1, SPI_SCK = PA5, SPI_MOSI = PA7, SPI_MISO = PA6, SPI_TX_DMA = DMA1_CH3, SPI_RX_DMA = DMA1_CH2, |     SPI = SPI1, SPI_SCK = PA5, SPI_MOSI = PA7, SPI_MISO = PA6, SPI_TX_DMA = DMA1_CH3, SPI_RX_DMA = DMA1_CH2, | ||||||
|     @irq UART = {USART1 => embassy_stm32::usart::InterruptHandler<embassy_stm32::peripherals::USART1>;}, |     @irq UART = {USART1 => embassy_stm32::usart::InterruptHandler<embassy_stm32::peripherals::USART1>;}, | ||||||
| ); | ); | ||||||
|  | #[cfg(feature = "stm32wl55jc")] | ||||||
|  | define_peris!( | ||||||
|  |     UART = USART1, UART_TX = PB6, UART_RX = PB7, UART_TX_DMA = DMA1_CH4, UART_RX_DMA = DMA1_CH5, | ||||||
|  |     SPI = SPI1, SPI_SCK = PA5, SPI_MOSI = PA7, SPI_MISO = PA6, SPI_TX_DMA = DMA1_CH3, SPI_RX_DMA = DMA1_CH2, | ||||||
|  |     @irq UART = {USART1 => embassy_stm32::usart::InterruptHandler<embassy_stm32::peripherals::USART1>;}, | ||||||
|  | ); | ||||||
|  |  | ||||||
| pub fn config() -> Config { | pub fn config() -> Config { | ||||||
|     #[allow(unused_mut)] |     #[allow(unused_mut)] | ||||||
| @@ -287,7 +295,14 @@ pub fn config() -> Config { | |||||||
|     #[cfg(any(feature = "stm32l496zg", feature = "stm32l4a6zg", feature = "stm32l4r5zi"))] |     #[cfg(any(feature = "stm32l496zg", feature = "stm32l4a6zg", feature = "stm32l4r5zi"))] | ||||||
|     { |     { | ||||||
|         use embassy_stm32::rcc::*; |         use embassy_stm32::rcc::*; | ||||||
|         config.rcc.mux = ClockSrc::PLL; |         #[cfg(feature = "stm32l4r5zi")] | ||||||
|  |         { | ||||||
|  |             config.rcc.mux = ClockSrc::PLL1_R; | ||||||
|  |         } | ||||||
|  |         #[cfg(not(feature = "stm32l4r5zi"))] | ||||||
|  |         { | ||||||
|  |             config.rcc.mux = ClockSrc::PLL1_P; | ||||||
|  |         } | ||||||
|         config.rcc.hsi16 = true; |         config.rcc.hsi16 = true; | ||||||
|         config.rcc.pll = Some(Pll { |         config.rcc.pll = Some(Pll { | ||||||
|             source: PLLSource::HSI, |             source: PLLSource::HSI, | ||||||
| @@ -299,14 +314,23 @@ pub fn config() -> Config { | |||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     #[cfg(feature = "stm32wl55jc")] | ||||||
|  |     { | ||||||
|  |         use embassy_stm32::rcc::*; | ||||||
|  |         config.rcc.mux = ClockSrc::MSI(MSIRange::RANGE32M); | ||||||
|  |         embassy_stm32::pac::RCC.ccipr().modify(|w| { | ||||||
|  |             w.set_rngsel(0b11); // msi | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     #[cfg(any(feature = "stm32l552ze"))] |     #[cfg(any(feature = "stm32l552ze"))] | ||||||
|     { |     { | ||||||
|         use embassy_stm32::rcc::*; |         use embassy_stm32::rcc::*; | ||||||
|         config.rcc.hsi16 = true; |         config.rcc.hsi16 = true; | ||||||
|         config.rcc.mux = ClockSrc::PLL; |         config.rcc.mux = ClockSrc::PLL1_R; | ||||||
|         config.rcc.pll = Some(Pll { |         config.rcc.pll = Some(Pll { | ||||||
|             // 110Mhz clock (16 / 4 * 55 / 2) |             // 110Mhz clock (16 / 4 * 55 / 2) | ||||||
|             source: PLLSource::HSI16, |             source: PLLSource::HSI, | ||||||
|             prediv: PllPreDiv::DIV4, |             prediv: PllPreDiv::DIV4, | ||||||
|             mul: PllMul::MUL55, |             mul: PllMul::MUL55, | ||||||
|             divp: None, |             divp: None, | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user