Skip to content

Commit 17255ac

Browse files
authored
Enable async ADC for C2 and H2 (#3082)
* Enable async ADC for H2 * Enable for C2 * Prevent disabling interrupt if an async ADC is in use * Introduce adc1/2 symbols
1 parent 9a34d0e commit 17255ac

File tree

13 files changed

+66
-34
lines changed

13 files changed

+66
-34
lines changed

esp-hal/CHANGELOG.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
1111
- SPI: Added support for 3-wire SPI (#2919)
1212
- UART: Add separate config for Rx and Tx (#2965)
1313
- Added accessor methods to config structs (#3011)
14+
- Async support for ADC oneshot reads for ESP32C2, ESP32C3, ESP32C6 and ESP32H2 (#2925, #3082)
1415

1516
### Changed
1617

@@ -106,7 +107,6 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
106107
- ESP32-S2: Made Wi-Fi peripheral non virtual. (#2942)
107108
- `UartRx::check_for_errors`, `Uart::check_for_rx_errors`, `{Uart, UartRx}::read_buffered_bytes` (#2935)
108109
- Added `i2c` interrupt API (#2944)
109-
- Async support for ADC oneshot reads for ESP32C3 and ESP32C6 (#2925)
110110

111111
### Changed
112112

esp-hal/src/analog/adc/riscv.rs

+49-23
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,12 @@
11
use core::marker::PhantomData;
22

3-
#[cfg(esp32c3)]
4-
use Interrupt::APB_ADC as InterruptSource;
5-
#[cfg(esp32c6)]
6-
use Interrupt::APB_SARADC as InterruptSource;
3+
cfg_if::cfg_if! {
4+
if #[cfg(esp32c6)] {
5+
use Interrupt::APB_SARADC as InterruptSource;
6+
} else {
7+
use Interrupt::APB_ADC as InterruptSource;
8+
}
9+
}
710

811
#[cfg(not(esp32h2))]
912
pub use self::calibration::*;
@@ -12,17 +15,13 @@ use super::{AdcCalSource, AdcConfig, Attenuation};
1215
use crate::clock::clocks_ll::regi2c_write_mask;
1316
#[cfg(any(esp32c2, esp32c3, esp32c6))]
1417
use crate::efuse::Efuse;
15-
#[cfg(any(esp32c3, esp32c6))]
1618
use crate::{
1719
analog::adc::asynch::AdcFuture,
1820
interrupt::{InterruptConfigurable, InterruptHandler},
19-
peripherals::Interrupt,
20-
Async,
21-
};
22-
use crate::{
2321
peripheral::PeripheralRef,
24-
peripherals::APB_SARADC,
22+
peripherals::{Interrupt, APB_SARADC},
2523
system::{GenericPeripheralGuard, Peripheral},
24+
Async,
2625
Blocking,
2726
};
2827

@@ -51,7 +50,7 @@ fn regi2c_write_mask(block: u8, host_id: u8, reg_add: u8, msb: u8, lsb: u8, data
5150
// https://github.com/espressif/esp-idf/blob/903af13e8/components/soc/esp32h2/include/soc/regi2c_saradc.h
5251
// https://github.com/espressif/esp-idf/blob/903af13e8/components/soc/esp32h4/include/soc/regi2c_saradc.h
5352
cfg_if::cfg_if! {
54-
if #[cfg(any(esp32c2, esp32c3, esp32c6, esp32h2))] {
53+
if #[cfg(adc1)] {
5554
const I2C_SAR_ADC: u8 = 0x69;
5655
const I2C_SAR_ADC_HOSTID: u8 = 0;
5756

@@ -82,7 +81,7 @@ cfg_if::cfg_if! {
8281
}
8382

8483
cfg_if::cfg_if! {
85-
if #[cfg(esp32c3)] {
84+
if #[cfg(adc2)] {
8685
const ADC_SAR2_ENCAL_GND_ADDR: u8 = 0x7;
8786
const ADC_SAR2_ENCAL_GND_ADDR_MSB: u8 = 7;
8887
const ADC_SAR2_ENCAL_GND_ADDR_LSB: u8 = 7;
@@ -194,6 +193,7 @@ pub trait RegisterAccess {
194193
fn set_init_code(data: u16);
195194
}
196195

196+
#[cfg(adc1)]
197197
impl RegisterAccess for crate::peripherals::ADC1 {
198198
fn config_onetime_sample(channel: u8, attenuation: u8) {
199199
APB_SARADC::regs().onetime_sample().modify(|_, w| unsafe {
@@ -256,6 +256,7 @@ impl RegisterAccess for crate::peripherals::ADC1 {
256256
}
257257
}
258258

259+
#[cfg(adc1)]
259260
impl super::CalibrationAccess for crate::peripherals::ADC1 {
260261
const ADC_CAL_CNT_MAX: u16 = ADC_CAL_CNT_MAX;
261262
const ADC_CAL_CHANNEL: u16 = ADC_CAL_CHANNEL;
@@ -296,7 +297,7 @@ impl super::CalibrationAccess for crate::peripherals::ADC1 {
296297
}
297298
}
298299

299-
#[cfg(esp32c3)]
300+
#[cfg(adc2)]
300301
impl RegisterAccess for crate::peripherals::ADC2 {
301302
fn config_onetime_sample(channel: u8, attenuation: u8) {
302303
APB_SARADC::regs().onetime_sample().modify(|_, w| unsafe {
@@ -357,7 +358,7 @@ impl RegisterAccess for crate::peripherals::ADC2 {
357358
}
358359
}
359360

360-
#[cfg(esp32c3)]
361+
#[cfg(adc2)]
361362
impl super::CalibrationAccess for crate::peripherals::ADC2 {
362363
const ADC_CAL_CNT_MAX: u16 = ADC_CAL_CNT_MAX;
363364
const ADC_CAL_CHANNEL: u16 = ADC_CAL_CHANNEL;
@@ -435,9 +436,9 @@ where
435436
}
436437
}
437438

438-
#[cfg(any(esp32c3, esp32c6))]
439439
/// Reconfigures the ADC driver to operate in asynchronous mode.
440440
pub fn into_async(mut self) -> Adc<'d, ADCI, Async> {
441+
asynch::acquire_async_adc();
441442
self.set_interrupt_handler(asynch::adc_interrupt_handler);
442443

443444
// Reset interrupt flags and disable oneshot reading to normalize state before
@@ -531,7 +532,6 @@ where
531532

532533
impl<ADCI> crate::private::Sealed for Adc<'_, ADCI, Blocking> {}
533534

534-
#[cfg(any(esp32c3, esp32c6))]
535535
impl<ADCI> InterruptConfigurable for Adc<'_, ADCI, Blocking> {
536536
fn set_interrupt_handler(&mut self, handler: InterruptHandler) {
537537
for core in crate::Cpu::other() {
@@ -545,7 +545,7 @@ impl<ADCI> InterruptConfigurable for Adc<'_, ADCI, Blocking> {
545545
}
546546
}
547547

548-
#[cfg(any(esp32c2, esp32c3, esp32c6))]
548+
#[cfg(all(adc1, not(esp32h2)))]
549549
impl super::AdcCalEfuse for crate::peripherals::ADC1 {
550550
fn init_code(atten: Attenuation) -> Option<u16> {
551551
Efuse::rtc_calib_init_code(1, atten)
@@ -560,7 +560,7 @@ impl super::AdcCalEfuse for crate::peripherals::ADC1 {
560560
}
561561
}
562562

563-
#[cfg(esp32c3)]
563+
#[cfg(adc2)]
564564
impl super::AdcCalEfuse for crate::peripherals::ADC2 {
565565
fn init_code(atten: Attenuation) -> Option<u16> {
566566
Efuse::rtc_calib_init_code(2, atten)
@@ -635,14 +635,18 @@ mod adc_implementation {
635635
}
636636
}
637637

638-
#[cfg(any(esp32c3, esp32c6))]
639638
impl<'d, ADCI> Adc<'d, ADCI, Async>
640639
where
641640
ADCI: RegisterAccess + 'd,
642641
{
643642
/// Create a new instance in [crate::Blocking] mode.
644643
pub fn into_blocking(self) -> Adc<'d, ADCI, Blocking> {
645-
crate::interrupt::disable(crate::Cpu::current(), InterruptSource);
644+
if asynch::release_async_adc() {
645+
// Disable ADC interrupt on all cores if the last async ADC instance is disabled
646+
for cpu in crate::Cpu::all() {
647+
crate::interrupt::disable(cpu, InterruptSource);
648+
}
649+
}
646650
Adc {
647651
_adc: self._adc,
648652
attenuations: self.attenuations,
@@ -697,7 +701,6 @@ where
697701
}
698702
}
699703

700-
#[cfg(any(esp32c3, esp32c6))]
701704
/// Async functionality
702705
pub(crate) mod asynch {
703706
use core::{
@@ -706,20 +709,42 @@ pub(crate) mod asynch {
706709
task::{Context, Poll},
707710
};
708711

712+
// We only have to count on devices that have multiple ADCs sharing the same interrupt
713+
#[cfg(all(adc1, adc2))]
714+
use portable_atomic::{AtomicU32, Ordering};
709715
use procmacros::handler;
710716

711717
use crate::{asynch::AtomicWaker, peripherals::APB_SARADC, Async};
712718

719+
#[cfg(all(adc1, adc2))]
720+
static ASYNC_ADC_COUNT: AtomicU32 = AtomicU32::new(0);
721+
722+
pub(super) fn acquire_async_adc() {
723+
#[cfg(all(adc1, adc2))]
724+
ASYNC_ADC_COUNT.fetch_add(1, Ordering::Relaxed);
725+
}
726+
727+
pub(super) fn release_async_adc() -> bool {
728+
cfg_if::cfg_if! {
729+
if #[cfg(all(adc1, adc2))] {
730+
ASYNC_ADC_COUNT.fetch_sub(1, Ordering::Relaxed) == 1
731+
} else {
732+
true
733+
}
734+
}
735+
}
736+
713737
#[handler]
714738
pub(crate) fn adc_interrupt_handler() {
715739
let saradc = APB_SARADC::regs();
716740
let interrupt_status = saradc.int_st().read();
717741

742+
#[cfg(adc1)]
718743
if interrupt_status.adc1_done().bit_is_set() {
719744
handle_async(crate::peripherals::ADC1)
720745
}
721746

722-
#[cfg(esp32c3)]
747+
#[cfg(adc2)]
723748
if interrupt_status.adc2_done().bit_is_set() {
724749
handle_async(crate::peripherals::ADC2)
725750
}
@@ -745,6 +770,7 @@ pub(crate) mod asynch {
745770
fn waker() -> &'static AtomicWaker;
746771
}
747772

773+
#[cfg(adc1)]
748774
impl AsyncAccess for crate::peripherals::ADC1 {
749775
fn enable_interrupt() {
750776
APB_SARADC::regs()
@@ -771,7 +797,7 @@ pub(crate) mod asynch {
771797
}
772798
}
773799

774-
#[cfg(esp32c3)]
800+
#[cfg(adc2)]
775801
impl AsyncAccess for crate::peripherals::ADC2 {
776802
fn enable_interrupt() {
777803
APB_SARADC::regs()

esp-hal/src/analog/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55
//! available on the device. For more information about a peripheral driver,
66
//! please refer to the relevant module documentation.
77
8-
#[cfg(adc)]
8+
#[cfg(any(adc1, adc2))]
99
pub mod adc;
1010
#[cfg(dac)]
1111
pub mod dac;

esp-hal/src/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -228,7 +228,7 @@ pub(crate) use unstable_module;
228228
unstable_module! {
229229
#[cfg(aes)]
230230
pub mod aes;
231-
#[cfg(any(adc, dac))]
231+
#[cfg(any(adc1, adc2, dac))]
232232
pub mod analog;
233233
pub mod asynch;
234234
#[cfg(assist_debug)]

esp-metadata/devices/esp32.toml

+2-1
Original file line numberDiff line numberDiff line change
@@ -52,7 +52,8 @@ peripherals = [
5252

5353
symbols = [
5454
# Additional peripherals defined by us (the developers):
55-
"adc",
55+
"adc1",
56+
"adc2",
5657
"dac",
5758
"pdma",
5859
"phy",

esp-metadata/devices/esp32c2.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ peripherals = [
3434

3535
symbols = [
3636
# Additional peripherals defined by us (the developers):
37-
"adc",
37+
"adc1",
3838
"assist_debug_sp_monitor",
3939
"gdma",
4040
"phy",

esp-metadata/devices/esp32c3.toml

+2-1
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,8 @@ peripherals = [
4545

4646
symbols = [
4747
# Additional peripherals defined by us (the developers):
48-
"adc",
48+
"adc1",
49+
"adc2",
4950
"assist_debug_sp_monitor",
5051
"assist_debug_region_monitor",
5152
"gdma",

esp-metadata/devices/esp32c6.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -71,7 +71,7 @@ peripherals = [
7171

7272
symbols = [
7373
# Additional peripherals defined by us (the developers):
74-
"adc",
74+
"adc1",
7575
"assist_debug_sp_monitor",
7676
"assist_debug_region_monitor",
7777
"gdma",

esp-metadata/devices/esp32h2.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,7 @@ peripherals = [
6363

6464
symbols = [
6565
# Additional peripherals defined by us (the developers):
66-
"adc",
66+
"adc1",
6767
"assist_debug_sp_monitor",
6868
"assist_debug_region_monitor",
6969
"gdma",

esp-metadata/devices/esp32p4.toml

+2
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,8 @@ peripherals = [
9393

9494
symbols = [
9595
# Additional peripherals defined by us (the developers):
96+
# "adc1",
97+
# "adc2",
9698
"clic",
9799
"very_large_intr_status",
98100
"gpio_bank_1",

esp-metadata/devices/esp32s2.toml

+2-1
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,8 @@ peripherals = [
4949

5050
symbols = [
5151
# Additional peripherals defined by us (the developers):
52-
"adc",
52+
"adc1",
53+
"adc2",
5354
"dac",
5455
"pdma",
5556
"phy",

esp-metadata/devices/esp32s3.toml

+2-1
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,8 @@ peripherals = [
6161

6262
symbols = [
6363
# Additional peripherals defined by us (the developers):
64-
"adc",
64+
"adc1",
65+
"adc2",
6566
"assist_debug_region_monitor",
6667
"gdma",
6768
"phy",

qa-test/src/bin/embassy_adc.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
//! This shows how to asynchronously read ADC data
22
3-
//% CHIPS: esp32c6 esp32c3
3+
//% CHIPS: esp32c2 esp32c3 esp32c6 esp32h2
44

55
#![no_std]
66
#![no_main]

0 commit comments

Comments
 (0)