mirror of
https://github.com/embassy-rs/embassy.git
synced 2024-11-22 06:42:32 +00:00
Merge pull request #3169 from Adancurusul/g4_dev
Add adc oversampling and adc differential for g4
This commit is contained in:
commit
1ea29f1d2e
@ -1,5 +1,9 @@
|
|||||||
#[allow(unused)]
|
#[allow(unused)]
|
||||||
|
#[cfg(stm32h7)]
|
||||||
use pac::adc::vals::{Adcaldif, Difsel, Exten};
|
use pac::adc::vals::{Adcaldif, Difsel, Exten};
|
||||||
|
#[allow(unused)]
|
||||||
|
#[cfg(stm32g4)]
|
||||||
|
use pac::adc::vals::{Adcaldif, Difsel, Exten, Rovsm, Trovs};
|
||||||
use pac::adccommon::vals::Presc;
|
use pac::adccommon::vals::Presc;
|
||||||
|
|
||||||
use super::{blocking_delay_us, Adc, AdcChannel, Instance, Resolution, SampleTime};
|
use super::{blocking_delay_us, Adc, AdcChannel, Instance, Resolution, SampleTime};
|
||||||
@ -228,6 +232,68 @@ impl<'d, T: Instance> Adc<'d, T> {
|
|||||||
Vbat {}
|
Vbat {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Enable differential channel.
|
||||||
|
/// Caution:
|
||||||
|
/// : When configuring the channel “i” in differential input mode, its negative input voltage VINN[i]
|
||||||
|
/// is connected to another channel. As a consequence, this channel is no longer usable in
|
||||||
|
/// single-ended mode or in differential mode and must never be configured to be converted.
|
||||||
|
/// Some channels are shared between ADC1/ADC2/ADC3/ADC4/ADC5: this can make the
|
||||||
|
/// channel on the other ADC unusable. The only exception is when ADC master and the slave
|
||||||
|
/// operate in interleaved mode.
|
||||||
|
#[cfg(stm32g4)]
|
||||||
|
pub fn set_differential_channel(&mut self, ch: usize, enable: bool) {
|
||||||
|
T::regs().cr().modify(|w| w.set_aden(false)); // disable adc
|
||||||
|
T::regs().difsel().modify(|w| {
|
||||||
|
w.set_difsel(
|
||||||
|
ch,
|
||||||
|
if enable {
|
||||||
|
Difsel::DIFFERENTIAL
|
||||||
|
} else {
|
||||||
|
Difsel::SINGLEENDED
|
||||||
|
},
|
||||||
|
);
|
||||||
|
});
|
||||||
|
T::regs().cr().modify(|w| w.set_aden(true));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(stm32g4)]
|
||||||
|
pub fn set_differential(&mut self, channel: &mut impl AdcChannel<T>, enable: bool) {
|
||||||
|
self.set_differential_channel(channel.channel() as usize, enable);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Set oversampling shift.
|
||||||
|
#[cfg(stm32g4)]
|
||||||
|
pub fn set_oversampling_shift(&mut self, shift: u8) {
|
||||||
|
T::regs().cfgr2().modify(|reg| reg.set_ovss(shift));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Set oversampling ratio.
|
||||||
|
#[cfg(stm32g4)]
|
||||||
|
pub fn set_oversampling_ratio(&mut self, ratio: u8) {
|
||||||
|
T::regs().cfgr2().modify(|reg| reg.set_ovsr(ratio));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Enable oversampling in regular mode.
|
||||||
|
#[cfg(stm32g4)]
|
||||||
|
pub fn enable_regular_oversampling_mode(&mut self, mode: Rovsm, trig_mode: Trovs, enable: bool) {
|
||||||
|
T::regs().cfgr2().modify(|reg| reg.set_trovs(trig_mode));
|
||||||
|
T::regs().cfgr2().modify(|reg| reg.set_rovsm(mode));
|
||||||
|
T::regs().cfgr2().modify(|reg| reg.set_rovse(enable));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reads that are not implemented as INJECTED in "blocking_read"
|
||||||
|
// #[cfg(stm32g4)]
|
||||||
|
// pub fn enalble_injected_oversampling_mode(&mut self, enable: bool) {
|
||||||
|
// T::regs().cfgr2().modify(|reg| reg.set_jovse(enable));
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[cfg(stm32g4)]
|
||||||
|
// pub fn enable_oversampling_regular_injected_mode(&mut self, enable: bool) {
|
||||||
|
// // the regularoversampling mode is forced to resumed mode (ROVSM bit ignored),
|
||||||
|
// T::regs().cfgr2().modify(|reg| reg.set_rovse(enable));
|
||||||
|
// T::regs().cfgr2().modify(|reg| reg.set_jovse(enable));
|
||||||
|
// }
|
||||||
|
|
||||||
/// Set the ADC sample time.
|
/// Set the ADC sample time.
|
||||||
pub fn set_sample_time(&mut self, sample_time: SampleTime) {
|
pub fn set_sample_time(&mut self, sample_time: SampleTime) {
|
||||||
self.sample_time = sample_time;
|
self.sample_time = sample_time;
|
||||||
|
@ -24,6 +24,7 @@ pub use _version::*;
|
|||||||
#[cfg(any(adc_f1, adc_f3, adc_v1, adc_l0, adc_f3_v1_1))]
|
#[cfg(any(adc_f1, adc_f3, adc_v1, adc_l0, adc_f3_v1_1))]
|
||||||
use embassy_sync::waitqueue::AtomicWaker;
|
use embassy_sync::waitqueue::AtomicWaker;
|
||||||
|
|
||||||
|
pub use crate::pac::adc::vals;
|
||||||
#[cfg(not(any(adc_f1, adc_f3_v2)))]
|
#[cfg(not(any(adc_f1, adc_f3_v2)))]
|
||||||
pub use crate::pac::adc::vals::Res as Resolution;
|
pub use crate::pac::adc::vals::Res as Resolution;
|
||||||
pub use crate::pac::adc::vals::SampleTime;
|
pub use crate::pac::adc::vals::SampleTime;
|
||||||
|
47
examples/stm32g4/src/bin/adc_differential.rs
Normal file
47
examples/stm32g4/src/bin/adc_differential.rs
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
//! adc differential mode example
|
||||||
|
//!
|
||||||
|
//! This example uses adc1 in differential mode
|
||||||
|
//! p:pa0 n:pa1
|
||||||
|
|
||||||
|
#![no_std]
|
||||||
|
#![no_main]
|
||||||
|
|
||||||
|
use defmt::*;
|
||||||
|
use embassy_executor::Spawner;
|
||||||
|
use embassy_stm32::adc::{Adc, SampleTime};
|
||||||
|
use embassy_stm32::Config;
|
||||||
|
use embassy_time::Timer;
|
||||||
|
use {defmt_rtt as _, panic_probe as _};
|
||||||
|
|
||||||
|
#[embassy_executor::main]
|
||||||
|
async fn main(_spawner: Spawner) {
|
||||||
|
let mut config = Config::default();
|
||||||
|
{
|
||||||
|
use embassy_stm32::rcc::*;
|
||||||
|
config.rcc.pll = Some(Pll {
|
||||||
|
source: PllSource::HSI,
|
||||||
|
prediv: PllPreDiv::DIV4,
|
||||||
|
mul: PllMul::MUL85,
|
||||||
|
divp: None,
|
||||||
|
divq: None,
|
||||||
|
// Main system clock at 170 MHz
|
||||||
|
divr: Some(PllRDiv::DIV2),
|
||||||
|
});
|
||||||
|
config.rcc.mux.adc12sel = mux::Adcsel::SYS;
|
||||||
|
config.rcc.sys = Sysclk::PLL1_R;
|
||||||
|
}
|
||||||
|
let mut p = embassy_stm32::init(config);
|
||||||
|
|
||||||
|
let mut adc = Adc::new(p.ADC1);
|
||||||
|
adc.set_sample_time(SampleTime::CYCLES247_5);
|
||||||
|
adc.set_differential(&mut p.PA0, true); //p:pa0,n:pa1
|
||||||
|
|
||||||
|
// can also use
|
||||||
|
// adc.set_differential_channel(1, true);
|
||||||
|
info!("adc initialized");
|
||||||
|
loop {
|
||||||
|
let measured = adc.blocking_read(&mut p.PA0);
|
||||||
|
info!("data: {}", measured);
|
||||||
|
Timer::after_millis(500).await;
|
||||||
|
}
|
||||||
|
}
|
57
examples/stm32g4/src/bin/adc_oversampling.rs
Normal file
57
examples/stm32g4/src/bin/adc_oversampling.rs
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
//! adc oversampling example
|
||||||
|
//!
|
||||||
|
//! This example uses adc oversampling to achieve 16bit data
|
||||||
|
|
||||||
|
#![no_std]
|
||||||
|
#![no_main]
|
||||||
|
|
||||||
|
use defmt::*;
|
||||||
|
use embassy_executor::Spawner;
|
||||||
|
use embassy_stm32::adc::vals::{Rovsm, Trovs};
|
||||||
|
use embassy_stm32::adc::{Adc, SampleTime};
|
||||||
|
use embassy_stm32::Config;
|
||||||
|
use embassy_time::Timer;
|
||||||
|
use {defmt_rtt as _, panic_probe as _};
|
||||||
|
|
||||||
|
#[embassy_executor::main]
|
||||||
|
async fn main(_spawner: Spawner) {
|
||||||
|
let mut config = Config::default();
|
||||||
|
{
|
||||||
|
use embassy_stm32::rcc::*;
|
||||||
|
config.rcc.pll = Some(Pll {
|
||||||
|
source: PllSource::HSI,
|
||||||
|
prediv: PllPreDiv::DIV4,
|
||||||
|
mul: PllMul::MUL85,
|
||||||
|
divp: None,
|
||||||
|
divq: None,
|
||||||
|
// Main system clock at 170 MHz
|
||||||
|
divr: Some(PllRDiv::DIV2),
|
||||||
|
});
|
||||||
|
config.rcc.mux.adc12sel = mux::Adcsel::SYS;
|
||||||
|
config.rcc.sys = Sysclk::PLL1_R;
|
||||||
|
}
|
||||||
|
let mut p = embassy_stm32::init(config);
|
||||||
|
|
||||||
|
let mut adc = Adc::new(p.ADC1);
|
||||||
|
adc.set_sample_time(SampleTime::CYCLES6_5);
|
||||||
|
// From https://www.st.com/resource/en/reference_manual/rm0440-stm32g4-series-advanced-armbased-32bit-mcus-stmicroelectronics.pdf
|
||||||
|
// page652 Oversampler
|
||||||
|
// Table 172. Maximum output results vs N and M. Grayed values indicates truncation
|
||||||
|
// 0x00 oversampling ratio X2
|
||||||
|
// 0x01 oversampling ratio X4
|
||||||
|
// 0x02 oversampling ratio X8
|
||||||
|
// 0x03 oversampling ratio X16
|
||||||
|
// 0x04 oversampling ratio X32
|
||||||
|
// 0x05 oversampling ratio X64
|
||||||
|
// 0x06 oversampling ratio X128
|
||||||
|
// 0x07 oversampling ratio X256
|
||||||
|
adc.set_oversampling_ratio(0x03); // ratio X3
|
||||||
|
adc.set_oversampling_shift(0b0000); // no shift
|
||||||
|
adc.enable_regular_oversampling_mode(Rovsm::RESUMED, Trovs::AUTOMATIC, true);
|
||||||
|
|
||||||
|
loop {
|
||||||
|
let measured = adc.blocking_read(&mut p.PA0);
|
||||||
|
info!("data: 0x{:X}", measured); //max 0xFFF0 -> 65520
|
||||||
|
Timer::after_millis(500).await;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user