Added ADC support for STM32C0.
This commit is contained in:
		
							parent
							
								
									9407ac67d3
								
							
						
					
					
						commit
						8c6fa83006
					
				
							
								
								
									
										468
									
								
								embassy-stm32/src/adc/c0.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										468
									
								
								embassy-stm32/src/adc/c0.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,468 @@ | ||||
| use pac::adc::vals::Scandir; | ||||
| #[allow(unused)] | ||||
| use pac::adc::vals::{Adstp, Align, Ckmode, Dmacfg, Exten, Ovrmod, Ovsr}; | ||||
| use pac::adccommon::vals::Presc; | ||||
| 
 | ||||
| use super::{ | ||||
|     blocking_delay_us, Adc, AdcChannel, AnyAdcChannel, Instance, Resolution, RxDma, SampleTime, SealedAdcChannel, | ||||
| }; | ||||
| use crate::dma::Transfer; | ||||
| use crate::time::Hertz; | ||||
| use crate::{pac, rcc, Peripheral}; | ||||
| 
 | ||||
| /// Default VREF voltage used for sample conversion to millivolts.
 | ||||
| pub const VREF_DEFAULT_MV: u32 = 3300; | ||||
| /// VREF voltage used for factory calibration of VREFINTCAL register.
 | ||||
| pub const VREF_CALIB_MV: u32 = 3300; | ||||
| 
 | ||||
| const MAX_ADC_CLK_FREQ: Hertz = Hertz::mhz(25); | ||||
| 
 | ||||
| const TIME_ADC_VOLTAGE_REGUALTOR_STARTUP_US: u32 = 20; | ||||
| 
 | ||||
| const TEMP_CHANNEL: u8 = 9; | ||||
| const VREF_CHANNEL: u8 = 10; | ||||
| 
 | ||||
| const NUM_HW_CHANNELS: u8 = 22; | ||||
| const CHSELR_SQ_SIZE: usize = 8; | ||||
| const CHSELR_SQ_MAX_CHANNEL: u8 = 14; | ||||
| const CHSELR_SQ_SEQUENCE_END_MARKER: u8 = 0b1111; | ||||
| 
 | ||||
| // NOTE: Vrefint/Temperature/Vbat are not available on all ADCs,
 | ||||
| // this currently cannot be modeled with stm32-data,
 | ||||
| // so these are available from the software on all ADCs.
 | ||||
| /// Internal voltage reference channel.
 | ||||
