Automatically set ADC clock prescaler on v2 ADC to respect max frequency
This commit is contained in:
		| @@ -1,4 +1,5 @@ | |||||||
| use crate::adc::{AdcPin, Instance}; | use crate::adc::{AdcPin, Instance}; | ||||||
|  | use crate::time::Hertz; | ||||||
| use core::marker::PhantomData; | use core::marker::PhantomData; | ||||||
| use embassy::util::Unborrow; | use embassy::util::Unborrow; | ||||||
| use embassy_hal_common::unborrow; | use embassy_hal_common::unborrow; | ||||||
| @@ -6,12 +7,12 @@ use embedded_hal_02::blocking::delay::DelayUs; | |||||||
|  |  | ||||||
| pub const VDDA_CALIB_MV: u32 = 3000; | pub const VDDA_CALIB_MV: u32 = 3000; | ||||||
|  |  | ||||||
| #[cfg(not(rcc_f4))] | #[cfg(not(any(rcc_f4, rcc_f7)))] | ||||||
| fn enable() { | fn enable() { | ||||||
|     todo!() |     todo!() | ||||||
| } | } | ||||||
|  |  | ||||||
| #[cfg(rcc_f4)] | #[cfg(any(rcc_f4, rcc_f7))] | ||||||
| fn enable() { | fn enable() { | ||||||
|     critical_section::with(|_| unsafe { |     critical_section::with(|_| unsafe { | ||||||
|         // TODO do not enable all adc clocks if not needed |         // TODO do not enable all adc clocks if not needed | ||||||
| @@ -114,6 +115,39 @@ impl Default for SampleTime { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | enum Prescaler { | ||||||
|  |     Div2, | ||||||
|  |     Div4, | ||||||
|  |     Div6, | ||||||
|  |     Div8, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl Prescaler { | ||||||
|  |     fn from_pclk2(freq: Hertz) -> Self { | ||||||
|  |         // Datasheet for both F4 and F7 specifies min frequency 0.6 MHz, typ freq. 30 MHz and max 36 MHz. | ||||||
|  |         const MAX_FREQUENCY: Hertz = Hertz(36_000_000); | ||||||
|  |         let raw_div = freq.0 / MAX_FREQUENCY.0; | ||||||
|  |         match raw_div { | ||||||
|  |             0..=1 => Self::Div2, | ||||||
|  |             2..=3 => Self::Div4, | ||||||
|  |             4..=5 => Self::Div6, | ||||||
|  |             6..=7 => Self::Div8, | ||||||
|  |             _ => panic!( | ||||||
|  |                 "Selected PCLK2 frequency is too high for ADC with largest possible prescaler." | ||||||
|  |             ), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn adcpre(&self) -> crate::pac::adccommon::vals::Adcpre { | ||||||
|  |         match self { | ||||||
|  |             Prescaler::Div2 => crate::pac::adccommon::vals::Adcpre::DIV2, | ||||||
|  |             Prescaler::Div4 => crate::pac::adccommon::vals::Adcpre::DIV4, | ||||||
|  |             Prescaler::Div6 => crate::pac::adccommon::vals::Adcpre::DIV6, | ||||||
|  |             Prescaler::Div8 => crate::pac::adccommon::vals::Adcpre::DIV8, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| pub struct Adc<'d, T: Instance> { | pub struct Adc<'d, T: Instance> { | ||||||
|     sample_time: SampleTime, |     sample_time: SampleTime, | ||||||
|     calibrated_vdda: u32, |     calibrated_vdda: u32, | ||||||
| @@ -128,6 +162,14 @@ where | |||||||
|     pub fn new(_peri: impl Unborrow<Target = T> + 'd, delay: &mut impl DelayUs<u32>) -> Self { |     pub fn new(_peri: impl Unborrow<Target = T> + 'd, delay: &mut impl DelayUs<u32>) -> Self { | ||||||
|         unborrow!(_peri); |         unborrow!(_peri); | ||||||
|         enable(); |         enable(); | ||||||
|  |  | ||||||
|  |         let presc = unsafe { Prescaler::from_pclk2(crate::rcc::get_freqs().apb2) }; | ||||||
|  |         unsafe { | ||||||
|  |             T::common_regs() | ||||||
|  |                 .ccr() | ||||||
|  |                 .modify(|w| w.set_adcpre(presc.adcpre())); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         unsafe { |         unsafe { | ||||||
|             // disable before config is set |             // disable before config is set | ||||||
|             T::regs().cr2().modify(|reg| { |             T::regs().cr2().modify(|reg| { | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								examples/stm32f7/src/bin/adc.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								examples/stm32f7/src/bin/adc.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | #![no_std] | ||||||
|  | #![no_main] | ||||||
|  | #![feature(type_alias_impl_trait)] | ||||||
|  |  | ||||||
|  | use defmt_rtt as _; // global logger | ||||||
|  | use panic_probe as _; | ||||||
|  |  | ||||||
|  | use defmt::*; | ||||||
|  | use embassy::executor::Spawner; | ||||||
|  | use embassy::time::{Delay, Duration, Timer}; | ||||||
|  | use embassy_stm32::adc::Adc; | ||||||
|  | use embassy_stm32::Peripherals; | ||||||
|  |  | ||||||
|  | #[embassy::main] | ||||||
|  | async fn main(_spawner: Spawner, p: Peripherals) { | ||||||
|  |     info!("Hello World!"); | ||||||
|  |  | ||||||
|  |     let mut adc = Adc::new(p.ADC1, &mut Delay); | ||||||
|  |     let mut pin = p.PA3; | ||||||
|  |  | ||||||
|  |     loop { | ||||||
|  |         let v = adc.read(&mut pin); | ||||||
|  |         info!("--> {} - {} mV", v, adc.to_millivolts(v)); | ||||||
|  |         Timer::after(Duration::from_millis(100)).await; | ||||||
|  |     } | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user