本期我们介绍使用ADC的多通道采样。 Y4 N4 n5 g) [4 {! Y
- O% c# R3 X# ^% i: [4 h, _
# x* I* P& G2 s! c. q$ s
" x7 Q$ M, @6 h! F1 D. Q3 V首先我们需要明白一个ADC外设通常有许多个通道,合理的使用这些通道可以实现多个采样点的采样和利用多通道弥补采样率不足的问题。
7 Y, e6 z! Z' G( F H u1 l7 m {% b& ~1 m) @- k! x! D
首先我们需要知道,我们并不能直接获取某个通道的值。1 _; e* i! d& T$ f" N
6 h- `# a. E P, h, _/ D% T! |; l
' }6 @+ _: l/ O$ ?/ A5 _8 J
' |6 U5 |) @/ w& E. U0 ~- t7 p
我们HAL库中的函数的函数仅有一个参数需要传入,因此我们不能直接的获取ADC的值,这时候我们就需要明白我们的ADC的转化规则。# w! @; r' X2 t5 X" _6 f5 z8 n
* l6 H2 k% g( L2 ]9 q" o7 ~
) W* [* W9 h$ F& p* E% W1 O/ _0 a' I, x' |+ X& a) b
在CubeMX中我们开启两个(或多个)ADC通道& V' m+ X9 y) P W C: O
* V. P/ V$ f+ k6 ]' u
. h( b% j8 a) ]7 O6 I4 h% C
{9 e9 F( P3 p1 z/ G0 _: ]开启扫描模式和离散转换模式,重点是这个离散转换模式。, b* z7 T% U0 I$ M9 `! o& s; }
$ @: b6 P2 E! J4 A
允许在单个触发事件中执行多个ADC转换,而不是在每个触发事件中执行一次转换。这对于需要连续获取多个通道的数据时非常有用,因为可以在单个触发事件中获取多个通道的数据,而不需要多次触发。
_6 E/ Y& _2 u( Z0 ^& l: G0 o K8 s4 |# A3 a
这里的触发事件我们可以选择软件触发/ Z8 G B) d7 A* C8 \* Z$ R
' f1 l j+ `: j% o: u+ f3 o" U2 J
3 b' |7 `1 W+ Z: W4 M2 L ?
' j" d- {4 J0 d! t' C这样子我们调用HAL_ADC_Start的时候就会开启ADC转化。5 l9 V- C& y4 A+ [0 e( R- E
( ~' R# r8 k7 u4 L
. z ?1 K/ Z5 m6 N8 \" K9 \
. R2 b: Q) M. `) g+ v; r* p
在下面的规则中规定我们的转换规则,这里我们是先测量Channel13再测量Channel11,我们调用HAL_ADC_GetValue的时候就会按照13->11->13->11.....的顺序获取我们的ADC的不同通道的值。6 N) ^1 k' A: t+ W
5 B! [/ g C/ c$ R z# z所以假如我们开启了4个通道,那么我们的多通道采样代码即可这样子写:
1 {* N0 F* ~& f$ D0 z- uint32_t adc_values[4]; // 存储4个通道的ADC值4 \4 I/ A R* u) e- |5 u
- 2 @& {; k. o9 W. B) t* p/ @" H9 h
- while (1)
" A( j( u8 i. X/ Q - {
! m; g; t7 a; d* u9 G! M; j - // 开始ADC转换
2 C* D: S' e5 w( q! b - HAL_ADC_Start(&hadc1);
+ d: N# J1 u& r% g -
0 Z( B2 c) E$ k+ l3 ^5 [& j - // 依次转换4个通道
F6 Q& P' W- a8 T+ m% T( z7 R - for (int i = 0; i < 4; i++). D- r8 d0 X" K* i" V3 R
- {' w' w+ q1 E; V6 r" T9 Q* v9 x5 Y
- // 等待转换完成* z( Z, k2 m- D: I$ A
- HAL_ADC_PollForConversion(&hadc1, HAL_MAX_DELAY);, l+ i( O/ ]8 R3 R8 m
- // 获取ADC值
8 G& {6 N3 d) i# w - adc_values[i] = HAL_ADC_GetValue(&hadc1);* j4 E$ P# c% y5 |
- }
. V1 v* _& q" g8 }; t" P) A- n - 8 D& K0 A6 _; X( e4 J! J1 m2 m
- 8 o, y: n8 L h# O! e
- HAL_ADC_Start(&hadc1);//再次开启采样7 t; a& j9 Q7 k5 R3 _( H
- & m1 X ^5 }) n- y* Y" J4 J8 B3 `& X
- // 在这里处理ADC值
/ r: } x5 x' c8 i - // adc_values数组包含了4个通道的ADC值" s* l3 k" R0 z* m% m" w
- }: O4 W' I2 b1 k' y" [5 U
- }
复制代码 4 i4 i0 `1 {5 U1 C6 K. w! R! u
这样子我们就可以获取四个通道的ADC采样的值了。
: G! s* Q* U/ W* d, V& W
& e0 A$ l4 a" y: x+ M# P2 wDMA
3 M5 U+ d, p d' z' x0 g同样,如果当我们配置了多通道DMA的时候,并且设置定时器实现采样率设置。# ~3 _$ V# W5 b
. M6 Q7 C9 Q0 W* d# z: i; S2 C
我们需要明白的是,DMA只是帮助我们为ADC的数据提供存储区,这样子就不需要利用MPU进行读取数据存放数据的操作了。3 z( S) R/ `, h2 C& {
/ S9 ~/ V6 b/ k% E
因此配置DMA并不影响ADC的采样规则,只是存入DMA缓存区的顺序发生了变化。
% ~$ I+ G7 |1 s0 g- for(int i = 0;i<Lenth/2;i++)
- Q9 M) R' q% L2 {- z$ b - {
/ T. c& e; U( M) `; \* k% N - ADC_V1[i] = ADC_Value[i*2];
0 ~6 S- R6 j1 x, z/ p7 G - ADC_V2[i] = ADC_Value[i*2+1]/ E- ]7 k/ \- m- h7 _4 @
-
9 u; _" k7 M% e3 {& R; c7 l' b - }
复制代码 / H3 Y2 @/ S* n( q5 Q3 q% R( w
因此了解DMA的本质与多通道ADC采样的规则就可以实现多通道的ADC连续采样啦(不过这样子采样率会减半)。
; A6 o1 |) o# c% d& n9 z
1 N: Q0 L G5 K- p' H# b( {+ L$ D+ s: U1 p8 r7 b
转载自:电路小白
" g2 W+ P% e7 E$ \8 ^0 D" t; u. v如有侵权请联系删除
7 N9 W2 P( ]9 S1 [
7 C8 a. ?5 z7 H2 j1 V. H: P
0 [" S k$ R% [9 ~ d' ~% D- a! y7 g; n# G' r- [) f- ?* {- [. E) L, n
|
stm32F104测试,四个通道数值都一样,感觉不太对,而且接3V3后,只有最后一个通道数值有变化
![image.png](data/attachment/forum/202407/12/142159m8ssww2jk1269m3j.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/300 "image.png")
[md]没读取一个通道都要开启一次ADC,把他也放到for循环里面试试