| pub struct VrefInt; | ||||
| impl<T: Instance> AdcChannel<T> for VrefInt {} | ||||
| impl<T: Instance> SealedAdcChannel<T> for VrefInt { | ||||
|     fn channel(&self) -> u8 { | ||||
|         VREF_CHANNEL | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /// Internal temperature channel.
 | ||||
| pub struct Temperature; | ||||
| impl<T: Instance> AdcChannel<T> for Temperature {} | ||||
| impl<T: Instance> SealedAdcChannel<T> for Temperature { | ||||
|     fn channel(&self) -> u8 { | ||||
|         TEMP_CHANNEL | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #[derive(Debug)] | ||||
| pub enum Prescaler { | ||||
|     NotDivided, | ||||
|     DividedBy2, | ||||
|     DividedBy4, | ||||
|     DividedBy6, | ||||
|     DividedBy8, | ||||
|     DividedBy10, | ||||
|     DividedBy12, | ||||
|     DividedBy16, | ||||
|     DividedBy32, | ||||
|     DividedBy64, | ||||
|     DividedBy128, | ||||
|     DividedBy256, | ||||
| } | ||||
| 
 | ||||
| impl Prescaler { | ||||
|     fn from_ker_ck(frequency: Hertz) -> Self { | ||||
|         let raw_prescaler = frequency.0 / MAX_ADC_CLK_FREQ.0; | ||||
|         match raw_prescaler { | ||||
|             0 => Self::NotDivided, | ||||
|             1 => Self::DividedBy2, | ||||
|             2..=3 => Self::DividedBy4, | ||||
|             4..=5 => Self::DividedBy6, | ||||
|             6..=7 => Self::DividedBy8, | ||||
|             8..=9 => Self::DividedBy10, | ||||
|             10..=11 => Self::DividedBy12, | ||||
|             _ => unimplemented!(), | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     #[allow(unused)] | ||||
|     fn divisor(&self) -> u32 { | ||||
|         match self { | ||||
|             Prescaler::NotDivided => 1, | ||||
|             Prescaler::DividedBy2 => 2, | ||||
|             Prescaler::DividedBy4 => 4, | ||||
|             Prescaler::DividedBy6 => 6, | ||||
|             Prescaler::DividedBy8 => 8, | ||||
|             Prescaler::DividedBy10 => 10, | ||||
|             Prescaler::DividedBy12 => 12, | ||||
|             Prescaler::DividedBy16 => 16, | ||||
|             Prescaler::DividedBy32 => 32, | ||||
|             Prescaler::DividedBy64 => 64, | ||||
|             Prescaler::DividedBy128 => 128, | ||||
|             Prescaler::DividedBy256 => 256, | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fn presc(&self) -> Presc { | ||||
|         match self { | ||||
|             Prescaler::NotDivided => Presc::DIV1, | ||||
|             Prescaler::DividedBy2 => Presc::DIV2, | ||||
|             Prescaler::DividedBy4 => Presc::DIV4, | ||||
|             Prescaler::DividedBy6 => Presc::DIV6, | ||||
|             Prescaler::DividedBy8 => Presc::DIV8, | ||||
|             Prescaler::DividedBy10 => Presc::DIV10, | ||||
|             Prescaler::DividedBy12 => Presc::DIV12, | ||||
|             Prescaler::DividedBy16 => Presc::DIV16, | ||||
|             Prescaler::DividedBy32 => Presc::DIV32, | ||||
|             Prescaler::DividedBy64 => Presc::DIV64, | ||||
|             Prescaler::DividedBy128 => Presc::DIV128, | ||||
|             Prescaler::DividedBy256 => Presc::DIV256, | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #[cfg(feature = "defmt")] | ||||
| impl<'a> defmt::Format for Prescaler { | ||||
|     fn format(&self, fmt: defmt::Formatter) { | ||||
|         match self { | ||||
|             Prescaler::NotDivided => defmt::write!(fmt, "Prescaler::NotDivided"), | ||||
|             Prescaler::DividedBy2 => defmt::write!(fmt, "Prescaler::DividedBy2"), | ||||
|             Prescaler::DividedBy4 => defmt::write!(fmt, "Prescaler::DividedBy4"), | ||||
|             Prescaler::DividedBy6 => defmt::write!(fmt, "Prescaler::DividedBy6"), | ||||
|             Prescaler::DividedBy8 => defmt::write!(fmt, "Prescaler::DividedBy8"), | ||||
|             Prescaler::DividedBy10 => defmt::write!(fmt, "Prescaler::DividedBy10"), | ||||
|             Prescaler::DividedBy12 => defmt::write!(fmt, "Prescaler::DividedBy12"), | ||||
|             Prescaler::DividedBy16 => defmt::write!(fmt, "Prescaler::DividedBy16"), | ||||
|             Prescaler::DividedBy32 => defmt::write!(fmt, "Prescaler::DividedBy32"), | ||||
|             Prescaler::DividedBy64 => defmt::write!(fmt, "Prescaler::DividedBy64"), | ||||
|             Prescaler::DividedBy128 => defmt::write!(fmt, "Prescaler::DividedBy128"), | ||||
|             Prescaler::DividedBy256 => defmt::write!(fmt, "Prescaler::DividedBy256"), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /// Number of samples used for averaging.
 | ||||
| /// TODO: Implement hardware averaging setting.
 | ||||
| #[allow(unused)] | ||||
| pub enum Averaging { | ||||
|     Disabled, | ||||
|     Samples2, | ||||
|     Samples4, | ||||
|     Samples8, | ||||
|     Samples16, | ||||
|     Samples32, | ||||
|     Samples64, | ||||
|     Samples128, | ||||
|     Samples256, | ||||
|     Samples512, | ||||
|     Samples1024, | ||||
| } | ||||
| 
 | ||||
| impl<'d, T: Instance> Adc<'d, T> { | ||||
|     /// Create a new ADC driver.
 | ||||
|     pub fn new(adc: impl Peripheral<P = T> + 'd, sample_time: SampleTime, resolution: Resolution) -> Self { | ||||
|         embassy_hal_internal::into_ref!(adc); | ||||
|         rcc::enable_and_reset::<T>(); | ||||
| 
 | ||||
|         T::regs().cfgr2().modify(|w| w.set_ckmode(Ckmode::SYSCLK)); | ||||
| 
 | ||||
|         let prescaler = Prescaler::from_ker_ck(T::frequency()); | ||||
|         T::common_regs().ccr().modify(|w| w.set_presc(prescaler.presc())); | ||||
| 
 | ||||
|         let frequency = Hertz(T::frequency().0 / prescaler.divisor()); | ||||
|         debug!("ADC frequency set to {} Hz", frequency.0); | ||||
| 
 | ||||
|         if frequency > MAX_ADC_CLK_FREQ { | ||||
|             panic!("Maximal allowed frequency for the ADC is {} MHz and it varies with different packages, refer to ST docs for more information.", MAX_ADC_CLK_FREQ.0 /  1_000_000 ); | ||||
|         } | ||||
| 
 | ||||
|         let mut s = Self { | ||||
|             adc, | ||||
|             sample_time: SampleTime::from_bits(0), | ||||
|         }; | ||||
| 
 | ||||
|         s.power_up(); | ||||
| 
 | ||||
|         s.set_resolution(resolution); | ||||
| 
 | ||||
|         s.calibrate(); | ||||
| 
 | ||||
|         s.enable(); | ||||
| 
 | ||||
|         s.configure_default(); | ||||
| 
 | ||||
|         s.set_sample_time_all_channels(sample_time); | ||||
| 
 | ||||
|         s | ||||
|     } | ||||
| 
 | ||||
|     fn power_up(&mut self) { | ||||
|         T::regs().cr().modify(|reg| { | ||||
|             reg.set_advregen(true); | ||||
|         }); | ||||
| 
 | ||||
|         // "The software must wait for the ADC voltage regulator startup time."
 | ||||
|         // See datasheet for the value.
 | ||||
|         blocking_delay_us(TIME_ADC_VOLTAGE_REGUALTOR_STARTUP_US + 1); | ||||
|     } | ||||
| 
 | ||||
|     fn calibrate(&mut self) { | ||||
|         // We have to make sure AUTOFF is OFF, but keep its value after calibration.
 | ||||
|         let autoff_value = T::regs().cfgr1().read().autoff(); | ||||
|         T::regs().cfgr1().modify(|w| w.set_autoff(false)); | ||||
| 
 | ||||
|         T::regs().cr().modify(|w| w.set_adcal(true)); | ||||
| 
 | ||||
|         // "ADCAL bit stays at 1 during all the calibration sequence."
 | ||||
|         // "It is then cleared by hardware as soon the calibration completes."
 | ||||
|         while T::regs().cr().read().adcal() {} | ||||
| 
 | ||||
|         debug!("ADC calibration value: {}.", T::regs().dr().read().data()); | ||||
| 
 | ||||
|         T::regs().cfgr1().modify(|w| w.set_autoff(autoff_value)); | ||||
|     } | ||||
| 
 | ||||
|     fn enable(&mut self) { | ||||
|         T::regs().isr().modify(|w| w.set_adrdy(true)); | ||||
|         T::regs().cr().modify(|w| w.set_aden(true)); | ||||
|         // ADRDY is "ADC ready". Wait until it will be True.
 | ||||
|         while !T::regs().isr().read().adrdy() {} | ||||
|     } | ||||
| 
 | ||||
|     fn configure_default(&mut self) { | ||||
|         // single conversion mode, software trigger
 | ||||
|         T::regs().cfgr1().modify(|w| { | ||||
|             w.set_cont(false); | ||||
|             w.set_exten(Exten::DISABLED); | ||||
|             w.set_align(Align::RIGHT); | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     /// Enable reading the voltage reference internal channel.
 | ||||
|     pub fn enable_vrefint(&self) -> VrefInt { | ||||
|         T::common_regs().ccr().modify(|reg| { | ||||
|             reg.set_vrefen(true); | ||||
|         }); | ||||
| 
 | ||||
|         VrefInt {} | ||||
|     } | ||||
| 
 | ||||
|     /// Enable reading the temperature internal channel.
 | ||||
|     pub fn enable_temperature(&self) -> Temperature { | ||||
|         debug!("Ensure that sample time is set to more than temperature sensor T_start from the datasheet!"); | ||||
|         T::common_regs().ccr().modify(|reg| { | ||||
|             reg.set_tsen(true); | ||||
|         }); | ||||
| 
 | ||||
|         Temperature {} | ||||
|     } | ||||
| 
 | ||||
|     /// Set the ADC sample time.
 | ||||
|     /// Shall only be called when ADC is not converting.
 | ||||
|     pub fn set_sample_time_all_channels(&mut self, sample_time: SampleTime) { | ||||
|         self.sample_time = sample_time; | ||||
| 
 | ||||
|         // Set all channels to use SMP1 field as source.
 | ||||
|         T::regs().smpr().modify(|w| { | ||||
|             w.smpsel(0); | ||||
|             w.set_smp1(sample_time); | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     /// Set the ADC resolution.
 | ||||
|     pub fn set_resolution(&mut self, resolution: Resolution) { | ||||
|         T::regs().cfgr1().modify(|reg| reg.set_res(resolution)); | ||||
|     } | ||||
| 
 | ||||
|     /// Perform a single conversion.
 | ||||
|     fn convert(&mut self) -> u16 { | ||||
|         // Set single conversion mode.
 | ||||
|         T::regs().cfgr1().modify(|w| w.set_cont(false)); | ||||
| 
 | ||||
|         // Start conversion
 | ||||
|         T::regs().cr().modify(|reg| { | ||||
|             reg.set_adstart(true); | ||||
|         }); | ||||
| 
 | ||||
|         // Waiting for End Of Conversion (EOC).
 | ||||
|         while !T::regs().isr().read().eoc() {} | ||||
| 
 | ||||
|         T::regs().dr().read().data() as u16 | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_read(&mut self, channel: &mut impl AdcChannel<T>) -> u16 { | ||||
|         Self::configure_channel(channel); | ||||
|         T::regs().cfgr1().write(|reg| { | ||||
|             reg.set_chselrmod(false); | ||||
|             reg.set_align(Align::RIGHT); | ||||
|         }); | ||||
|         self.convert() | ||||
|     } | ||||
| 
 | ||||
|     fn setup_channel_sequencer<'a>(channel_sequence: impl ExactSizeIterator<Item = &'a mut AnyAdcChannel<T>>) { | ||||
|         assert!( | ||||
|             channel_sequence.len() <= CHSELR_SQ_SIZE, | ||||
|             "Seqenced read set cannot be more than {} in size.", | ||||
|             CHSELR_SQ_SIZE | ||||
|         ); | ||||
|         let mut last_sq_set: usize = 0; | ||||
|         T::regs().chselr_sq().write(|w| { | ||||
|             for (i, channel) in channel_sequence.enumerate() { | ||||
|                 assert!( | ||||
|                     channel.channel() <= CHSELR_SQ_MAX_CHANNEL, | ||||
|                     "Sequencer only support HW channels smaller than {}.", | ||||
|                     CHSELR_SQ_MAX_CHANNEL | ||||
|                 ); | ||||
|                 w.set_sq(i, channel.channel()); | ||||
|                 last_sq_set = i; | ||||
|             } | ||||
| 
 | ||||
|             for i in (last_sq_set + 1)..CHSELR_SQ_SIZE { | ||||
|                 w.set_sq(i, CHSELR_SQ_SEQUENCE_END_MARKER); | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|         Self::apply_channel_conf() | ||||
|     } | ||||
| 
 | ||||
|     async fn dma_convert(&mut self, rx_dma: &mut impl RxDma<T>, readings: &mut [u16]) { | ||||
|         // Enable overrun control, so no new DMA requests will be generated until
 | ||||
|         // previous DR values is read.
 | ||||
|         T::regs().isr().modify(|reg| { | ||||
|             reg.set_ovr(true); | ||||
|         }); | ||||
| 
 | ||||
|         // Set continuous mode with oneshot dma.
 | ||||
|         T::regs().cfgr1().modify(|reg| { | ||||
|             reg.set_discen(false); | ||||
|             reg.set_cont(true); | ||||
|             reg.set_dmacfg(Dmacfg::DMA_ONE_SHOT); | ||||
|             reg.set_dmaen(true); | ||||
|             reg.set_ovrmod(Ovrmod::PRESERVE); | ||||
|         }); | ||||
| 
 | ||||
|         let request = rx_dma.request(); | ||||
|         let transfer = unsafe { | ||||
|             Transfer::new_read( | ||||
|                 rx_dma, | ||||
|                 request, | ||||
|                 T::regs().dr().as_ptr() as *mut u16, | ||||
|                 readings, | ||||
|                 Default::default(), | ||||
|             ) | ||||
|         }; | ||||
| 
 | ||||
|         // Start conversion.
 | ||||
|         T::regs().cr().modify(|reg| { | ||||
|             reg.set_adstart(true); | ||||
|         }); | ||||
| 
 | ||||
|         // Wait for conversion sequence to finish.
 | ||||
|         transfer.await; | ||||
| 
 | ||||
|         // Ensure conversions are finished.
 | ||||
|         Self::cancel_conversions(); | ||||
| 
 | ||||
|         // Reset configuration.
 | ||||
|         T::regs().cfgr1().modify(|reg| { | ||||
|             reg.set_cont(false); | ||||
|             reg.set_dmacfg(Dmacfg::from_bits(0)); | ||||
|             reg.set_dmaen(false); | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     /// Read one or multiple ADC channels using DMA in hardware order.
 | ||||
|     /// Readings will be ordered based on **hardware** ADC channel number and `scandir` setting.
 | ||||
|     /// Readings won't be in the same order as in the `set`!
 | ||||
|     ///
 | ||||
|     /// In STM32C0, channels bigger than 14 cannot be read using sequencer, so you have to use
 | ||||
|     /// either blocking reads or use the mechanism to read in HW order (CHSELRMOD=0).
 | ||||
|     /// TODO(chudsaviet): externalize generic code and merge with read().
 | ||||
|     pub async fn read_in_hw_order( | ||||
|         &mut self, | ||||
|         rx_dma: &mut impl RxDma<T>, | ||||
|         hw_channel_selection: u32, | ||||
|         scandir: Scandir, | ||||
|         readings: &mut [u16], | ||||
|     ) { | ||||
|         assert!( | ||||
|             hw_channel_selection != 0, | ||||
|             "Some bits in `hw_channel_selection` shall be set." | ||||
|         ); | ||||
|         assert!( | ||||
|             (hw_channel_selection >> NUM_HW_CHANNELS) == 0, | ||||
|             "STM32C0 only have {} ADC channels. `hw_channel_selection` cannot have bits higher than this number set.", | ||||
|             NUM_HW_CHANNELS | ||||
|         ); | ||||
|         // To check for correct readings slice size, we shall solve Hamming weight problem,
 | ||||
|         // which is either slow or memory consuming.
 | ||||
|         // Since we have limited resources, we don't do it here.
 | ||||
|         // Not doing this have a great potential for a bug through.
 | ||||
| 
 | ||||
|         // Ensure no conversions are ongoing.
 | ||||
|         Self::cancel_conversions(); | ||||
| 
 | ||||
|         T::regs().cfgr1().modify(|reg| { | ||||
|             reg.set_chselrmod(false); | ||||
|             reg.set_scandir(scandir); | ||||
|             reg.set_align(Align::RIGHT); | ||||
|         }); | ||||
| 
 | ||||
|         // Set required channels for multi-convert.
 | ||||
|         unsafe { (T::regs().chselr().as_ptr() as *mut u32).write_volatile(hw_channel_selection) } | ||||
| 
 | ||||
|         Self::apply_channel_conf(); | ||||
| 
 | ||||
|         self.dma_convert(rx_dma, readings).await | ||||
|     } | ||||
| 
 | ||||
|     // Read ADC channels in specified order using DMA (CHSELRMOD = 1).
 | ||||
|     // In STM32C0, only lower 14 ADC channels can be read this way.
 | ||||
|     // For other channels, use `read_in_hw_order()` or blocking read.
 | ||||
|     pub async fn read( | ||||
|         &mut self, | ||||
|         rx_dma: &mut impl RxDma<T>, | ||||
|         channel_sequence: impl ExactSizeIterator<Item = &mut AnyAdcChannel<T>>, | ||||
|         readings: &mut [u16], | ||||
|     ) { | ||||
|         assert!( | ||||
|             channel_sequence.len() != 0, | ||||
|             "Asynchronous read channel sequence cannot be empty." | ||||
|         ); | ||||
|         assert!( | ||||
|             channel_sequence.len() == readings.len(), | ||||
|             "Channel sequence length must be equal to readings length." | ||||
|         ); | ||||
| 
 | ||||
|         // Ensure no conversions are ongoing.
 | ||||
|         Self::cancel_conversions(); | ||||
| 
 | ||||
|         T::regs().cfgr1().modify(|reg| { | ||||
|             reg.set_chselrmod(true); | ||||
|             reg.set_align(Align::RIGHT); | ||||
|         }); | ||||
| 
 | ||||
|         Self::setup_channel_sequencer(channel_sequence); | ||||
| 
 | ||||
|         self.dma_convert(rx_dma, readings).await | ||||
|     } | ||||
| 
 | ||||
|     fn configure_channel(channel: &mut impl AdcChannel<T>) { | ||||
|         channel.setup(); | ||||
|         // write() because we want all other bits to be set to 0.
 | ||||
|         T::regs() | ||||
|             .chselr() | ||||
|             .write(|w| w.set_chsel(channel.channel().into(), true)); | ||||
| 
 | ||||
|         Self::apply_channel_conf(); | ||||
|     } | ||||
| 
 | ||||
|     fn apply_channel_conf() { | ||||
|         // Trigger and wait for the channel selection procedure to complete.
 | ||||
|         T::regs().isr().modify(|w| w.set_ccrdy(false)); | ||||
|         while !T::regs().isr().read().ccrdy() {} | ||||
|     } | ||||
| 
 | ||||
|     fn cancel_conversions() { | ||||
|         if T::regs().cr().read().adstart() && !T::regs().cr().read().addis() { | ||||
|             T::regs().cr().modify(|reg| { | ||||
|                 reg.set_adstp(Adstp::STOP); | ||||
|             }); | ||||
|             while T::regs().cr().read().adstart() {} | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -14,6 +14,7 @@ | ||||
| #[cfg_attr(any(adc_v3, adc_g0, adc_h5, adc_u0), path = "v3.rs")] | ||||
| #[cfg_attr(any(adc_v4, adc_u5), path = "v4.rs")] | ||||
| #[cfg_attr(adc_g4, path = "g4.rs")] | ||||
| #[cfg_attr(adc_c0, path = "c0.rs")] | ||||
| mod _version; | ||||
| 
 | ||||
| use core::marker::PhantomData; | ||||
| @ -71,7 +72,7 @@ trait SealedInstance { | ||||
| } | ||||
| 
 | ||||
| pub(crate) trait SealedAdcChannel<T> { | ||||
|     #[cfg(any(adc_v1, adc_l0, adc_v2, adc_g4, adc_v4, adc_u5))] | ||||
|     #[cfg(any(adc_v1, adc_c0, adc_l0, adc_v2, adc_g4, adc_v4, adc_u5))] | ||||
|     fn setup(&mut self) {} | ||||
| 
 | ||||
|     #[allow(unused)] | ||||
| @ -106,7 +107,8 @@ pub(crate) fn blocking_delay_us(us: u32) { | ||||
|     adc_g0, | ||||
|     adc_u0, | ||||
|     adc_h5, | ||||
|     adc_u5 | ||||
|     adc_u5, | ||||
|     adc_c0 | ||||
| )))] | ||||
| #[allow(private_bounds)] | ||||
| pub trait Instance: SealedInstance + crate::Peripheral<P = Self> { | ||||
| @ -126,7 +128,8 @@ pub trait Instance: SealedInstance + crate::Peripheral<P = Self> { | ||||
|     adc_g0, | ||||
|     adc_u0, | ||||
|     adc_h5, | ||||
|     adc_u5 | ||||
|     adc_u5, | ||||
|     adc_c0 | ||||
| ))] | ||||
| #[allow(private_bounds)] | ||||
| pub trait Instance: SealedInstance + crate::Peripheral<P = Self> + crate::rcc::RccPeripheral { | ||||
| @ -164,6 +167,13 @@ impl<T: Instance> SealedAdcChannel<T> for AnyAdcChannel<T> { | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<T> AnyAdcChannel<T> { | ||||
|     #[allow(unused)] | ||||
|     pub fn get_hw_channel(&self) -> u8 { | ||||
|         self.channel | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #[cfg(adc_u5)] | ||||
| foreach_adc!( | ||||
|     (ADC4, $common_inst:ident, $clock:ident) => { | ||||
| @ -225,7 +235,7 @@ macro_rules! impl_adc_pin { | ||||
|     ($inst:ident, $pin:ident, $ch:expr) => { | ||||
|         impl crate::adc::AdcChannel<peripherals::$inst> for crate::peripherals::$pin {} | ||||
|         impl crate::adc::SealedAdcChannel<peripherals::$inst> for crate::peripherals::$pin { | ||||
|             #[cfg(any(adc_v1, adc_l0, adc_v2, adc_g4, adc_v4, adc_u5))] | ||||
|             #[cfg(any(adc_v1, adc_c0, adc_l0, adc_v2, adc_g4, adc_v4, adc_u5))] | ||||
|             fn setup(&mut self) { | ||||
|                 <Self as crate::gpio::SealedPin>::set_as_analog(self); | ||||
|             } | ||||
| @ -254,7 +264,7 @@ pub const fn resolution_to_max_count(res: Resolution) -> u32 { | ||||
|         Resolution::BITS12 => (1 << 12) - 1, | ||||
|         Resolution::BITS10 => (1 << 10) - 1, | ||||
|         Resolution::BITS8 => (1 << 8) - 1, | ||||
|         #[cfg(any(adc_v1, adc_v2, adc_v3, adc_l0, adc_g0, adc_f3, adc_f3_v1_1, adc_h5))] | ||||
|         #[cfg(any(adc_v1, adc_v2, adc_v3, adc_l0, adc_c0, adc_g0, adc_f3, adc_f3_v1_1, adc_h5))] | ||||
|         Resolution::BITS6 => (1 << 6) - 1, | ||||
|         #[allow(unreachable_patterns)] | ||||
|         _ => core::unreachable!(), | ||||
|  | ||||
| @ -180,6 +180,9 @@ pub(crate) unsafe fn init(config: Config) { | ||||
|         lsi: None, | ||||
|         lse: None, | ||||
|     ); | ||||
| 
 | ||||
|     RCC.ccipr() | ||||
|         .modify(|w| w.set_adc1sel(stm32_metapac::rcc::vals::Adcsel::SYS)); | ||||
| } | ||||
| 
 | ||||
| mod max { | ||||
|  | ||||
							
								
								
									
										57
									
								
								examples/stm32c0/src/bin/adc.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								examples/stm32c0/src/bin/adc.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,57 @@ | ||||
| #![no_std] | ||||
| #![no_main] | ||||
| 
 | ||||
| use defmt::*; | ||||
| use embassy_executor::Spawner; | ||||
| use embassy_stm32::adc::vals::Scandir; | ||||
| use embassy_stm32::adc::{Adc, AdcChannel, AnyAdcChannel, Resolution, SampleTime}; | ||||
| use embassy_stm32::peripherals::ADC1; | ||||
| use embassy_time::Timer; | ||||
| use {defmt_rtt as _, panic_probe as _}; | ||||
| 
 | ||||
| #[embassy_executor::main] | ||||
| async fn main(_spawner: Spawner) { | ||||
|     let config = Default::default(); | ||||
|     let p = embassy_stm32::init(config); | ||||
| 
 | ||||
|     info!("ADC STM32C0 example."); | ||||
| 
 | ||||
|     // We need to set certain sample time to be able to read temp sensor.
 | ||||
|     let mut adc = Adc::new(p.ADC1, SampleTime::CYCLES12_5, Resolution::BITS12); | ||||
|     let mut temp = adc.enable_temperature().degrade_adc(); | ||||
|     let mut vref = adc.enable_vrefint().degrade_adc(); | ||||
|     let mut pin0 = p.PA0.degrade_adc(); | ||||
| 
 | ||||
|     let mut dma = p.DMA1_CH1; | ||||
|     let mut read_buffer: [u16; 3] = [0; 3]; | ||||
| 
 | ||||
|     loop { | ||||
|         info!("============================"); | ||||
|         let blocking_temp = adc.blocking_read(&mut temp); | ||||
|         let blocking_vref = adc.blocking_read(&mut vref); | ||||
|         let blocing_pin0 = adc.blocking_read(&mut pin0); | ||||
|         info!( | ||||
|             "Blocking ADC read: vref = {}, temp = {}, pin0 = {}.", | ||||
|             blocking_vref, blocking_temp, blocing_pin0 | ||||
|         ); | ||||
| 
 | ||||
|         let channels_seqence: [&mut AnyAdcChannel<ADC1>; 3] = [&mut vref, &mut temp, &mut pin0]; | ||||
|         adc.read(&mut dma, channels_seqence.into_iter(), &mut read_buffer).await; | ||||
|         // Values are ordered according to hardware ADC channel number!
 | ||||
|         info!( | ||||
|             "DMA ADC read in set: vref = {}, temp = {}, pin0 = {}.", | ||||
|             read_buffer[0], read_buffer[1], read_buffer[2] | ||||
|         ); | ||||
| 
 | ||||
|         let hw_channel_selection: u32 = | ||||
|             (1 << temp.get_hw_channel()) + (1 << vref.get_hw_channel()) + (1 << pin0.get_hw_channel()); | ||||
|         adc.read_in_hw_order(&mut dma, hw_channel_selection, Scandir::UP, &mut read_buffer) | ||||
|             .await; | ||||
|         info!( | ||||
|             "DMA ADC read in hardware order: vref = {}, temp = {}, pin0 = {}.", | ||||
|             read_buffer[2], read_buffer[1], read_buffer[0] | ||||
|         ); | ||||
| 
 | ||||
|         Timer::after_millis(2000).await; | ||||
|     } | ||||
| } | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user