Added ADC support for STM32C0.

This commit is contained in:
Timofei Korostelev 2025-02-17 21:40:35 -08:00 committed by Dario Nieuwenhuis
parent 9407ac67d3
commit 8c6fa83006
4 changed files with 543 additions and 5 deletions

468
embassy-stm32/src/adc/c0.rs Normal file
View 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() {}
}
}
}

View File

@ -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!(),

View File

@ -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 {

View 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;
}
}