1
1
use core:: marker:: PhantomData ;
2
2
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
+ }
7
10
8
11
#[ cfg( not( esp32h2) ) ]
9
12
pub use self :: calibration:: * ;
@@ -12,17 +15,13 @@ use super::{AdcCalSource, AdcConfig, Attenuation};
12
15
use crate :: clock:: clocks_ll:: regi2c_write_mask;
13
16
#[ cfg( any( esp32c2, esp32c3, esp32c6) ) ]
14
17
use crate :: efuse:: Efuse ;
15
- #[ cfg( any( esp32c3, esp32c6) ) ]
16
18
use crate :: {
17
19
analog:: adc:: asynch:: AdcFuture ,
18
20
interrupt:: { InterruptConfigurable , InterruptHandler } ,
19
- peripherals:: Interrupt ,
20
- Async ,
21
- } ;
22
- use crate :: {
23
21
peripheral:: PeripheralRef ,
24
- peripherals:: APB_SARADC ,
22
+ peripherals:: { Interrupt , APB_SARADC } ,
25
23
system:: { GenericPeripheralGuard , Peripheral } ,
24
+ Async ,
26
25
Blocking ,
27
26
} ;
28
27
@@ -51,7 +50,7 @@ fn regi2c_write_mask(block: u8, host_id: u8, reg_add: u8, msb: u8, lsb: u8, data
51
50
// https://github.com/espressif/esp-idf/blob/903af13e8/components/soc/esp32h2/include/soc/regi2c_saradc.h
52
51
// https://github.com/espressif/esp-idf/blob/903af13e8/components/soc/esp32h4/include/soc/regi2c_saradc.h
53
52
cfg_if:: cfg_if! {
54
- if #[ cfg( any ( esp32c2 , esp32c3 , esp32c6 , esp32h2 ) ) ] {
53
+ if #[ cfg( adc1 ) ] {
55
54
const I2C_SAR_ADC : u8 = 0x69 ;
56
55
const I2C_SAR_ADC_HOSTID : u8 = 0 ;
57
56
@@ -82,7 +81,7 @@ cfg_if::cfg_if! {
82
81
}
83
82
84
83
cfg_if:: cfg_if! {
85
- if #[ cfg( esp32c3 ) ] {
84
+ if #[ cfg( adc2 ) ] {
86
85
const ADC_SAR2_ENCAL_GND_ADDR : u8 = 0x7 ;
87
86
const ADC_SAR2_ENCAL_GND_ADDR_MSB : u8 = 7 ;
88
87
const ADC_SAR2_ENCAL_GND_ADDR_LSB : u8 = 7 ;
@@ -194,6 +193,7 @@ pub trait RegisterAccess {
194
193
fn set_init_code ( data : u16 ) ;
195
194
}
196
195
196
+ #[ cfg( adc1) ]
197
197
impl RegisterAccess for crate :: peripherals:: ADC1 {
198
198
fn config_onetime_sample ( channel : u8 , attenuation : u8 ) {
199
199
APB_SARADC :: regs ( ) . onetime_sample ( ) . modify ( |_, w| unsafe {
@@ -256,6 +256,7 @@ impl RegisterAccess for crate::peripherals::ADC1 {
256
256
}
257
257
}
258
258
259
+ #[ cfg( adc1) ]
259
260
impl super :: CalibrationAccess for crate :: peripherals:: ADC1 {
260
261
const ADC_CAL_CNT_MAX : u16 = ADC_CAL_CNT_MAX ;
261
262
const ADC_CAL_CHANNEL : u16 = ADC_CAL_CHANNEL ;
@@ -296,7 +297,7 @@ impl super::CalibrationAccess for crate::peripherals::ADC1 {
296
297
}
297
298
}
298
299
299
- #[ cfg( esp32c3 ) ]
300
+ #[ cfg( adc2 ) ]
300
301
impl RegisterAccess for crate :: peripherals:: ADC2 {
301
302
fn config_onetime_sample ( channel : u8 , attenuation : u8 ) {
302
303
APB_SARADC :: regs ( ) . onetime_sample ( ) . modify ( |_, w| unsafe {
@@ -357,7 +358,7 @@ impl RegisterAccess for crate::peripherals::ADC2 {
357
358
}
358
359
}
359
360
360
- #[ cfg( esp32c3 ) ]
361
+ #[ cfg( adc2 ) ]
361
362
impl super :: CalibrationAccess for crate :: peripherals:: ADC2 {
362
363
const ADC_CAL_CNT_MAX : u16 = ADC_CAL_CNT_MAX ;
363
364
const ADC_CAL_CHANNEL : u16 = ADC_CAL_CHANNEL ;
@@ -435,9 +436,9 @@ where
435
436
}
436
437
}
437
438
438
- #[ cfg( any( esp32c3, esp32c6) ) ]
439
439
/// Reconfigures the ADC driver to operate in asynchronous mode.
440
440
pub fn into_async ( mut self ) -> Adc < ' d , ADCI , Async > {
441
+ asynch:: acquire_async_adc ( ) ;
441
442
self . set_interrupt_handler ( asynch:: adc_interrupt_handler) ;
442
443
443
444
// Reset interrupt flags and disable oneshot reading to normalize state before
@@ -531,7 +532,6 @@ where
531
532
532
533
impl < ADCI > crate :: private:: Sealed for Adc < ' _ , ADCI , Blocking > { }
533
534
534
- #[ cfg( any( esp32c3, esp32c6) ) ]
535
535
impl < ADCI > InterruptConfigurable for Adc < ' _ , ADCI , Blocking > {
536
536
fn set_interrupt_handler ( & mut self , handler : InterruptHandler ) {
537
537
for core in crate :: Cpu :: other ( ) {
@@ -545,7 +545,7 @@ impl<ADCI> InterruptConfigurable for Adc<'_, ADCI, Blocking> {
545
545
}
546
546
}
547
547
548
- #[ cfg( any ( esp32c2 , esp32c3 , esp32c6 ) ) ]
548
+ #[ cfg( all ( adc1 , not ( esp32h2 ) ) ) ]
549
549
impl super :: AdcCalEfuse for crate :: peripherals:: ADC1 {
550
550
fn init_code ( atten : Attenuation ) -> Option < u16 > {
551
551
Efuse :: rtc_calib_init_code ( 1 , atten)
@@ -560,7 +560,7 @@ impl super::AdcCalEfuse for crate::peripherals::ADC1 {
560
560
}
561
561
}
562
562
563
- #[ cfg( esp32c3 ) ]
563
+ #[ cfg( adc2 ) ]
564
564
impl super :: AdcCalEfuse for crate :: peripherals:: ADC2 {
565
565
fn init_code ( atten : Attenuation ) -> Option < u16 > {
566
566
Efuse :: rtc_calib_init_code ( 2 , atten)
@@ -635,14 +635,18 @@ mod adc_implementation {
635
635
}
636
636
}
637
637
638
- #[ cfg( any( esp32c3, esp32c6) ) ]
639
638
impl < ' d , ADCI > Adc < ' d , ADCI , Async >
640
639
where
641
640
ADCI : RegisterAccess + ' d ,
642
641
{
643
642
/// Create a new instance in [crate::Blocking] mode.
644
643
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
+ }
646
650
Adc {
647
651
_adc : self . _adc ,
648
652
attenuations : self . attenuations ,
@@ -697,7 +701,6 @@ where
697
701
}
698
702
}
699
703
700
- #[ cfg( any( esp32c3, esp32c6) ) ]
701
704
/// Async functionality
702
705
pub ( crate ) mod asynch {
703
706
use core:: {
@@ -706,20 +709,42 @@ pub(crate) mod asynch {
706
709
task:: { Context , Poll } ,
707
710
} ;
708
711
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 } ;
709
715
use procmacros:: handler;
710
716
711
717
use crate :: { asynch:: AtomicWaker , peripherals:: APB_SARADC , Async } ;
712
718
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
+
713
737
#[ handler]
714
738
pub ( crate ) fn adc_interrupt_handler ( ) {
715
739
let saradc = APB_SARADC :: regs ( ) ;
716
740
let interrupt_status = saradc. int_st ( ) . read ( ) ;
717
741
742
+ #[ cfg( adc1) ]
718
743
if interrupt_status. adc1_done ( ) . bit_is_set ( ) {
719
744
handle_async ( crate :: peripherals:: ADC1 )
720
745
}
721
746
722
- #[ cfg( esp32c3 ) ]
747
+ #[ cfg( adc2 ) ]
723
748
if interrupt_status. adc2_done ( ) . bit_is_set ( ) {
724
749
handle_async ( crate :: peripherals:: ADC2 )
725
750
}
@@ -745,6 +770,7 @@ pub(crate) mod asynch {
745
770
fn waker ( ) -> & ' static AtomicWaker ;
746
771
}
747
772
773
+ #[ cfg( adc1) ]
748
774
impl AsyncAccess for crate :: peripherals:: ADC1 {
749
775
fn enable_interrupt ( ) {
750
776
APB_SARADC :: regs ( )
@@ -771,7 +797,7 @@ pub(crate) mod asynch {
771
797
}
772
798
}
773
799
774
- #[ cfg( esp32c3 ) ]
800
+ #[ cfg( adc2 ) ]
775
801
impl AsyncAccess for crate :: peripherals:: ADC2 {
776
802
fn enable_interrupt ( ) {
777
803
APB_SARADC :: regs ( )
0 commit comments