Compare commits
	
		
			25 Commits
		
	
	
		
			embassy-ex
			...
			update-hea
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 4647df14b1 | ||
|  | b3367be9c8 | ||
|  | da4feb3693 | ||
|  | 05a4bb3a4a | ||
|  | 4b4c28d875 | ||
|  | 1977acdb11 | ||
|  | 0b015bd727 | ||
|  | 79acb560ec | ||
|  | 94586576a0 | ||
|  | a3e200d011 | ||
|  | 3bccb67231 | ||
|  | 553f0158c0 | ||
|  | b44a5fcaf6 | ||
|  | 8effff3383 | ||
|  | 05f983c607 | ||
|  | d44383e9a7 | ||
|  | 37a773c037 | ||
|  | d1adc93614 | ||
|  | 78a7ee7ec4 | ||
|  | e3fe13e905 | ||
|  | 38bfa6916f | ||
|  | 50139752bc | ||
|  | db4cd73894 | ||
|  | 8369f7614a | ||
|  | 326bc98bd2 | 
| @@ -10,7 +10,7 @@ edition = "2021" | |||||||
| # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| heapless = "0.7.16" | heapless = "0.8" | ||||||
| defmt = { version = "0.3", optional = true } | defmt = { version = "0.3", optional = true } | ||||||
| log = { version = "0.4", default-features = false, optional = true } | log = { version = "0.4", default-features = false, optional = true } | ||||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-rc.1" } | embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-rc.1" } | ||||||
|   | |||||||
| @@ -15,9 +15,9 @@ embassy-net-driver-channel = { version = "0.2.0", path = "../embassy-net-driver- | |||||||
| embedded-hal = { version = "1.0.0-rc.1" } | embedded-hal = { version = "1.0.0-rc.1" } | ||||||
| embedded-hal-async = { version = "=1.0.0-rc.1" } | embedded-hal-async = { version = "=1.0.0-rc.1" } | ||||||
|  |  | ||||||
| noproto = { git="https://github.com/embassy-rs/noproto", default-features = false, features = ["derive"] } | noproto = { git="https://github.com/embassy-rs/noproto", rev = "f5e6d1f325b6ad4e344f60452b09576e24671f62", default-features = false, features = ["derive"] } | ||||||
| #noproto = { version = "0.1", path = "/home/dirbaio/noproto", default-features = false, features = ["derive"] } | #noproto = { version = "0.1", path = "/home/dirbaio/noproto", default-features = false, features = ["derive"] } | ||||||
| heapless = "0.7.16" | heapless = "0.8" | ||||||
|  |  | ||||||
| [package.metadata.embassy_docs] | [package.metadata.embassy_docs] | ||||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-net-esp-hosted-v$VERSION/embassy-net-esp-hosted/src/" | src_base = "https://github.com/embassy-rs/embassy/blob/embassy-net-esp-hosted-v$VERSION/embassy-net-esp-hosted/src/" | ||||||
|   | |||||||
| @@ -97,8 +97,8 @@ impl<'a> Control<'a> { | |||||||
|  |  | ||||||
|     pub async fn connect(&mut self, ssid: &str, password: &str) -> Result<(), Error> { |     pub async fn connect(&mut self, ssid: &str, password: &str) -> Result<(), Error> { | ||||||
|         let req = proto::CtrlMsgReqConnectAp { |         let req = proto::CtrlMsgReqConnectAp { | ||||||
|             ssid: String::from(ssid), |             ssid: unwrap!(String::try_from(ssid)), | ||||||
|             pwd: String::from(password), |             pwd: unwrap!(String::try_from(password)), | ||||||
|             bssid: String::new(), |             bssid: String::new(), | ||||||
|             listen_interval: 3, |             listen_interval: 3, | ||||||
|             is_wpa3_supported: false, |             is_wpa3_supported: false, | ||||||
|   | |||||||
| @@ -46,7 +46,7 @@ igmp = ["smoltcp/proto-igmp"] | |||||||
| defmt = { version = "0.3", optional = true } | defmt = { version = "0.3", optional = true } | ||||||
| log = { version = "0.4.14", optional = true } | log = { version = "0.4.14", optional = true } | ||||||
|  |  | ||||||
| smoltcp = { version = "0.10.0", default-features = false, features = [ | smoltcp = { git = "https://github.com/smoltcp-rs/smoltcp", rev = "9b791ae3057e10f7afcb70c67deb5daf714293a9", default-features = false, features = [ | ||||||
|   "socket", |   "socket", | ||||||
|   "async", |   "async", | ||||||
| ] } | ] } | ||||||
| @@ -57,10 +57,10 @@ embassy-sync = { version = "0.4.0", path = "../embassy-sync" } | |||||||
| embedded-io-async = { version = "0.6.0", optional = true } | embedded-io-async = { version = "0.6.0", optional = true } | ||||||
|  |  | ||||||
| managed = { version = "0.8.0", default-features = false, features = [ "map" ] } | managed = { version = "0.8.0", default-features = false, features = [ "map" ] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| as-slice = "0.2.1" | as-slice = "0.2.1" | ||||||
| generic-array = { version = "0.14.4", default-features = false } | generic-array = { version = "0.14.4", default-features = false } | ||||||
| stable_deref_trait = { version = "1.2.0", default-features = false } | stable_deref_trait = { version = "1.2.0", default-features = false } | ||||||
| futures = { version = "0.3.17", default-features = false, features = [ "async-await" ] } | futures = { version = "0.3.17", default-features = false, features = [ "async-await" ] } | ||||||
| atomic-pool = "1.0" | atomic-pool = "1.0" | ||||||
| embedded-nal-async = { version = "0.6.0", optional = true } | embedded-nal-async = { version = "0.7", optional = true } | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ | |||||||
|  |  | ||||||
| It builds on [`smoltcp`](https://github.com/smoltcp-rs/smoltcp). It provides a higher-level and more opinionated | It builds on [`smoltcp`](https://github.com/smoltcp-rs/smoltcp). It provides a higher-level and more opinionated | ||||||
| API. It glues together the components provided by `smoltcp`, handling the low-level details with defaults and | API. It glues together the components provided by `smoltcp`, handling the low-level details with defaults and | ||||||
| memory management designed to work well for embedded systems, aiiming for a more "Just Works" experience. | memory management designed to work well for embedded systems, aiming for a more "Just Works" experience. | ||||||
|  |  | ||||||
| ## Features | ## Features | ||||||
|  |  | ||||||
|   | |||||||
| @@ -101,7 +101,8 @@ where | |||||||
|     async fn get_host_by_address( |     async fn get_host_by_address( | ||||||
|         &self, |         &self, | ||||||
|         _addr: embedded_nal_async::IpAddr, |         _addr: embedded_nal_async::IpAddr, | ||||||
|     ) -> Result<heapless::String<256>, Self::Error> { |         _result: &mut [u8], | ||||||
|  |     ) -> Result<usize, Self::Error> { | ||||||
|         todo!() |         todo!() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -618,10 +618,7 @@ pub mod client { | |||||||
|         async fn connect<'a>( |         async fn connect<'a>( | ||||||
|             &'a self, |             &'a self, | ||||||
|             remote: embedded_nal_async::SocketAddr, |             remote: embedded_nal_async::SocketAddr, | ||||||
|         ) -> Result<Self::Connection<'a>, Self::Error> |         ) -> Result<Self::Connection<'a>, Self::Error> { | ||||||
|         where |  | ||||||
|             Self: 'a, |  | ||||||
|         { |  | ||||||
|             let addr: crate::IpAddress = match remote.ip() { |             let addr: crate::IpAddress = match remote.ip() { | ||||||
|                 #[cfg(feature = "proto-ipv4")] |                 #[cfg(feature = "proto-ipv4")] | ||||||
|                 IpAddr::V4(addr) => crate::IpAddress::Ipv4(crate::Ipv4Address::from_bytes(&addr.octets())), |                 IpAddr::V4(addr) => crate::IpAddress::Ipv4(crate::Ipv4Address::from_bytes(&addr.octets())), | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ embassy-net-driver = { version = "0.2.0", path = "../embassy-net-driver", option | |||||||
|  |  | ||||||
| defmt = { version = "0.3", optional = true } | defmt = { version = "0.3", optional = true } | ||||||
| cortex-m = "0.7.6" | cortex-m = "0.7.6" | ||||||
| heapless = "0.7.16" | heapless = "0.8" | ||||||
| aligned = "0.4.1" | aligned = "0.4.1" | ||||||
|  |  | ||||||
| bit_field = "0.10.2" | bit_field = "0.10.2" | ||||||
|   | |||||||
| @@ -18,7 +18,7 @@ flavors = [ | |||||||
|     { regex_feature = "stm32f7.*", target = "thumbv7em-none-eabi" }, |     { regex_feature = "stm32f7.*", target = "thumbv7em-none-eabi" }, | ||||||
|     { regex_feature = "stm32c0.*", target = "thumbv6m-none-eabi" }, |     { regex_feature = "stm32c0.*", target = "thumbv6m-none-eabi" }, | ||||||
|     { regex_feature = "stm32g0.*", target = "thumbv6m-none-eabi" }, |     { regex_feature = "stm32g0.*", target = "thumbv6m-none-eabi" }, | ||||||
|     { regex_feature = "stm32g4.*", target = "thumbv7em-none-eabi" }, |     { regex_feature = "stm32g4.*", target = "thumbv7em-none-eabi", features = ["low-power"] }, | ||||||
|     { regex_feature = "stm32h5.*", target = "thumbv8m.main-none-eabihf" }, |     { regex_feature = "stm32h5.*", target = "thumbv8m.main-none-eabihf" }, | ||||||
|     { regex_feature = "stm32h7.*", target = "thumbv7em-none-eabi" }, |     { regex_feature = "stm32h7.*", target = "thumbv7em-none-eabi" }, | ||||||
|     { regex_feature = "stm32l0.*", target = "thumbv6m-none-eabi", features = ["low-power"] }, |     { regex_feature = "stm32l0.*", target = "thumbv6m-none-eabi", features = ["low-power"] }, | ||||||
|   | |||||||
| @@ -119,13 +119,11 @@ impl<'a> TDesRing<'a> { | |||||||
|         // "Preceding reads and writes cannot be moved past subsequent writes." |         // "Preceding reads and writes cannot be moved past subsequent writes." | ||||||
|         fence(Ordering::Release); |         fence(Ordering::Release); | ||||||
|  |  | ||||||
|         self.index = self.index + 1; |  | ||||||
|         if self.index == self.descriptors.len() { |  | ||||||
|             self.index = 0; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // signal DMA it can try again. |         // signal DMA it can try again. | ||||||
|         ETH.ethernet_dma().dmactx_dtpr().write(|w| w.0 = 0) |         // See issue #2129 | ||||||
|  |         ETH.ethernet_dma().dmactx_dtpr().write(|w| w.0 = &td as *const _ as u32); | ||||||
|  |  | ||||||
|  |         self.index = (self.index + 1) % self.descriptors.len(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -237,21 +235,19 @@ impl<'a> RDesRing<'a> { | |||||||
|  |  | ||||||
|     /// Pop the packet previously returned by `available`. |     /// Pop the packet previously returned by `available`. | ||||||
|     pub(crate) fn pop_packet(&mut self) { |     pub(crate) fn pop_packet(&mut self) { | ||||||
|         let descriptor = &mut self.descriptors[self.index]; |         let rd = &mut self.descriptors[self.index]; | ||||||
|         assert!(descriptor.available()); |         assert!(rd.available()); | ||||||
|  |  | ||||||
|         self.descriptors[self.index].set_ready(self.buffers[self.index].0.as_mut_ptr()); |         rd.set_ready(self.buffers[self.index].0.as_mut_ptr()); | ||||||
|  |  | ||||||
|         // "Preceding reads and writes cannot be moved past subsequent writes." |         // "Preceding reads and writes cannot be moved past subsequent writes." | ||||||
|         fence(Ordering::Release); |         fence(Ordering::Release); | ||||||
|  |  | ||||||
|         // signal DMA it can try again. |         // signal DMA it can try again. | ||||||
|         ETH.ethernet_dma().dmacrx_dtpr().write(|w| w.0 = 0); |         // See issue #2129 | ||||||
|  |         ETH.ethernet_dma().dmacrx_dtpr().write(|w| w.0 = &rd as *const _ as u32); | ||||||
|  |  | ||||||
|         // Increment index. |         // Increment index. | ||||||
|         self.index += 1; |         self.index = (self.index + 1) % self.descriptors.len(); | ||||||
|         if self.index == self.descriptors.len() { |  | ||||||
|             self.index = 0 |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -33,6 +33,61 @@ use embassy_hal_internal::Peripheral; | |||||||
| use crate::peripherals::RTC; | use crate::peripherals::RTC; | ||||||
| use crate::rtc::sealed::Instance; | use crate::rtc::sealed::Instance; | ||||||
|  |  | ||||||
|  | #[allow(dead_code)] | ||||||
|  | #[repr(u8)] | ||||||
|  | #[derive(Clone, Copy, Debug)] | ||||||
|  | pub(crate) enum WakeupPrescaler { | ||||||
|  |     Div2 = 2, | ||||||
|  |     Div4 = 4, | ||||||
|  |     Div8 = 8, | ||||||
|  |     Div16 = 16, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[cfg(any(stm32wb, stm32f4, stm32l0, stm32g4))] | ||||||
|  | impl From<WakeupPrescaler> for crate::pac::rtc::vals::Wucksel { | ||||||
|  |     fn from(val: WakeupPrescaler) -> Self { | ||||||
|  |         use crate::pac::rtc::vals::Wucksel; | ||||||
|  |  | ||||||
|  |         match val { | ||||||
|  |             WakeupPrescaler::Div2 => Wucksel::DIV2, | ||||||
|  |             WakeupPrescaler::Div4 => Wucksel::DIV4, | ||||||
|  |             WakeupPrescaler::Div8 => Wucksel::DIV8, | ||||||
|  |             WakeupPrescaler::Div16 => Wucksel::DIV16, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[cfg(any(stm32wb, stm32f4, stm32l0, stm32g4))] | ||||||
|  | impl From<crate::pac::rtc::vals::Wucksel> for WakeupPrescaler { | ||||||
|  |     fn from(val: crate::pac::rtc::vals::Wucksel) -> Self { | ||||||
|  |         use crate::pac::rtc::vals::Wucksel; | ||||||
|  |  | ||||||
|  |         match val { | ||||||
|  |             Wucksel::DIV2 => WakeupPrescaler::Div2, | ||||||
|  |             Wucksel::DIV4 => WakeupPrescaler::Div4, | ||||||
|  |             Wucksel::DIV8 => WakeupPrescaler::Div8, | ||||||
|  |             Wucksel::DIV16 => WakeupPrescaler::Div16, | ||||||
|  |             _ => unreachable!(), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[cfg(feature = "low-power")] | ||||||
|  | impl WakeupPrescaler { | ||||||
|  |     pub fn compute_min(val: u32) -> Self { | ||||||
|  |         *[ | ||||||
|  |             WakeupPrescaler::Div2, | ||||||
|  |             WakeupPrescaler::Div4, | ||||||
|  |             WakeupPrescaler::Div8, | ||||||
|  |             WakeupPrescaler::Div16, | ||||||
|  |         ] | ||||||
|  |         .iter() | ||||||
|  |         .skip_while(|psc| **psc as u32 <= val) | ||||||
|  |         .next() | ||||||
|  |         .unwrap_or(&WakeupPrescaler::Div16) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| /// Errors that can occur on methods on [RtcClock] | /// Errors that can occur on methods on [RtcClock] | ||||||
| #[non_exhaustive] | #[non_exhaustive] | ||||||
| #[derive(Clone, Debug, PartialEq, Eq)] | #[derive(Clone, Debug, PartialEq, Eq)] | ||||||
| @@ -277,6 +332,114 @@ impl Rtc { | |||||||
|     pub fn write_backup_register(&self, register: usize, value: u32) { |     pub fn write_backup_register(&self, register: usize, value: u32) { | ||||||
|         RTC::write_backup_register(&RTC::regs(), register, value) |         RTC::write_backup_register(&RTC::regs(), register, value) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     #[cfg(feature = "low-power")] | ||||||
|  |     /// start the wakeup alarm and wtih a duration that is as close to but less than | ||||||
|  |     /// the requested duration, and record the instant the wakeup alarm was started | ||||||
|  |     pub(crate) fn start_wakeup_alarm( | ||||||
|  |         &self, | ||||||
|  |         requested_duration: embassy_time::Duration, | ||||||
|  |         cs: critical_section::CriticalSection, | ||||||
|  |     ) { | ||||||
|  |         use embassy_time::{Duration, TICK_HZ}; | ||||||
|  |  | ||||||
|  |         #[cfg(any(rtc_v3, rtc_v3u5))] | ||||||
|  |         use crate::pac::rtc::vals::Calrf; | ||||||
|  |  | ||||||
|  |         // Panic if the rcc mod knows we're not using low-power rtc | ||||||
|  |         #[cfg(any(rcc_wb, rcc_f4, rcc_f410))] | ||||||
|  |         unsafe { crate::rcc::get_freqs() }.rtc.unwrap(); | ||||||
|  |  | ||||||
|  |         let requested_duration = requested_duration.as_ticks().clamp(0, u32::MAX as u64); | ||||||
|  |         let rtc_hz = Self::frequency().0 as u64; | ||||||
|  |         let rtc_ticks = requested_duration * rtc_hz / TICK_HZ; | ||||||
|  |         let prescaler = WakeupPrescaler::compute_min((rtc_ticks / u16::MAX as u64) as u32); | ||||||
|  |  | ||||||
|  |         // adjust the rtc ticks to the prescaler and subtract one rtc tick | ||||||
|  |         let rtc_ticks = rtc_ticks / prescaler as u64; | ||||||
|  |         let rtc_ticks = rtc_ticks.clamp(0, (u16::MAX - 1) as u64).saturating_sub(1) as u16; | ||||||
|  |  | ||||||
|  |         self.write(false, |regs| { | ||||||
|  |             regs.cr().modify(|w| w.set_wute(false)); | ||||||
|  |  | ||||||
|  |             #[cfg(any( | ||||||
|  |                 rtc_v2f0, rtc_v2f2, rtc_v2f3, rtc_v2f4, rtc_v2f7, rtc_v2h7, rtc_v2l0, rtc_v2l1, rtc_v2l4, rtc_v2wb | ||||||
|  |             ))] | ||||||
|  |             { | ||||||
|  |                 regs.isr().modify(|w| w.set_wutf(false)); | ||||||
|  |                 while !regs.isr().read().wutwf() {} | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             #[cfg(any(rtc_v3, rtc_v3u5))] | ||||||
|  |             { | ||||||
|  |                 regs.scr().write(|w| w.set_cwutf(Calrf::CLEAR)); | ||||||
|  |                 while !regs.icsr().read().wutwf() {} | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             regs.cr().modify(|w| w.set_wucksel(prescaler.into())); | ||||||
|  |             regs.wutr().write(|w| w.set_wut(rtc_ticks)); | ||||||
|  |             regs.cr().modify(|w| w.set_wute(true)); | ||||||
|  |             regs.cr().modify(|w| w.set_wutie(true)); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         let instant = self.instant().unwrap(); | ||||||
|  |         trace!( | ||||||
|  |             "rtc: start wakeup alarm for {} ms (psc: {}, ticks: {}) at {}", | ||||||
|  |             Duration::from_ticks(rtc_ticks as u64 * TICK_HZ * prescaler as u64 / rtc_hz).as_millis(), | ||||||
|  |             prescaler as u32, | ||||||
|  |             rtc_ticks, | ||||||
|  |             instant, | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         assert!(self.stop_time.borrow(cs).replace(Some(instant)).is_none()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[cfg(feature = "low-power")] | ||||||
|  |     /// stop the wakeup alarm and return the time elapsed since `start_wakeup_alarm` | ||||||
|  |     /// was called, otherwise none | ||||||
|  |     pub(crate) fn stop_wakeup_alarm(&self, cs: critical_section::CriticalSection) -> Option<embassy_time::Duration> { | ||||||
|  |         use crate::interrupt::typelevel::Interrupt; | ||||||
|  |         #[cfg(any(rtc_v3, rtc_v3u5))] | ||||||
|  |         use crate::pac::rtc::vals::Calrf; | ||||||
|  |  | ||||||
|  |         let instant = self.instant().unwrap(); | ||||||
|  |         if RTC::regs().cr().read().wute() { | ||||||
|  |             trace!("rtc: stop wakeup alarm at {}", instant); | ||||||
|  |  | ||||||
|  |             self.write(false, |regs| { | ||||||
|  |                 regs.cr().modify(|w| w.set_wutie(false)); | ||||||
|  |                 regs.cr().modify(|w| w.set_wute(false)); | ||||||
|  |  | ||||||
|  |                 #[cfg(any( | ||||||
|  |                     rtc_v2f0, rtc_v2f2, rtc_v2f3, rtc_v2f4, rtc_v2f7, rtc_v2h7, rtc_v2l0, rtc_v2l1, rtc_v2l4, rtc_v2wb | ||||||
|  |                 ))] | ||||||
|  |                 regs.isr().modify(|w| w.set_wutf(false)); | ||||||
|  |  | ||||||
|  |                 #[cfg(any(rtc_v3, rtc_v3u5))] | ||||||
|  |                 regs.scr().write(|w| w.set_cwutf(Calrf::CLEAR)); | ||||||
|  |  | ||||||
|  |                 crate::pac::EXTI | ||||||
|  |                     .pr(0) | ||||||
|  |                     .modify(|w| w.set_line(RTC::EXTI_WAKEUP_LINE, true)); | ||||||
|  |  | ||||||
|  |                 <RTC as crate::rtc::sealed::Instance>::WakeupInterrupt::unpend(); | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         self.stop_time.borrow(cs).take().map(|stop_time| instant - stop_time) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[cfg(feature = "low-power")] | ||||||
|  |     pub(crate) fn enable_wakeup_line(&self) { | ||||||
|  |         use crate::interrupt::typelevel::Interrupt; | ||||||
|  |         use crate::pac::EXTI; | ||||||
|  |  | ||||||
|  |         <RTC as crate::rtc::sealed::Instance>::WakeupInterrupt::unpend(); | ||||||
|  |         unsafe { <RTC as crate::rtc::sealed::Instance>::WakeupInterrupt::enable() }; | ||||||
|  |  | ||||||
|  |         EXTI.rtsr(0).modify(|w| w.set_line(RTC::EXTI_WAKEUP_LINE, true)); | ||||||
|  |         EXTI.imr(0).modify(|w| w.set_line(RTC::EXTI_WAKEUP_LINE, true)); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| pub(crate) fn byte_to_bcd2(byte: u8) -> (u8, u8) { | pub(crate) fn byte_to_bcd2(byte: u8) -> (u8, u8) { | ||||||
|   | |||||||
| @@ -6,145 +6,7 @@ use crate::peripherals::RTC; | |||||||
| use crate::rtc::sealed::Instance; | use crate::rtc::sealed::Instance; | ||||||
|  |  | ||||||
| #[allow(dead_code)] | #[allow(dead_code)] | ||||||
| #[repr(u8)] |  | ||||||
| #[derive(Clone, Copy, Debug)] |  | ||||||
| pub(crate) enum WakeupPrescaler { |  | ||||||
|     Div2 = 2, |  | ||||||
|     Div4 = 4, |  | ||||||
|     Div8 = 8, |  | ||||||
|     Div16 = 16, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[cfg(any(stm32wb, stm32f4, stm32l0))] |  | ||||||
| impl From<WakeupPrescaler> for crate::pac::rtc::vals::Wucksel { |  | ||||||
|     fn from(val: WakeupPrescaler) -> Self { |  | ||||||
|         use crate::pac::rtc::vals::Wucksel; |  | ||||||
|  |  | ||||||
|         match val { |  | ||||||
|             WakeupPrescaler::Div2 => Wucksel::DIV2, |  | ||||||
|             WakeupPrescaler::Div4 => Wucksel::DIV4, |  | ||||||
|             WakeupPrescaler::Div8 => Wucksel::DIV8, |  | ||||||
|             WakeupPrescaler::Div16 => Wucksel::DIV16, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[cfg(any(stm32wb, stm32f4, stm32l0))] |  | ||||||
| impl From<crate::pac::rtc::vals::Wucksel> for WakeupPrescaler { |  | ||||||
|     fn from(val: crate::pac::rtc::vals::Wucksel) -> Self { |  | ||||||
|         use crate::pac::rtc::vals::Wucksel; |  | ||||||
|  |  | ||||||
|         match val { |  | ||||||
|             Wucksel::DIV2 => WakeupPrescaler::Div2, |  | ||||||
|             Wucksel::DIV4 => WakeupPrescaler::Div4, |  | ||||||
|             Wucksel::DIV8 => WakeupPrescaler::Div8, |  | ||||||
|             Wucksel::DIV16 => WakeupPrescaler::Div16, |  | ||||||
|             _ => unreachable!(), |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[allow(dead_code)] |  | ||||||
| impl WakeupPrescaler { |  | ||||||
|     pub fn compute_min(val: u32) -> Self { |  | ||||||
|         *[ |  | ||||||
|             WakeupPrescaler::Div2, |  | ||||||
|             WakeupPrescaler::Div4, |  | ||||||
|             WakeupPrescaler::Div8, |  | ||||||
|             WakeupPrescaler::Div16, |  | ||||||
|         ] |  | ||||||
|         .iter() |  | ||||||
|         .skip_while(|psc| **psc as u32 <= val) |  | ||||||
|         .next() |  | ||||||
|         .unwrap_or(&WakeupPrescaler::Div16) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl super::Rtc { | impl super::Rtc { | ||||||
|     #[cfg(feature = "low-power")] |  | ||||||
|     /// start the wakeup alarm and wtih a duration that is as close to but less than |  | ||||||
|     /// the requested duration, and record the instant the wakeup alarm was started |  | ||||||
|     pub(crate) fn start_wakeup_alarm( |  | ||||||
|         &self, |  | ||||||
|         requested_duration: embassy_time::Duration, |  | ||||||
|         cs: critical_section::CriticalSection, |  | ||||||
|     ) { |  | ||||||
|         use embassy_time::{Duration, TICK_HZ}; |  | ||||||
|  |  | ||||||
|         // Panic if the rcc mod knows we're not using low-power rtc |  | ||||||
|         #[cfg(any(rcc_wb, rcc_f4, rcc_f410))] |  | ||||||
|         unsafe { crate::rcc::get_freqs() }.rtc.unwrap(); |  | ||||||
|  |  | ||||||
|         let requested_duration = requested_duration.as_ticks().clamp(0, u32::MAX as u64); |  | ||||||
|         let rtc_hz = Self::frequency().0 as u64; |  | ||||||
|         let rtc_ticks = requested_duration * rtc_hz / TICK_HZ; |  | ||||||
|         let prescaler = WakeupPrescaler::compute_min((rtc_ticks / u16::MAX as u64) as u32); |  | ||||||
|  |  | ||||||
|         // adjust the rtc ticks to the prescaler and subtract one rtc tick |  | ||||||
|         let rtc_ticks = rtc_ticks / prescaler as u64; |  | ||||||
|         let rtc_ticks = rtc_ticks.clamp(0, (u16::MAX - 1) as u64).saturating_sub(1) as u16; |  | ||||||
|  |  | ||||||
|         self.write(false, |regs| { |  | ||||||
|             regs.cr().modify(|w| w.set_wute(false)); |  | ||||||
|             regs.isr().modify(|w| w.set_wutf(false)); |  | ||||||
|             while !regs.isr().read().wutwf() {} |  | ||||||
|  |  | ||||||
|             regs.cr().modify(|w| w.set_wucksel(prescaler.into())); |  | ||||||
|             regs.wutr().write(|w| w.set_wut(rtc_ticks)); |  | ||||||
|             regs.cr().modify(|w| w.set_wute(true)); |  | ||||||
|             regs.cr().modify(|w| w.set_wutie(true)); |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         let instant = self.instant().unwrap(); |  | ||||||
|         trace!( |  | ||||||
|             "rtc: start wakeup alarm for {} ms (psc: {}, ticks: {}) at {}", |  | ||||||
|             Duration::from_ticks(rtc_ticks as u64 * TICK_HZ * prescaler as u64 / rtc_hz).as_millis(), |  | ||||||
|             prescaler as u32, |  | ||||||
|             rtc_ticks, |  | ||||||
|             instant, |  | ||||||
|         ); |  | ||||||
|  |  | ||||||
|         assert!(self.stop_time.borrow(cs).replace(Some(instant)).is_none()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     #[cfg(feature = "low-power")] |  | ||||||
|     /// stop the wakeup alarm and return the time elapsed since `start_wakeup_alarm` |  | ||||||
|     /// was called, otherwise none |  | ||||||
|     pub(crate) fn stop_wakeup_alarm(&self, cs: critical_section::CriticalSection) -> Option<embassy_time::Duration> { |  | ||||||
|         use crate::interrupt::typelevel::Interrupt; |  | ||||||
|  |  | ||||||
|         let instant = self.instant().unwrap(); |  | ||||||
|         if RTC::regs().cr().read().wute() { |  | ||||||
|             trace!("rtc: stop wakeup alarm at {}", instant); |  | ||||||
|  |  | ||||||
|             self.write(false, |regs| { |  | ||||||
|                 regs.cr().modify(|w| w.set_wutie(false)); |  | ||||||
|                 regs.cr().modify(|w| w.set_wute(false)); |  | ||||||
|                 regs.isr().modify(|w| w.set_wutf(false)); |  | ||||||
|  |  | ||||||
|                 crate::pac::EXTI |  | ||||||
|                     .pr(0) |  | ||||||
|                     .modify(|w| w.set_line(RTC::EXTI_WAKEUP_LINE, true)); |  | ||||||
|  |  | ||||||
|                 <RTC as crate::rtc::sealed::Instance>::WakeupInterrupt::unpend(); |  | ||||||
|             }); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         self.stop_time.borrow(cs).take().map(|stop_time| instant - stop_time) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     #[cfg(feature = "low-power")] |  | ||||||
|     pub(crate) fn enable_wakeup_line(&self) { |  | ||||||
|         use crate::interrupt::typelevel::Interrupt; |  | ||||||
|         use crate::pac::EXTI; |  | ||||||
|  |  | ||||||
|         <RTC as crate::rtc::sealed::Instance>::WakeupInterrupt::unpend(); |  | ||||||
|         unsafe { <RTC as crate::rtc::sealed::Instance>::WakeupInterrupt::enable() }; |  | ||||||
|  |  | ||||||
|         EXTI.rtsr(0).modify(|w| w.set_line(RTC::EXTI_WAKEUP_LINE, true)); |  | ||||||
|         EXTI.imr(0).modify(|w| w.set_line(RTC::EXTI_WAKEUP_LINE, true)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /// Applies the RTC config |     /// Applies the RTC config | ||||||
|     /// It this changes the RTC clock source the time will be reset |     /// It this changes the RTC clock source the time will be reset | ||||||
|     pub(super) fn configure(&mut self, async_psc: u8, sync_psc: u16) { |     pub(super) fn configure(&mut self, async_psc: u8, sync_psc: u16) { | ||||||
|   | |||||||
| @@ -95,7 +95,7 @@ impl super::Rtc { | |||||||
|         }) |         }) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub(super) fn write<F, R>(&mut self, init_mode: bool, f: F) -> R |     pub(super) fn write<F, R>(&self, init_mode: bool, f: F) -> R | ||||||
|     where |     where | ||||||
|         F: FnOnce(&crate::pac::rtc::Rtc) -> R, |         F: FnOnce(&crate::pac::rtc::Rtc) -> R, | ||||||
|     { |     { | ||||||
| @@ -129,6 +129,12 @@ impl super::Rtc { | |||||||
| impl sealed::Instance for crate::peripherals::RTC { | impl sealed::Instance for crate::peripherals::RTC { | ||||||
|     const BACKUP_REGISTER_COUNT: usize = 32; |     const BACKUP_REGISTER_COUNT: usize = 32; | ||||||
|  |  | ||||||
|  |     #[cfg(all(feature = "low-power", stm32g4))] | ||||||
|  |     const EXTI_WAKEUP_LINE: usize = 20; | ||||||
|  |  | ||||||
|  |     #[cfg(all(feature = "low-power", stm32g4))] | ||||||
|  |     type WakeupInterrupt = crate::interrupt::typelevel::RTC_WKUP; | ||||||
|  |  | ||||||
|     fn read_backup_register(_rtc: &Rtc, register: usize) -> Option<u32> { |     fn read_backup_register(_rtc: &Rtc, register: usize) -> Option<u32> { | ||||||
|         #[allow(clippy::if_same_then_else)] |         #[allow(clippy::if_same_then_else)] | ||||||
|         if register < Self::BACKUP_REGISTER_COUNT { |         if register < Self::BACKUP_REGISTER_COUNT { | ||||||
|   | |||||||
| @@ -33,7 +33,7 @@ log = { version = "0.4.14", optional = true } | |||||||
|  |  | ||||||
| futures-util = { version = "0.3.17", default-features = false } | futures-util = { version = "0.3.17", default-features = false } | ||||||
| critical-section = "1.1" | critical-section = "1.1" | ||||||
| heapless = "0.7.5" | heapless = "0.8" | ||||||
| cfg-if = "1.0.0" | cfg-if = "1.0.0" | ||||||
| embedded-io-async = { version = "0.6.0", optional = true } | embedded-io-async = { version = "0.6.0", optional = true } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -248,7 +248,7 @@ embedded-hal-async = { version = "=1.0.0-rc.1", optional = true} | |||||||
| futures-util = { version = "0.3.17", default-features = false } | futures-util = { version = "0.3.17", default-features = false } | ||||||
| critical-section = "1.1" | critical-section = "1.1" | ||||||
| cfg-if = "1.0.0" | cfg-if = "1.0.0" | ||||||
| heapless = "0.7" | heapless = "0.8" | ||||||
|  |  | ||||||
| # WASM dependencies | # WASM dependencies | ||||||
| wasm-bindgen = { version = "0.2.81", optional = true } | wasm-bindgen = { version = "0.2.81", optional = true } | ||||||
|   | |||||||
| @@ -19,6 +19,7 @@ pub struct LoggerState<'d> { | |||||||
|     device_descriptor: [u8; 32], |     device_descriptor: [u8; 32], | ||||||
|     config_descriptor: [u8; 128], |     config_descriptor: [u8; 128], | ||||||
|     bos_descriptor: [u8; 16], |     bos_descriptor: [u8; 16], | ||||||
|  |     msos_descriptor: [u8; 256], | ||||||
|     control_buf: [u8; 64], |     control_buf: [u8; 64], | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -30,6 +31,7 @@ impl<'d> LoggerState<'d> { | |||||||
|             device_descriptor: [0; 32], |             device_descriptor: [0; 32], | ||||||
|             config_descriptor: [0; 128], |             config_descriptor: [0; 128], | ||||||
|             bos_descriptor: [0; 16], |             bos_descriptor: [0; 16], | ||||||
|  |             msos_descriptor: [0; 256], | ||||||
|             control_buf: [0; 64], |             control_buf: [0; 64], | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -73,6 +75,7 @@ impl<const N: usize> UsbLogger<N> { | |||||||
|             &mut state.device_descriptor, |             &mut state.device_descriptor, | ||||||
|             &mut state.config_descriptor, |             &mut state.config_descriptor, | ||||||
|             &mut state.bos_descriptor, |             &mut state.bos_descriptor, | ||||||
|  |             &mut state.msos_descriptor, | ||||||
|             &mut state.control_buf, |             &mut state.control_buf, | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,7 +13,6 @@ target = "thumbv7em-none-eabi" | |||||||
| [features] | [features] | ||||||
| defmt = ["dep:defmt", "embassy-usb-driver/defmt"] | defmt = ["dep:defmt", "embassy-usb-driver/defmt"] | ||||||
| usbd-hid = ["dep:usbd-hid", "dep:ssmarshal"] | usbd-hid = ["dep:usbd-hid", "dep:ssmarshal"] | ||||||
| msos-descriptor = [] |  | ||||||
| default = ["usbd-hid"] | default = ["usbd-hid"] | ||||||
|  |  | ||||||
| # BEGIN AUTOGENERATED CONFIG FEATURES | # BEGIN AUTOGENERATED CONFIG FEATURES | ||||||
| @@ -46,7 +45,7 @@ embassy-net-driver-channel = { version = "0.2.0", path = "../embassy-net-driver- | |||||||
|  |  | ||||||
| defmt = { version = "0.3", optional = true } | defmt = { version = "0.3", optional = true } | ||||||
| log = { version = "0.4.14", optional = true } | log = { version = "0.4.14", optional = true } | ||||||
| heapless = "0.7.10" | heapless = "0.8" | ||||||
|  |  | ||||||
| # for HID | # for HID | ||||||
| usbd-hid = { version = "0.6.0", optional = true } | usbd-hid = { version = "0.6.0", optional = true } | ||||||
|   | |||||||
| @@ -3,7 +3,6 @@ use heapless::Vec; | |||||||
| use crate::config::MAX_HANDLER_COUNT; | use crate::config::MAX_HANDLER_COUNT; | ||||||
| use crate::descriptor::{BosWriter, DescriptorWriter}; | use crate::descriptor::{BosWriter, DescriptorWriter}; | ||||||
| use crate::driver::{Driver, Endpoint, EndpointType}; | use crate::driver::{Driver, Endpoint, EndpointType}; | ||||||
| #[cfg(feature = "msos-descriptor")] |  | ||||||
| use crate::msos::{DeviceLevelDescriptor, FunctionLevelDescriptor, MsOsDescriptorWriter}; | use crate::msos::{DeviceLevelDescriptor, FunctionLevelDescriptor, MsOsDescriptorWriter}; | ||||||
| use crate::types::{InterfaceNumber, StringIndex}; | use crate::types::{InterfaceNumber, StringIndex}; | ||||||
| use crate::{Handler, Interface, UsbDevice, MAX_INTERFACE_COUNT, STRING_INDEX_CUSTOM_START}; | use crate::{Handler, Interface, UsbDevice, MAX_INTERFACE_COUNT, STRING_INDEX_CUSTOM_START}; | ||||||
| @@ -133,7 +132,6 @@ pub struct Builder<'d, D: Driver<'d>> { | |||||||
|     config_descriptor: DescriptorWriter<'d>, |     config_descriptor: DescriptorWriter<'d>, | ||||||
|     bos_descriptor: BosWriter<'d>, |     bos_descriptor: BosWriter<'d>, | ||||||
|  |  | ||||||
|     #[cfg(feature = "msos-descriptor")] |  | ||||||
|     msos_descriptor: MsOsDescriptorWriter<'d>, |     msos_descriptor: MsOsDescriptorWriter<'d>, | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -149,7 +147,7 @@ impl<'d, D: Driver<'d>> Builder<'d, D> { | |||||||
|         device_descriptor_buf: &'d mut [u8], |         device_descriptor_buf: &'d mut [u8], | ||||||
|         config_descriptor_buf: &'d mut [u8], |         config_descriptor_buf: &'d mut [u8], | ||||||
|         bos_descriptor_buf: &'d mut [u8], |         bos_descriptor_buf: &'d mut [u8], | ||||||
|         #[cfg(feature = "msos-descriptor")] msos_descriptor_buf: &'d mut [u8], |         msos_descriptor_buf: &'d mut [u8], | ||||||
|         control_buf: &'d mut [u8], |         control_buf: &'d mut [u8], | ||||||
|     ) -> Self { |     ) -> Self { | ||||||
|         // Magic values specified in USB-IF ECN on IADs. |         // Magic values specified in USB-IF ECN on IADs. | ||||||
| @@ -189,14 +187,12 @@ impl<'d, D: Driver<'d>> Builder<'d, D> { | |||||||
|             config_descriptor, |             config_descriptor, | ||||||
|             bos_descriptor, |             bos_descriptor, | ||||||
|  |  | ||||||
|             #[cfg(feature = "msos-descriptor")] |  | ||||||
|             msos_descriptor: MsOsDescriptorWriter::new(msos_descriptor_buf), |             msos_descriptor: MsOsDescriptorWriter::new(msos_descriptor_buf), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /// Creates the [`UsbDevice`] instance with the configuration in this builder. |     /// Creates the [`UsbDevice`] instance with the configuration in this builder. | ||||||
|     pub fn build(mut self) -> UsbDevice<'d, D> { |     pub fn build(mut self) -> UsbDevice<'d, D> { | ||||||
|         #[cfg(feature = "msos-descriptor")] |  | ||||||
|         let msos_descriptor = self.msos_descriptor.build(&mut self.bos_descriptor); |         let msos_descriptor = self.msos_descriptor.build(&mut self.bos_descriptor); | ||||||
|  |  | ||||||
|         self.config_descriptor.end_configuration(); |         self.config_descriptor.end_configuration(); | ||||||
| @@ -206,7 +202,6 @@ impl<'d, D: Driver<'d>> Builder<'d, D> { | |||||||
|         info!("USB: device_descriptor used: {}", self.device_descriptor.position()); |         info!("USB: device_descriptor used: {}", self.device_descriptor.position()); | ||||||
|         info!("USB: config_descriptor used: {}", self.config_descriptor.position()); |         info!("USB: config_descriptor used: {}", self.config_descriptor.position()); | ||||||
|         info!("USB: bos_descriptor used: {}", self.bos_descriptor.writer.position()); |         info!("USB: bos_descriptor used: {}", self.bos_descriptor.writer.position()); | ||||||
|         #[cfg(feature = "msos-descriptor")] |  | ||||||
|         info!("USB: msos_descriptor used: {}", msos_descriptor.len()); |         info!("USB: msos_descriptor used: {}", msos_descriptor.len()); | ||||||
|         info!("USB: control_buf size: {}", self.control_buf.len()); |         info!("USB: control_buf size: {}", self.control_buf.len()); | ||||||
|  |  | ||||||
| @@ -217,10 +212,9 @@ impl<'d, D: Driver<'d>> Builder<'d, D> { | |||||||
|             self.device_descriptor.into_buf(), |             self.device_descriptor.into_buf(), | ||||||
|             self.config_descriptor.into_buf(), |             self.config_descriptor.into_buf(), | ||||||
|             self.bos_descriptor.writer.into_buf(), |             self.bos_descriptor.writer.into_buf(), | ||||||
|  |             msos_descriptor, | ||||||
|             self.interfaces, |             self.interfaces, | ||||||
|             self.control_buf, |             self.control_buf, | ||||||
|             #[cfg(feature = "msos-descriptor")] |  | ||||||
|             msos_descriptor, |  | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -251,7 +245,6 @@ impl<'d, D: Driver<'d>> Builder<'d, D> { | |||||||
|             builder: self, |             builder: self, | ||||||
|             iface_count_index, |             iface_count_index, | ||||||
|  |  | ||||||
|             #[cfg(feature = "msos-descriptor")] |  | ||||||
|             first_interface, |             first_interface, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -275,7 +268,6 @@ impl<'d, D: Driver<'d>> Builder<'d, D> { | |||||||
|         StringIndex::new(index) |         StringIndex::new(index) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[cfg(feature = "msos-descriptor")] |  | ||||||
|     /// Add an MS OS 2.0 Descriptor Set. |     /// Add an MS OS 2.0 Descriptor Set. | ||||||
|     /// |     /// | ||||||
|     /// Panics if called more than once. |     /// Panics if called more than once. | ||||||
| @@ -283,13 +275,11 @@ impl<'d, D: Driver<'d>> Builder<'d, D> { | |||||||
|         self.msos_descriptor.header(windows_version, vendor_code); |         self.msos_descriptor.header(windows_version, vendor_code); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[cfg(feature = "msos-descriptor")] |  | ||||||
|     /// Add an MS OS 2.0 Device Level Feature Descriptor. |     /// Add an MS OS 2.0 Device Level Feature Descriptor. | ||||||
|     pub fn msos_feature<T: DeviceLevelDescriptor>(&mut self, desc: T) { |     pub fn msos_feature<T: DeviceLevelDescriptor>(&mut self, desc: T) { | ||||||
|         self.msos_descriptor.device_feature(desc); |         self.msos_descriptor.device_feature(desc); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[cfg(feature = "msos-descriptor")] |  | ||||||
|     /// Gets the underlying [`MsOsDescriptorWriter`] to allow adding subsets and features for classes that |     /// Gets the underlying [`MsOsDescriptorWriter`] to allow adding subsets and features for classes that | ||||||
|     /// do not add their own. |     /// do not add their own. | ||||||
|     pub fn msos_writer(&mut self) -> &mut MsOsDescriptorWriter<'d> { |     pub fn msos_writer(&mut self) -> &mut MsOsDescriptorWriter<'d> { | ||||||
| @@ -306,13 +296,11 @@ pub struct FunctionBuilder<'a, 'd, D: Driver<'d>> { | |||||||
|     builder: &'a mut Builder<'d, D>, |     builder: &'a mut Builder<'d, D>, | ||||||
|     iface_count_index: Option<usize>, |     iface_count_index: Option<usize>, | ||||||
|  |  | ||||||
|     #[cfg(feature = "msos-descriptor")] |  | ||||||
|     first_interface: InterfaceNumber, |     first_interface: InterfaceNumber, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'a, 'd, D: Driver<'d>> Drop for FunctionBuilder<'a, 'd, D> { | impl<'a, 'd, D: Driver<'d>> Drop for FunctionBuilder<'a, 'd, D> { | ||||||
|     fn drop(&mut self) { |     fn drop(&mut self) { | ||||||
|         #[cfg(feature = "msos-descriptor")] |  | ||||||
|         self.builder.msos_descriptor.end_function(); |         self.builder.msos_descriptor.end_function(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -344,7 +332,6 @@ impl<'a, 'd, D: Driver<'d>> FunctionBuilder<'a, 'd, D> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[cfg(feature = "msos-descriptor")] |  | ||||||
|     /// Add an MS OS 2.0 Function Level Feature Descriptor. |     /// Add an MS OS 2.0 Function Level Feature Descriptor. | ||||||
|     pub fn msos_feature<T: FunctionLevelDescriptor>(&mut self, desc: T) { |     pub fn msos_feature<T: FunctionLevelDescriptor>(&mut self, desc: T) { | ||||||
|         if !self.builder.msos_descriptor.is_in_config_subset() { |         if !self.builder.msos_descriptor.is_in_config_subset() { | ||||||
| @@ -355,7 +342,6 @@ impl<'a, 'd, D: Driver<'d>> FunctionBuilder<'a, 'd, D> { | |||||||
|             self.builder.msos_descriptor.function(self.first_interface); |             self.builder.msos_descriptor.function(self.first_interface); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         #[cfg(feature = "msos-descriptor")] |  | ||||||
|         self.builder.msos_descriptor.function_feature(desc); |         self.builder.msos_descriptor.function_feature(desc); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -175,10 +175,7 @@ pub struct UsbBufferReport { | |||||||
|     /// Number of bos descriptor bytes used |     /// Number of bos descriptor bytes used | ||||||
|     pub bos_descriptor_used: usize, |     pub bos_descriptor_used: usize, | ||||||
|     /// Number of msos descriptor bytes used |     /// Number of msos descriptor bytes used | ||||||
|     /// |     pub msos_descriptor_used: usize, | ||||||
|     /// Will be `None` if the "msos-descriptor" feature is not active. |  | ||||||
|     /// Otherwise will return Some(bytes). |  | ||||||
|     pub msos_descriptor_used: Option<usize>, |  | ||||||
|     /// Size of the control buffer |     /// Size of the control buffer | ||||||
|     pub control_buffer_size: usize, |     pub control_buffer_size: usize, | ||||||
| } | } | ||||||
| @@ -197,6 +194,7 @@ struct Inner<'d, D: Driver<'d>> { | |||||||
|     device_descriptor: &'d [u8], |     device_descriptor: &'d [u8], | ||||||
|     config_descriptor: &'d [u8], |     config_descriptor: &'d [u8], | ||||||
|     bos_descriptor: &'d [u8], |     bos_descriptor: &'d [u8], | ||||||
|  |     msos_descriptor: crate::msos::MsOsDescriptorSet<'d>, | ||||||
|  |  | ||||||
|     device_state: UsbDeviceState, |     device_state: UsbDeviceState, | ||||||
|     suspended: bool, |     suspended: bool, | ||||||
| @@ -212,9 +210,6 @@ struct Inner<'d, D: Driver<'d>> { | |||||||
|  |  | ||||||
|     interfaces: Vec<Interface, MAX_INTERFACE_COUNT>, |     interfaces: Vec<Interface, MAX_INTERFACE_COUNT>, | ||||||
|     handlers: Vec<&'d mut dyn Handler, MAX_HANDLER_COUNT>, |     handlers: Vec<&'d mut dyn Handler, MAX_HANDLER_COUNT>, | ||||||
|  |  | ||||||
|     #[cfg(feature = "msos-descriptor")] |  | ||||||
|     msos_descriptor: crate::msos::MsOsDescriptorSet<'d>, |  | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'d, D: Driver<'d>> UsbDevice<'d, D> { | impl<'d, D: Driver<'d>> UsbDevice<'d, D> { | ||||||
| @@ -225,9 +220,9 @@ impl<'d, D: Driver<'d>> UsbDevice<'d, D> { | |||||||
|         device_descriptor: &'d [u8], |         device_descriptor: &'d [u8], | ||||||
|         config_descriptor: &'d [u8], |         config_descriptor: &'d [u8], | ||||||
|         bos_descriptor: &'d [u8], |         bos_descriptor: &'d [u8], | ||||||
|  |         msos_descriptor: crate::msos::MsOsDescriptorSet<'d>, | ||||||
|         interfaces: Vec<Interface, MAX_INTERFACE_COUNT>, |         interfaces: Vec<Interface, MAX_INTERFACE_COUNT>, | ||||||
|         control_buf: &'d mut [u8], |         control_buf: &'d mut [u8], | ||||||
|         #[cfg(feature = "msos-descriptor")] msos_descriptor: crate::msos::MsOsDescriptorSet<'d>, |  | ||||||
|     ) -> UsbDevice<'d, D> { |     ) -> UsbDevice<'d, D> { | ||||||
|         // Start the USB bus. |         // Start the USB bus. | ||||||
|         // This prevent further allocation by consuming the driver. |         // This prevent further allocation by consuming the driver. | ||||||
| @@ -242,6 +237,7 @@ impl<'d, D: Driver<'d>> UsbDevice<'d, D> { | |||||||
|                 device_descriptor, |                 device_descriptor, | ||||||
|                 config_descriptor, |                 config_descriptor, | ||||||
|                 bos_descriptor, |                 bos_descriptor, | ||||||
|  |                 msos_descriptor, | ||||||
|  |  | ||||||
|                 device_state: UsbDeviceState::Unpowered, |                 device_state: UsbDeviceState::Unpowered, | ||||||
|                 suspended: false, |                 suspended: false, | ||||||
| @@ -251,8 +247,6 @@ impl<'d, D: Driver<'d>> UsbDevice<'d, D> { | |||||||
|                 set_address_pending: false, |                 set_address_pending: false, | ||||||
|                 interfaces, |                 interfaces, | ||||||
|                 handlers, |                 handlers, | ||||||
|                 #[cfg(feature = "msos-descriptor")] |  | ||||||
|                 msos_descriptor, |  | ||||||
|             }, |             }, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -261,16 +255,11 @@ impl<'d, D: Driver<'d>> UsbDevice<'d, D> { | |||||||
|     /// |     /// | ||||||
|     /// Useful for tuning buffer sizes for actual usage |     /// Useful for tuning buffer sizes for actual usage | ||||||
|     pub fn buffer_usage(&self) -> UsbBufferReport { |     pub fn buffer_usage(&self) -> UsbBufferReport { | ||||||
|         #[cfg(not(feature = "msos-descriptor"))] |  | ||||||
|         let mdu = None; |  | ||||||
|         #[cfg(feature = "msos-descriptor")] |  | ||||||
|         let mdu = Some(self.inner.msos_descriptor.len()); |  | ||||||
|  |  | ||||||
|         UsbBufferReport { |         UsbBufferReport { | ||||||
|             device_descriptor_used: self.inner.device_descriptor.len(), |             device_descriptor_used: self.inner.device_descriptor.len(), | ||||||
|             config_descriptor_used: self.inner.config_descriptor.len(), |             config_descriptor_used: self.inner.config_descriptor.len(), | ||||||
|             bos_descriptor_used: self.inner.bos_descriptor.len(), |             bos_descriptor_used: self.inner.bos_descriptor.len(), | ||||||
|             msos_descriptor_used: mdu, |             msos_descriptor_used: self.inner.msos_descriptor.len(), | ||||||
|             control_buffer_size: self.control_buf.len(), |             control_buffer_size: self.control_buf.len(), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -684,7 +673,7 @@ impl<'d, D: Driver<'d>> Inner<'d, D> { | |||||||
|                 } |                 } | ||||||
|                 _ => InResponse::Rejected, |                 _ => InResponse::Rejected, | ||||||
|             }, |             }, | ||||||
|             #[cfg(feature = "msos-descriptor")] |  | ||||||
|             (RequestType::Vendor, Recipient::Device) => { |             (RequestType::Vendor, Recipient::Device) => { | ||||||
|                 if !self.msos_descriptor.is_empty() |                 if !self.msos_descriptor.is_empty() | ||||||
|                     && req.request == self.msos_descriptor.vendor_code() |                     && req.request == self.msos_descriptor.vendor_code() | ||||||
|   | |||||||
| @@ -1,5 +1,3 @@ | |||||||
| #![cfg(feature = "msos-descriptor")] |  | ||||||
|  |  | ||||||
| //! Microsoft OS Descriptors | //! Microsoft OS Descriptors | ||||||
| //! | //! | ||||||
| //! <https://docs.microsoft.com/en-us/windows-hardware/drivers/usbcon/microsoft-os-2-0-descriptors-specification> | //! <https://docs.microsoft.com/en-us/windows-hardware/drivers/usbcon/microsoft-os-2-0-descriptors-specification> | ||||||
|   | |||||||
| @@ -34,7 +34,7 @@ embassy-executor = { version = "0.3.1", path = "../../embassy-executor", feature | |||||||
| embassy-time = { version = "0.1.5", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime"] } | embassy-time = { version = "0.1.5", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime"] } | ||||||
| embassy-nrf = { version = "0.1.0", path = "../../embassy-nrf", features = ["defmt", "nrf52840", "time-driver-rtc1", "gpiote", "unstable-pac", "time"] } | embassy-nrf = { version = "0.1.0", path = "../../embassy-nrf", features = ["defmt", "nrf52840", "time-driver-rtc1", "gpiote", "unstable-pac", "time"] } | ||||||
| embassy-net = { version = "0.2.0", path = "../../embassy-net", features = ["defmt", "tcp", "dhcpv4", "medium-ethernet"], optional = true } | embassy-net = { version = "0.2.0", path = "../../embassy-net", features = ["defmt", "tcp", "dhcpv4", "medium-ethernet"], optional = true } | ||||||
| embassy-usb = { version = "0.1.0", path = "../../embassy-usb", features = ["defmt", "msos-descriptor",], optional = true } | embassy-usb = { version = "0.1.0", path = "../../embassy-usb", features = ["defmt"], optional = true } | ||||||
| embedded-io = { version = "0.6.0", features = ["defmt-03"]  } | embedded-io = { version = "0.6.0", features = ["defmt-03"]  } | ||||||
| embedded-io-async = { version = "0.6.0", optional = true, features = ["defmt-03"] } | embedded-io-async = { version = "0.6.0", optional = true, features = ["defmt-03"] } | ||||||
| embassy-lora = { version = "0.1.0", path = "../../embassy-lora", features = ["time", "defmt"], optional = true } | embassy-lora = { version = "0.1.0", path = "../../embassy-lora", features = ["time", "defmt"], optional = true } | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ st7789 = "0.6.1" | |||||||
| display-interface = "0.4.1" | display-interface = "0.4.1" | ||||||
| byte-slice-cast = { version = "1.2.0", default-features = false } | byte-slice-cast = { version = "1.2.0", default-features = false } | ||||||
| smart-leds = "0.3.0" | smart-leds = "0.3.0" | ||||||
| heapless = "0.7.15" | heapless = "0.8" | ||||||
| usbd-hid = "0.6.1" | usbd-hid = "0.6.1" | ||||||
|  |  | ||||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-rc.1" } | embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-rc.1" } | ||||||
|   | |||||||
| @@ -75,6 +75,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -71,6 +71,7 @@ async fn main(spawner: Spawner) { | |||||||
|         &mut make_static!([0; 256])[..], |         &mut make_static!([0; 256])[..], | ||||||
|         &mut make_static!([0; 256])[..], |         &mut make_static!([0; 256])[..], | ||||||
|         &mut make_static!([0; 256])[..], |         &mut make_static!([0; 256])[..], | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut make_static!([0; 128])[..], |         &mut make_static!([0; 128])[..], | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -41,7 +41,7 @@ async fn main(_spawner: Spawner) { | |||||||
|     let mut config_descriptor = [0; 256]; |     let mut config_descriptor = [0; 256]; | ||||||
|     let mut bos_descriptor = [0; 256]; |     let mut bos_descriptor = [0; 256]; | ||||||
|     // You can also add a Microsoft OS descriptor. |     // You can also add a Microsoft OS descriptor. | ||||||
|     // let mut msos_descriptor = [0; 256]; |     let mut msos_descriptor = [0; 256]; | ||||||
|     let mut control_buf = [0; 64]; |     let mut control_buf = [0; 64]; | ||||||
|     let request_handler = MyRequestHandler {}; |     let request_handler = MyRequestHandler {}; | ||||||
|     let mut device_handler = MyDeviceHandler::new(); |     let mut device_handler = MyDeviceHandler::new(); | ||||||
| @@ -54,7 +54,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|         // &mut msos_descriptor, |         &mut msos_descriptor, | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -58,6 +58,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										199
									
								
								examples/rp/src/bin/usb_raw.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										199
									
								
								examples/rp/src/bin/usb_raw.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,199 @@ | |||||||
|  | //! Example of using USB without a pre-defined class, but instead responding to | ||||||
|  | //! raw USB control requests. | ||||||
|  | //! | ||||||
|  | //! The host computer can either: | ||||||
|  | //! * send a command, with a 16-bit request ID, a 16-bit value, and an optional data buffer | ||||||
|  | //! * request some data, with a 16-bit request ID, a 16-bit value, and a length of data to receive | ||||||
|  | //! | ||||||
|  | //! For higher throughput data, you can add some bulk endpoints after creating the alternate, | ||||||
|  | //! but for low rate command/response, plain control transfers can be very simple and effective. | ||||||
|  | //! | ||||||
|  | //! Example code to send/receive data using `nusb`: | ||||||
|  | //! | ||||||
|  | //! ```ignore | ||||||
|  | //! use futures_lite::future::block_on; | ||||||
|  | //! use nusb::transfer::{ControlIn, ControlOut, ControlType, Recipient}; | ||||||
|  | //! | ||||||
|  | //! fn main() { | ||||||
|  | //!     let di = nusb::list_devices() | ||||||
|  | //!         .unwrap() | ||||||
|  | //!         .find(|d| d.vendor_id() == 0xc0de && d.product_id() == 0xcafe) | ||||||
|  | //!         .expect("no device found"); | ||||||
|  | //!     let device = di.open().expect("error opening device"); | ||||||
|  | //!     let interface = device.claim_interface(0).expect("error claiming interface"); | ||||||
|  | //! | ||||||
|  | //!     // Send "hello world" to device | ||||||
|  | //!     let result = block_on(interface.control_out(ControlOut { | ||||||
|  | //!         control_type: ControlType::Vendor, | ||||||
|  | //!         recipient: Recipient::Interface, | ||||||
|  | //!         request: 100, | ||||||
|  | //!         value: 200, | ||||||
|  | //!         index: 0, | ||||||
|  | //!         data: b"hello world", | ||||||
|  | //!     })); | ||||||
|  | //!     println!("{result:?}"); | ||||||
|  | //! | ||||||
|  | //!     // Receive "hello" from device | ||||||
|  | //!     let result = block_on(interface.control_in(ControlIn { | ||||||
|  | //!         control_type: ControlType::Vendor, | ||||||
|  | //!         recipient: Recipient::Interface, | ||||||
|  | //!         request: 101, | ||||||
|  | //!         value: 201, | ||||||
|  | //!         index: 0, | ||||||
|  | //!         length: 5, | ||||||
|  | //!     })); | ||||||
|  | //!     println!("{result:?}"); | ||||||
|  | //! } | ||||||
|  | //! ``` | ||||||
|  |  | ||||||
|  | #![no_std] | ||||||
|  | #![no_main] | ||||||
|  | #![feature(type_alias_impl_trait)] | ||||||
|  |  | ||||||
|  | use defmt::info; | ||||||
|  | use embassy_executor::Spawner; | ||||||
|  | use embassy_rp::bind_interrupts; | ||||||
|  | use embassy_rp::peripherals::USB; | ||||||
|  | use embassy_rp::usb::{Driver, InterruptHandler}; | ||||||
|  | use embassy_usb::control::{InResponse, OutResponse, Recipient, Request, RequestType}; | ||||||
|  | use embassy_usb::msos::{self, windows_version}; | ||||||
|  | use embassy_usb::types::InterfaceNumber; | ||||||
|  | use embassy_usb::{Builder, Config, Handler}; | ||||||
|  | use {defmt_rtt as _, panic_probe as _}; | ||||||
|  |  | ||||||
|  | // This is a randomly generated GUID to allow clients on Windows to find our device | ||||||
|  | const DEVICE_INTERFACE_GUIDS: &[&str] = &["{AFB9A6FB-30BA-44BC-9232-806CFC875321}"]; | ||||||
|  |  | ||||||
|  | bind_interrupts!(struct Irqs { | ||||||
|  |     USBCTRL_IRQ => InterruptHandler<USB>; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | #[embassy_executor::main] | ||||||
|  | async fn main(_spawner: Spawner) { | ||||||
|  |     info!("Hello there!"); | ||||||
|  |  | ||||||
|  |     let p = embassy_rp::init(Default::default()); | ||||||
|  |  | ||||||
|  |     // Create the driver, from the HAL. | ||||||
|  |     let driver = Driver::new(p.USB, Irqs); | ||||||
|  |  | ||||||
|  |     // Create embassy-usb Config | ||||||
|  |     let mut config = Config::new(0xc0de, 0xcafe); | ||||||
|  |     config.manufacturer = Some("Embassy"); | ||||||
|  |     config.product = Some("USB raw example"); | ||||||
|  |     config.serial_number = Some("12345678"); | ||||||
|  |     config.max_power = 100; | ||||||
|  |     config.max_packet_size_0 = 64; | ||||||
|  |  | ||||||
|  |     // // Required for windows compatibility. | ||||||
|  |     // // https://developer.nordicsemi.com/nRF_Connect_SDK/doc/1.9.1/kconfig/CONFIG_CDC_ACM_IAD.html#help | ||||||
|  |     config.device_class = 0xEF; | ||||||
|  |     config.device_sub_class = 0x02; | ||||||
|  |     config.device_protocol = 0x01; | ||||||
|  |     config.composite_with_iads = true; | ||||||
|  |  | ||||||
|  |     // Create embassy-usb DeviceBuilder using the driver and config. | ||||||
|  |     // It needs some buffers for building the descriptors. | ||||||
|  |     let mut device_descriptor = [0; 256]; | ||||||
|  |     let mut config_descriptor = [0; 256]; | ||||||
|  |     let mut bos_descriptor = [0; 256]; | ||||||
|  |     let mut msos_descriptor = [0; 256]; | ||||||
|  |     let mut control_buf = [0; 64]; | ||||||
|  |  | ||||||
|  |     let mut handler = ControlHandler { | ||||||
|  |         if_num: InterfaceNumber(0), | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     let mut builder = Builder::new( | ||||||
|  |         driver, | ||||||
|  |         config, | ||||||
|  |         &mut device_descriptor, | ||||||
|  |         &mut config_descriptor, | ||||||
|  |         &mut bos_descriptor, | ||||||
|  |         &mut msos_descriptor, | ||||||
|  |         &mut control_buf, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     // Add the Microsoft OS Descriptor (MSOS/MOD) descriptor. | ||||||
|  |     // We tell Windows that this entire device is compatible with the "WINUSB" feature, | ||||||
|  |     // which causes it to use the built-in WinUSB driver automatically, which in turn | ||||||
|  |     // can be used by libusb/rusb software without needing a custom driver or INF file. | ||||||
|  |     // In principle you might want to call msos_feature() just on a specific function, | ||||||
|  |     // if your device also has other functions that still use standard class drivers. | ||||||
|  |     builder.msos_descriptor(windows_version::WIN8_1, 0); | ||||||
|  |     builder.msos_feature(msos::CompatibleIdFeatureDescriptor::new("WINUSB", "")); | ||||||
|  |     builder.msos_feature(msos::RegistryPropertyFeatureDescriptor::new( | ||||||
|  |         "DeviceInterfaceGUIDs", | ||||||
|  |         msos::PropertyData::RegMultiSz(DEVICE_INTERFACE_GUIDS), | ||||||
|  |     )); | ||||||
|  |  | ||||||
|  |     // Add a vendor-specific function (class 0xFF), and corresponding interface, | ||||||
|  |     // that uses our custom handler. | ||||||
|  |     let mut function = builder.function(0xFF, 0, 0); | ||||||
|  |     let mut interface = function.interface(); | ||||||
|  |     let _alt = interface.alt_setting(0xFF, 0, 0, None); | ||||||
|  |     handler.if_num = interface.interface_number(); | ||||||
|  |     drop(function); | ||||||
|  |     builder.handler(&mut handler); | ||||||
|  |  | ||||||
|  |     // Build the builder. | ||||||
|  |     let mut usb = builder.build(); | ||||||
|  |  | ||||||
|  |     // Run the USB device. | ||||||
|  |     usb.run().await; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// Handle CONTROL endpoint requests and responses. For many simple requests and responses | ||||||
|  | /// you can get away with only using the control endpoint. | ||||||
|  | struct ControlHandler { | ||||||
|  |     if_num: InterfaceNumber, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl Handler for ControlHandler { | ||||||
|  |     /// Respond to HostToDevice control messages, where the host sends us a command and | ||||||
|  |     /// optionally some data, and we can only acknowledge or reject it. | ||||||
|  |     fn control_out<'a>(&'a mut self, req: Request, buf: &'a [u8]) -> Option<OutResponse> { | ||||||
|  |         // Log the request before filtering to help with debugging. | ||||||
|  |         info!("Got control_out, request={}, buf={:a}", req, buf); | ||||||
|  |  | ||||||
|  |         // Only handle Vendor request types to an Interface. | ||||||
|  |         if req.request_type != RequestType::Vendor || req.recipient != Recipient::Interface { | ||||||
|  |             return None; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Ignore requests to other interfaces. | ||||||
|  |         if req.index != self.if_num.0 as u16 { | ||||||
|  |             return None; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Accept request 100, value 200, reject others. | ||||||
|  |         if req.request == 100 && req.value == 200 { | ||||||
|  |             Some(OutResponse::Accepted) | ||||||
|  |         } else { | ||||||
|  |             Some(OutResponse::Rejected) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Respond to DeviceToHost control messages, where the host requests some data from us. | ||||||
|  |     fn control_in<'a>(&'a mut self, req: Request, buf: &'a mut [u8]) -> Option<InResponse<'a>> { | ||||||
|  |         info!("Got control_in, request={}", req); | ||||||
|  |  | ||||||
|  |         // Only handle Vendor request types to an Interface. | ||||||
|  |         if req.request_type != RequestType::Vendor || req.recipient != Recipient::Interface { | ||||||
|  |             return None; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Ignore requests to other interfaces. | ||||||
|  |         if req.index != self.if_num.0 as u16 { | ||||||
|  |             return None; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Respond "hello" to request 101, value 201, when asked for 5 bytes, otherwise reject. | ||||||
|  |         if req.request == 101 && req.value == 201 && req.length == 5 { | ||||||
|  |             buf[..5].copy_from_slice(b"hello"); | ||||||
|  |             Some(InResponse::Accepted(&buf[..5])) | ||||||
|  |         } else { | ||||||
|  |             Some(InResponse::Rejected) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										142
									
								
								examples/rp/src/bin/usb_raw_bulk.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										142
									
								
								examples/rp/src/bin/usb_raw_bulk.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,142 @@ | |||||||
|  | //! Example of using USB without a pre-defined class, but instead using raw USB bulk transfers. | ||||||
|  | //! | ||||||
|  | //! Example code to send/receive data using `nusb`: | ||||||
|  | //! | ||||||
|  | //! ```ignore | ||||||
|  | //! use futures_lite::future::block_on; | ||||||
|  | //! use nusb::transfer::RequestBuffer; | ||||||
|  | //! | ||||||
|  | //! const BULK_OUT_EP: u8 = 0x01; | ||||||
|  | //! const BULK_IN_EP: u8 = 0x81; | ||||||
|  | //! | ||||||
|  | //! fn main() { | ||||||
|  | //!     let di = nusb::list_devices() | ||||||
|  | //!         .unwrap() | ||||||
|  | //!         .find(|d| d.vendor_id() == 0xc0de && d.product_id() == 0xcafe) | ||||||
|  | //!         .expect("no device found"); | ||||||
|  | //!     let device = di.open().expect("error opening device"); | ||||||
|  | //!     let interface = device.claim_interface(0).expect("error claiming interface"); | ||||||
|  | //! | ||||||
|  | //!     let result = block_on(interface.bulk_out(BULK_OUT_EP, b"hello world".into())); | ||||||
|  | //!     println!("{result:?}"); | ||||||
|  | //!     let result = block_on(interface.bulk_in(BULK_IN_EP, RequestBuffer::new(64))); | ||||||
|  | //!     println!("{result:?}"); | ||||||
|  | //! } | ||||||
|  | //! ``` | ||||||
|  |  | ||||||
|  | #![no_std] | ||||||
|  | #![no_main] | ||||||
|  | #![feature(type_alias_impl_trait)] | ||||||
|  |  | ||||||
|  | use defmt::info; | ||||||
|  | use embassy_executor::Spawner; | ||||||
|  | use embassy_futures::join::join; | ||||||
|  | use embassy_rp::bind_interrupts; | ||||||
|  | use embassy_rp::peripherals::USB; | ||||||
|  | use embassy_rp::usb::{Driver, InterruptHandler}; | ||||||
|  | use embassy_usb::driver::{Endpoint, EndpointIn, EndpointOut}; | ||||||
|  | use embassy_usb::msos::{self, windows_version}; | ||||||
|  | use embassy_usb::{Builder, Config}; | ||||||
|  | use {defmt_rtt as _, panic_probe as _}; | ||||||
|  |  | ||||||
|  | // This is a randomly generated GUID to allow clients on Windows to find our device | ||||||
|  | const DEVICE_INTERFACE_GUIDS: &[&str] = &["{AFB9A6FB-30BA-44BC-9232-806CFC875321}"]; | ||||||
|  |  | ||||||
|  | bind_interrupts!(struct Irqs { | ||||||
|  |     USBCTRL_IRQ => InterruptHandler<USB>; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | #[embassy_executor::main] | ||||||
|  | async fn main(_spawner: Spawner) { | ||||||
|  |     info!("Hello there!"); | ||||||
|  |  | ||||||
|  |     let p = embassy_rp::init(Default::default()); | ||||||
|  |  | ||||||
|  |     // Create the driver, from the HAL. | ||||||
|  |     let driver = Driver::new(p.USB, Irqs); | ||||||
|  |  | ||||||
|  |     // Create embassy-usb Config | ||||||
|  |     let mut config = Config::new(0xc0de, 0xcafe); | ||||||
|  |     config.manufacturer = Some("Embassy"); | ||||||
|  |     config.product = Some("USB raw example"); | ||||||
|  |     config.serial_number = Some("12345678"); | ||||||
|  |     config.max_power = 100; | ||||||
|  |     config.max_packet_size_0 = 64; | ||||||
|  |  | ||||||
|  |     // // Required for windows compatibility. | ||||||
|  |     // // https://developer.nordicsemi.com/nRF_Connect_SDK/doc/1.9.1/kconfig/CONFIG_CDC_ACM_IAD.html#help | ||||||
|  |     config.device_class = 0xEF; | ||||||
|  |     config.device_sub_class = 0x02; | ||||||
|  |     config.device_protocol = 0x01; | ||||||
|  |     config.composite_with_iads = true; | ||||||
|  |  | ||||||
|  |     // Create embassy-usb DeviceBuilder using the driver and config. | ||||||
|  |     // It needs some buffers for building the descriptors. | ||||||
|  |     let mut device_descriptor = [0; 256]; | ||||||
|  |     let mut config_descriptor = [0; 256]; | ||||||
|  |     let mut bos_descriptor = [0; 256]; | ||||||
|  |     let mut msos_descriptor = [0; 256]; | ||||||
|  |     let mut control_buf = [0; 64]; | ||||||
|  |  | ||||||
|  |     let mut builder = Builder::new( | ||||||
|  |         driver, | ||||||
|  |         config, | ||||||
|  |         &mut device_descriptor, | ||||||
|  |         &mut config_descriptor, | ||||||
|  |         &mut bos_descriptor, | ||||||
|  |         &mut msos_descriptor, | ||||||
|  |         &mut control_buf, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     // Add the Microsoft OS Descriptor (MSOS/MOD) descriptor. | ||||||
|  |     // We tell Windows that this entire device is compatible with the "WINUSB" feature, | ||||||
|  |     // which causes it to use the built-in WinUSB driver automatically, which in turn | ||||||
|  |     // can be used by libusb/rusb software without needing a custom driver or INF file. | ||||||
|  |     // In principle you might want to call msos_feature() just on a specific function, | ||||||
|  |     // if your device also has other functions that still use standard class drivers. | ||||||
|  |     builder.msos_descriptor(windows_version::WIN8_1, 0); | ||||||
|  |     builder.msos_feature(msos::CompatibleIdFeatureDescriptor::new("WINUSB", "")); | ||||||
|  |     builder.msos_feature(msos::RegistryPropertyFeatureDescriptor::new( | ||||||
|  |         "DeviceInterfaceGUIDs", | ||||||
|  |         msos::PropertyData::RegMultiSz(DEVICE_INTERFACE_GUIDS), | ||||||
|  |     )); | ||||||
|  |  | ||||||
|  |     // Add a vendor-specific function (class 0xFF), and corresponding interface, | ||||||
|  |     // that uses our custom handler. | ||||||
|  |     let mut function = builder.function(0xFF, 0, 0); | ||||||
|  |     let mut interface = function.interface(); | ||||||
|  |     let mut alt = interface.alt_setting(0xFF, 0, 0, None); | ||||||
|  |     let mut read_ep = alt.endpoint_bulk_out(64); | ||||||
|  |     let mut write_ep = alt.endpoint_bulk_in(64); | ||||||
|  |     drop(function); | ||||||
|  |  | ||||||
|  |     // Build the builder. | ||||||
|  |     let mut usb = builder.build(); | ||||||
|  |  | ||||||
|  |     // Run the USB device. | ||||||
|  |     let usb_fut = usb.run(); | ||||||
|  |  | ||||||
|  |     // Do stuff with the class! | ||||||
|  |     let echo_fut = async { | ||||||
|  |         loop { | ||||||
|  |             read_ep.wait_enabled().await; | ||||||
|  |             info!("Connected"); | ||||||
|  |             loop { | ||||||
|  |                 let mut data = [0; 64]; | ||||||
|  |                 match read_ep.read(&mut data).await { | ||||||
|  |                     Ok(n) => { | ||||||
|  |                         info!("Got bulk: {:a}", data[..n]); | ||||||
|  |                         // Echo back to the host: | ||||||
|  |                         write_ep.write(&data[..n]).await.ok(); | ||||||
|  |                     } | ||||||
|  |                     Err(_) => break, | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             info!("Disconnected"); | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     // Run everything concurrently. | ||||||
|  |     // If we had made everything `'static` above instead, we could do this using separate tasks instead. | ||||||
|  |     join(usb_fut, echo_fut).await; | ||||||
|  | } | ||||||
| @@ -60,6 +60,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,7 +14,6 @@ embassy-net-ppp = { version = "0.1.0", path = "../../embassy-net-ppp", features | |||||||
| embedded-io-async = { version = "0.6.0" } | embedded-io-async = { version = "0.6.0" } | ||||||
| embedded-io-adapters = { version = "0.6.0", features = ["futures-03"] } | embedded-io-adapters = { version = "0.6.0", features = ["futures-03"] } | ||||||
| critical-section = { version = "1.1", features = ["std"] } | critical-section = { version = "1.1", features = ["std"] } | ||||||
| smoltcp = { version = "0.10.0", features = ["dns-max-server-count-4"] } |  | ||||||
|  |  | ||||||
| async-io = "1.6.0" | async-io = "1.6.0" | ||||||
| env_logger = "0.9.0" | env_logger = "0.9.0" | ||||||
| @@ -23,7 +22,7 @@ log = "0.4.14" | |||||||
| nix = "0.26.2" | nix = "0.26.2" | ||||||
| clap = { version = "3.0.0-beta.5", features = ["derive"] } | clap = { version = "3.0.0-beta.5", features = ["derive"] } | ||||||
| rand_core = { version = "0.6.3", features = ["std"] } | rand_core = { version = "0.6.3", features = ["std"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| static_cell = { version = "2", features = ["nightly"]} | static_cell = { version = "2", features = ["nightly"]} | ||||||
|  |  | ||||||
| [profile.release] | [profile.release] | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
|  |  | ||||||
| [profile.release] | [profile.release] | ||||||
| debug = 2 | debug = 2 | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| nb = "1.0.0" | nb = "1.0.0" | ||||||
|  |  | ||||||
| [profile.dev] | [profile.dev] | ||||||
|   | |||||||
| @@ -60,6 +60,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| nb = "1.0.0" | nb = "1.0.0" | ||||||
|  |  | ||||||
| [profile.release] | [profile.release] | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| nb = "1.0.0" | nb = "1.0.0" | ||||||
| embedded-storage = "0.3.0" | embedded-storage = "0.3.0" | ||||||
| static_cell = { version = "2", features = ["nightly"]} | static_cell = { version = "2", features = ["nightly"]} | ||||||
|   | |||||||
| @@ -57,6 +57,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| nb = "1.0.0" | nb = "1.0.0" | ||||||
| embedded-storage = "0.3.0" | embedded-storage = "0.3.0" | ||||||
| static_cell = { version = "2", features = ["nightly"]} | static_cell = { version = "2", features = ["nightly"]} | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ embassy-stm32 = { version = "0.1.0", path = "../../embassy-stm32", features = [" | |||||||
| embassy-sync = { version = "0.4.0", path = "../../embassy-sync", features = ["defmt"] } | embassy-sync = { version = "0.4.0", path = "../../embassy-sync", features = ["defmt"] } | ||||||
| embassy-executor = { version = "0.3.1", path = "../../embassy-executor", features = ["nightly", "arch-cortex-m", "executor-thread", "executor-interrupt", "defmt", "integrated-timers"] } | embassy-executor = { version = "0.3.1", path = "../../embassy-executor", features = ["nightly", "arch-cortex-m", "executor-thread", "executor-interrupt", "defmt", "integrated-timers"] } | ||||||
| embassy-time = { version = "0.1.5", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime", "unstable-traits", "tick-hz-32_768"] } | embassy-time = { version = "0.1.5", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime", "unstable-traits", "tick-hz-32_768"] } | ||||||
| embassy-usb = { version = "0.1.0", path = "../../embassy-usb", features = ["defmt"] } | embassy-usb = { version = "0.1.0", path = "../../embassy-usb", features = ["defmt" ] } | ||||||
| embassy-net = { version = "0.2.0", path = "../../embassy-net", features = ["defmt", "tcp", "dhcpv4", "medium-ethernet", "nightly"] } | embassy-net = { version = "0.2.0", path = "../../embassy-net", features = ["defmt", "tcp", "dhcpv4", "medium-ethernet", "nightly"] } | ||||||
|  |  | ||||||
| defmt = "0.3" | defmt = "0.3" | ||||||
| @@ -23,7 +23,7 @@ embedded-io = { version = "0.6.0" } | |||||||
| embedded-io-async = { version = "0.6.0" } | embedded-io-async = { version = "0.6.0" } | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| nb = "1.0.0" | nb = "1.0.0" | ||||||
| embedded-storage = "0.3.0" | embedded-storage = "0.3.0" | ||||||
| micromath = "2.0.0" | micromath = "2.0.0" | ||||||
|   | |||||||
| @@ -94,6 +94,7 @@ async fn main(spawner: Spawner) { | |||||||
|         &mut make_static!([0; 256])[..], |         &mut make_static!([0; 256])[..], | ||||||
|         &mut make_static!([0; 256])[..], |         &mut make_static!([0; 256])[..], | ||||||
|         &mut make_static!([0; 256])[..], |         &mut make_static!([0; 256])[..], | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut make_static!([0; 128])[..], |         &mut make_static!([0; 128])[..], | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -56,10 +56,16 @@ use embassy_stm32::time::Hertz; | |||||||
| use embassy_stm32::usb_otg::Driver; | use embassy_stm32::usb_otg::Driver; | ||||||
| use embassy_stm32::{bind_interrupts, peripherals, usb_otg, Config}; | use embassy_stm32::{bind_interrupts, peripherals, usb_otg, Config}; | ||||||
| use embassy_usb::control::{InResponse, OutResponse, Recipient, Request, RequestType}; | use embassy_usb::control::{InResponse, OutResponse, Recipient, Request, RequestType}; | ||||||
|  | use embassy_usb::msos::{self, windows_version}; | ||||||
| use embassy_usb::types::InterfaceNumber; | use embassy_usb::types::InterfaceNumber; | ||||||
| use embassy_usb::{Builder, Handler}; | use embassy_usb::{Builder, Handler}; | ||||||
| use {defmt_rtt as _, panic_probe as _}; | use {defmt_rtt as _, panic_probe as _}; | ||||||
|  |  | ||||||
|  | // Randomly generated UUID because Windows requires you provide one to use WinUSB. | ||||||
|  | // In principle WinUSB-using software could find this device (or a specific interface | ||||||
|  | // on it) by its GUID instead of using the VID/PID, but in practice that seems unhelpful. | ||||||
|  | const DEVICE_INTERFACE_GUIDS: &[&str] = &["{DAC2087C-63FA-458D-A55D-827C0762DEC7}"]; | ||||||
|  |  | ||||||
| bind_interrupts!(struct Irqs { | bind_interrupts!(struct Irqs { | ||||||
|     OTG_FS => usb_otg::InterruptHandler<peripherals::USB_OTG_FS>; |     OTG_FS => usb_otg::InterruptHandler<peripherals::USB_OTG_FS>; | ||||||
| }); | }); | ||||||
| @@ -114,6 +120,7 @@ async fn main(_spawner: Spawner) { | |||||||
|     let mut device_descriptor = [0; 256]; |     let mut device_descriptor = [0; 256]; | ||||||
|     let mut config_descriptor = [0; 256]; |     let mut config_descriptor = [0; 256]; | ||||||
|     let mut bos_descriptor = [0; 256]; |     let mut bos_descriptor = [0; 256]; | ||||||
|  |     let mut msos_descriptor = [0; 256]; | ||||||
|     let mut control_buf = [0; 64]; |     let mut control_buf = [0; 64]; | ||||||
|  |  | ||||||
|     let mut handler = ControlHandler { |     let mut handler = ControlHandler { | ||||||
| @@ -126,9 +133,23 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut msos_descriptor, | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     // Add the Microsoft OS Descriptor (MSOS/MOD) descriptor. | ||||||
|  |     // We tell Windows that this entire device is compatible with the "WINUSB" feature, | ||||||
|  |     // which causes it to use the built-in WinUSB driver automatically, which in turn | ||||||
|  |     // can be used by libusb/rusb software without needing a custom driver or INF file. | ||||||
|  |     // In principle you might want to call msos_feature() just on a specific function, | ||||||
|  |     // if your device also has other functions that still use standard class drivers. | ||||||
|  |     builder.msos_descriptor(windows_version::WIN8_1, 0); | ||||||
|  |     builder.msos_feature(msos::CompatibleIdFeatureDescriptor::new("WINUSB", "")); | ||||||
|  |     builder.msos_feature(msos::RegistryPropertyFeatureDescriptor::new( | ||||||
|  |         "DeviceInterfaceGUIDs", | ||||||
|  |         msos::PropertyData::RegMultiSz(DEVICE_INTERFACE_GUIDS), | ||||||
|  |     )); | ||||||
|  |  | ||||||
|     // Add a vendor-specific function (class 0xFF), and corresponding interface, |     // Add a vendor-specific function (class 0xFF), and corresponding interface, | ||||||
|     // that uses our custom handler. |     // that uses our custom handler. | ||||||
|     let mut function = builder.function(0xFF, 0, 0); |     let mut function = builder.function(0xFF, 0, 0); | ||||||
|   | |||||||
| @@ -77,6 +77,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -22,7 +22,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| nb = "1.0.0" | nb = "1.0.0" | ||||||
| rand_core = "0.6.3" | rand_core = "0.6.3" | ||||||
| critical-section = "1.1" | critical-section = "1.1" | ||||||
|   | |||||||
| @@ -77,6 +77,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| portable-atomic = { version = "1.5", features = ["unsafe-assume-single-core"] } | portable-atomic = { version = "1.5", features = ["unsafe-assume-single-core"] } | ||||||
|  |  | ||||||
| [profile.release] | [profile.release] | ||||||
|   | |||||||
| @@ -22,7 +22,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
|  |  | ||||||
| [profile.release] | [profile.release] | ||||||
| debug = 2 | debug = 2 | ||||||
|   | |||||||
| @@ -75,6 +75,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -22,10 +22,10 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-rc.1" } | embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-rc.1" } | ||||||
| embedded-hal-async = { version = "=1.0.0-rc.1" } | embedded-hal-async = { version = "=1.0.0-rc.1" } | ||||||
| embedded-nal-async = { version = "0.6.0" } | embedded-nal-async = { version = "0.7" } | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| rand_core = "0.6.3" | rand_core = "0.6.3" | ||||||
| critical-section = "1.1" | critical-section = "1.1" | ||||||
| micromath = "2.0.0" | micromath = "2.0.0" | ||||||
|   | |||||||
| @@ -82,6 +82,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ embassy-stm32 = { version = "0.1.0", path = "../../embassy-stm32", features = [" | |||||||
| embassy-sync = { version = "0.4.0", path = "../../embassy-sync", features = ["defmt"] } | embassy-sync = { version = "0.4.0", path = "../../embassy-sync", features = ["defmt"] } | ||||||
| embassy-executor = { version = "0.3.1", path = "../../embassy-executor", features = ["nightly", "arch-cortex-m", "executor-thread", "defmt", "integrated-timers"] } | embassy-executor = { version = "0.3.1", path = "../../embassy-executor", features = ["nightly", "arch-cortex-m", "executor-thread", "defmt", "integrated-timers"] } | ||||||
| embassy-time = { version = "0.1.5", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime", "unstable-traits", "tick-hz-32_768"] } | embassy-time = { version = "0.1.5", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime", "unstable-traits", "tick-hz-32_768"] } | ||||||
| embassy-net = { version = "0.2.0", path = "../../embassy-net", features = ["defmt", "nightly", "tcp", "dhcpv4", "medium-ethernet", "proto-ipv6"] } | embassy-net = { version = "0.2.0", path = "../../embassy-net", features = ["defmt", "nightly", "tcp", "dhcpv4", "medium-ethernet", "proto-ipv6", "dns"] } | ||||||
| embedded-io-async = { version = "0.6.0" } | embedded-io-async = { version = "0.6.0" } | ||||||
| embassy-usb = { version = "0.1.0", path = "../../embassy-usb", features = ["defmt"] } | embassy-usb = { version = "0.1.0", path = "../../embassy-usb", features = ["defmt"] } | ||||||
|  |  | ||||||
| @@ -22,10 +22,10 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-rc.1" } | embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-rc.1" } | ||||||
| embedded-hal-async = { version = "=1.0.0-rc.1" } | embedded-hal-async = { version = "=1.0.0-rc.1" } | ||||||
| embedded-nal-async = { version = "0.6.0" } | embedded-nal-async = { version = "0.7" } | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| rand_core = "0.6.3" | rand_core = "0.6.3" | ||||||
| critical-section = "1.1" | critical-section = "1.1" | ||||||
| micromath = "2.0.0" | micromath = "2.0.0" | ||||||
|   | |||||||
| @@ -78,6 +78,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ cortex-m = { version = "0.7.6", features = ["inline-asm", "critical-section-sing | |||||||
| cortex-m-rt = "0.7.0" | cortex-m-rt = "0.7.0" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| static_cell = { version = "2" } | static_cell = { version = "2" } | ||||||
| portable-atomic = { version = "1.5", features = ["unsafe-assume-single-core"] } | portable-atomic = { version = "1.5", features = ["unsafe-assume-single-core"] } | ||||||
|   | |||||||
| @@ -18,7 +18,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| embedded-storage = "0.3.0" | embedded-storage = "0.3.0" | ||||||
|  |  | ||||||
| [profile.release] | [profile.release] | ||||||
|   | |||||||
| @@ -29,7 +29,7 @@ embedded-hal-async = { version = "=1.0.0-rc.1" } | |||||||
| embedded-hal-bus = { version = "=0.1.0-rc.1", features = ["async"] } | embedded-hal-bus = { version = "=0.1.0-rc.1", features = ["async"] } | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| chrono = { version = "^0.4", default-features = false } | chrono = { version = "^0.4", default-features = false } | ||||||
| rand = { version = "0.8.5", default-features = false } | rand = { version = "0.8.5", default-features = false } | ||||||
| static_cell = { version = "2", features = ["nightly"]} | static_cell = { version = "2", features = ["nightly"]} | ||||||
|   | |||||||
| @@ -72,6 +72,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -23,7 +23,7 @@ cortex-m = { version = "0.7.6", features = ["inline-asm", "critical-section-sing | |||||||
| cortex-m-rt = "0.7.0" | cortex-m-rt = "0.7.0" | ||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| rand_core = { version = "0.6.3", default-features = false } | rand_core = { version = "0.6.3", default-features = false } | ||||||
| embedded-io-async = { version = "0.6.0" } | embedded-io-async = { version = "0.6.0" } | ||||||
| static_cell = { version = "2", features = ["nightly"]} | static_cell = { version = "2", features = ["nightly"]} | ||||||
|   | |||||||
| @@ -82,6 +82,7 @@ async fn main(spawner: Spawner) { | |||||||
|         &mut make_static!([0; 256])[..], |         &mut make_static!([0; 256])[..], | ||||||
|         &mut make_static!([0; 256])[..], |         &mut make_static!([0; 256])[..], | ||||||
|         &mut make_static!([0; 256])[..], |         &mut make_static!([0; 256])[..], | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut make_static!([0; 128])[..], |         &mut make_static!([0; 128])[..], | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -62,6 +62,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -57,6 +57,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
|  |  | ||||||
| micromath = "2.0.0" | micromath = "2.0.0" | ||||||
|  |  | ||||||
|   | |||||||
| @@ -67,6 +67,7 @@ async fn main(_spawner: Spawner) { | |||||||
|         &mut device_descriptor, |         &mut device_descriptor, | ||||||
|         &mut config_descriptor, |         &mut config_descriptor, | ||||||
|         &mut bos_descriptor, |         &mut bos_descriptor, | ||||||
|  |         &mut [], // no msos descriptors | ||||||
|         &mut control_buf, |         &mut control_buf, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| static_cell = { version = "2", features = ["nightly"]} | static_cell = { version = "2", features = ["nightly"]} | ||||||
|  |  | ||||||
| [features] | [features] | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ cortex-m-rt = "0.7.0" | |||||||
| embedded-hal = "0.2.6" | embedded-hal = "0.2.6" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| static_cell = { version = "2", features = ["nightly"]} | static_cell = { version = "2", features = ["nightly"]} | ||||||
|  |  | ||||||
| [profile.release] | [profile.release] | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ embedded-hal = "0.2.6" | |||||||
| embedded-storage = "0.3.0" | embedded-storage = "0.3.0" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||||
| heapless = { version = "0.7.5", default-features = false } | heapless = { version = "0.8", default-features = false } | ||||||
| chrono = { version = "^0.4", default-features = false } | chrono = { version = "^0.4", default-features = false } | ||||||
|  |  | ||||||
| [profile.release] | [profile.release] | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ stm32f103c8 = ["embassy-stm32/stm32f103c8", "not-gpdma"] | |||||||
| stm32f429zi = ["embassy-stm32/stm32f429zi", "chrono", "eth", "stop", "can", "not-gpdma", "dac-adc-pin", "rng"] | stm32f429zi = ["embassy-stm32/stm32f429zi", "chrono", "eth", "stop", "can", "not-gpdma", "dac-adc-pin", "rng"] | ||||||
| stm32g071rb = ["embassy-stm32/stm32g071rb", "cm0", "not-gpdma", "dac-adc-pin"] | stm32g071rb = ["embassy-stm32/stm32g071rb", "cm0", "not-gpdma", "dac-adc-pin"] | ||||||
| stm32c031c6 = ["embassy-stm32/stm32c031c6", "cm0", "not-gpdma"] | stm32c031c6 = ["embassy-stm32/stm32c031c6", "cm0", "not-gpdma"] | ||||||
| stm32g491re = ["embassy-stm32/stm32g491re", "chrono", "not-gpdma", "rng"] | stm32g491re = ["embassy-stm32/stm32g491re", "chrono", "stop", "not-gpdma", "rng"] | ||||||
| stm32h755zi = ["embassy-stm32/stm32h755zi-cm7", "chrono", "not-gpdma", "eth", "dac-adc-pin", "rng"] | stm32h755zi = ["embassy-stm32/stm32h755zi-cm7", "chrono", "not-gpdma", "eth", "dac-adc-pin", "rng"] | ||||||
| stm32h753zi = ["embassy-stm32/stm32h753zi", "chrono", "not-gpdma", "eth", "rng"] | stm32h753zi = ["embassy-stm32/stm32h753zi", "chrono", "not-gpdma", "eth", "rng"] | ||||||
| stm32h7a3zi = ["embassy-stm32/stm32h7a3zi", "not-gpdma", "rng"] | stm32h7a3zi = ["embassy-stm32/stm32h7a3zi", "not-gpdma", "rng"] | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user