|
EDA365欢迎您登录!
您需要 登录 才可以下载或查看,没有帐号?注册
x
一、基础知识" k+ v/ \7 L+ J' C% w. c
1. 18个通道,可测16个外部和2个内部信号源,可设置成单侧、连续、扫描、间断模式执行. [* r# n9 x6 L( c
2. 12位精度$ v, ?0 w% ^; @* c- @- `
3. 扫描模式,通道0到通道n的自动转化
! R9 x- X% v- L0 ~5 N4. 自校准+ P( T+ B5 Z# ]
5. 按通道配置采样时间4 Z. O' E0 [: y+ F& m
6. 间断模式?
6 \) k0 V1 n: _$ b1 i& i2 G+ l7. 双ADC模式?8 {: o+ {7 B% }3 b5 r: y
8. 供电要求2.4~3.6V% j$ h, o$ Y) G, B% R: {& Y/ A9 v8 K
9. 输入范围0~3.6V
+ m! R( p% ^5 u7 g二、功能描述7 K3 W+ r! x( S r4 n0 j! b- ?
1. 通道选择:2 B4 m9 l: n# w$ j
STM32的每个ADC模块通过内部的模拟多路开关,可以切换到不同的输入通道并进行转换。在任意多个通道上以任意顺序进行的一系列转换构成成组转换。例如,可以如下顺序完成转换:通道3、通道8、通道2、通道2、通道0、通道2、通道2、通道15。 R {2 a1 f. [- o* N
有2种划分转换组的方式:规则通道组和注入通道组。通常规则通道组中可以安排最多16个通道,而注入通道组可以安排最多4个通道。
' c1 t2 n% y8 F& v 在执行规则通道组扫描转换时,如有例外处理则可启用注入通道组的转换。规则转换和注入转换均有外部触发选项,规则通道转换期间有DMA请求产生,而注入转换则无DMA请求,需要用查询或中断的方式保存转换的数据。) K0 `. c# ?9 U M. [
一个不太恰当的比喻是:规则通道组的转换好比是程序的正常执行,而注入通道组的转换则好比是程序正常执行之外的一个中断处理程序。 规则组由多达16个转换组成。规则通道和它们的转换顺序在ADC_SQRx寄存器中选择。规则组中转换的总数写入ADC_SQR1寄存器的L[3:0]位中。: W; u9 i0 T2 y* D: N# c- L5 O
注入组由多达4个转换组成。注入通道和它们的转换顺序在ADC_JSQR寄存器中选择。注入组里的转换总数目写入ADC_JSQR寄存器的L[1:0]位中。2 u7 ^3 {% o @
如果规则转换已经在运行,为了在注入转换后确保同步,所有的ADC(主和从)的规则转换被停止,并在注入转换结束时同步恢复 H$ _# ?; U% D
4 B5 q. [* A1 f9 e/ a) ~! |+ f3 [
2. 单次转换模式:
: [1 y) y( K' ~7 f+ O 单次转换模式里,ADC 只执行一次转换。
2 g. j$ j8 ]9 U* H+ S' @" q3 [0 k/ _" ^3. 连续转换模式:7 @4 f2 u# a7 l/ P7 u% Z) q
在连续转换模式中,当前面ADC 转换一结束马上就启动另一次转换。
! P4 X8 [4 h' _5 P2 z4. 扫描模式:" L( _$ p* B, W( s
此模式用来扫描一组模拟通道。
3 L0 d9 `& s: [! o# O5. 注入模式管理:6 w `! [9 {+ H! g
(1)触发注入。详见参考手册
o5 h6 B& K4 e" g% F& z(2)自动注入。如果设置了 JAUTO 位,在规则组通道之后,注入组通道被自动转换。这可以用
9 V: ^% D* N. e! Q. a+ `: \" w 来转换在ADC_SQRx 和ADC_JSQR 寄存器中设置的多至20 个转换序列。' e6 N* S2 l# n- Q
在此模式里,必须禁止注入通道的外部触发。. t5 ?. B$ d0 |! p
6. 间断模式:* Z4 k7 ^: e# U9 D5 Q/ g/ Q
(1)规则组
3 ]6 b2 D8 E$ V( d# {) @7 C 此模式通过设置 ADC_CR1 寄存器上的DISCEN 位激活。它可以用来执行一个短序列的n 次转换(n<=8),此转换是ADC_SQRx 寄存器所选择的转换序列的一部分。N 由ADC_CR1 寄存器的DISCNUM[2:0]位给出。一个外部触发信号可以启动ADC_SQRx 寄存器中描述的下一轮 n 次转换,直到此序列所有的转换完成为止。总的序列长度由ADC_SQR1 寄存器的L[3:0]定义。
" E6 `, w* g# q. P- ?6 p 举例:
7 \$ o, Z( ^" k9 x n=3,被转换的通道 = 0, 1, 2, 3, 6, 7, 9, 108 G! V2 N( r* t6 l1 w1 }! w
第一次触发:转换的序列为 0,1,2+ X4 x$ x8 N4 U- F1 [$ |. r) r
第二次触发:转换的序列为 3,6,7) r4 F5 o# D4 Y! Q- ~1 m7 M% f
第三次触发:转换的序列为 9,10,并产生EOC 事件, `) t# u8 j/ Q1 d0 B8 S
第四次触发:转换的序列 0,1,2
8 K# ]" ~/ Q5 N# ~* F+ t 注意: 当一规则组以间断模式转换时,转换序列结束后不自动从头开始。当所有子组被转换完成,下一次触发启动第一个子组的转换。在上面的例子中,第四次触 发重新转换第一子组的通道 0,1 和2。
* t( C' ^7 U: Y# B6 a: p8 ^(2)注入组
4 w" @0 R4 x5 Z3 b6 J' F9 K 此模式通过设置 ADC_CR1 寄存器的JDISCEN 位激活。在一个外部触发事件后,给模式按序转换ADC_JSQR 寄存器中选择的序列。
: g- Q5 F1 Y! i7 K; r0 ?一个外部触发信号可以启动ADC_JSQR 寄存器选择的下一个通道序列的转换,直到序列中所有的转换完成为止。总的序列长度ADC_JSQR 寄存器的JL[1:0]位定义。
" y) |% K. m) U( O# Y 例子:
# l0 ^ h) |6 P6 L2 B3 O# @ y0 Rn=1,被转换的通道 = 1,2,3# f6 x( ?% u/ L* H+ ]
第一次触发:通道1 被转换: Z* J' {' J1 V, j: l4 h0 i
第二次触发:通道2 被转换. y0 h* I4 g- `" V- u! d
第三次触发:通道3 被转换,并且产生EOC 和JEOC 事件! W7 X) e; b# O( u
第四次触发:通道1 被转换
5 I5 A) q/ ~) b; i; U* u注意:1 当完成所有注入通道转换,下个触发启动第1 个注入通道的转换。在上述例子中,第四个触发重新转换第1 个注入通道1。9 k7 a7 [) x4 {; f
2 不能同时使用自动注入和间断模式。3 必须避免同时为规则和注入组设置间断模式。间断模式只能作用于一组转换。
8 }% i% E) S5 a o4 `+ }5 [7. 双ADC模式9 H; E; F, _0 C4 P
8. 数据对齐4 G: k/ E9 f' ~) y. s6 ]9 B1 j) \
ADC_CR2 寄存器中的ALIGN位选择转换后数据储存的对齐方式。数据可以左对齐或右对齐,如图146 和图147 所示。$ {0 O/ h$ z$ t. [9 v$ o' u+ K. X7 B
注入组通道转换的数据值已经减去了在ADC_JOFRx 寄存器中定义的偏移量,因此结果可以是一个负值。SEXT 位是扩展的符号值。
1 h+ R' e& b$ W* Z% o* r对于规则组通道,不需减去偏移值,因此只有12 个位有效。
~' ~# w) C5 J/ Y2 Y) F数据右对齐:
6 h+ ?1 Q5 W J7 }注入组; j8 A! [1 F; U8 F5 N
SEXT SEXT SEXT SEXT D11 D10 D9 D8 D7 D6 D5 D4 D3 D2 D1 D0% v: j6 r! z* A0 U
规则组
% b9 z+ A& [) L: u. d4 B0 0 0 0 D11 D10 D9 D8 D7 D6 D5 D4 D3 D2 D1 D0
* F6 F7 c# q* R1 h3 W数据左对齐:
0 |" H. f% g5 x2 K) E" ^注入组. | _2 u- `3 A$ w- B3 L; h
SEXT D11 D10 D9 D8 D7 D6 D5 D4 D3 D2 D1 D0 0 0 08 a: V/ Z% r: k9 r/ F8 n2 T
规则组$ U2 O+ g. n: h0 T! B+ R
D11 D10 D9 D8 D7 D6 D5 D4 D3 D2 D1 D0 0 0 0 0% h8 I) N- ]( a8 P
三、 程序设计与软件配置
1 y. U! T7 x2 l0 z1./ p- n& M7 I. w8 R0 }
void ADC_Config(void) B+ o" g: G, b2 N$ y) z5 x
{
: ?$ j& }5 `- `) Y0 N* o* Y2 ? ADC_InitTypeDef ADC_InitStructure;//定义ADC初始化结构体变量
+ U7 A3 o4 U% x$ j: p# y2 J: _: r ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//ADC1和ADC2工作在独立模式
" _$ X! X* Q& o E6 y# m% ] a ADC_InitStructure.ADC_ScanConvMode = ENABLE; //使能扫描
; J: {# ?; ~+ c& l0 p ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;//ADC转换工作在连续模式
7 g! ?; v0 V7 s2 ~6 [ ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//由软件控制转换
, @3 G x% m# p. \% ^% ?: i ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//转换数据右对齐
1 L1 v' l l8 v1 q. X- I- u! n" s ADC_InitStructure.ADC_NbrOfChannel = 1;//转换通道数目为1
5 A5 O! z9 W: x ADC_Init(ADC1, &ADC_InitStructure); //初始化ADC
- T7 t0 z/ B/ ]0 A ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 1, ADC_SampleTime_28Cycles5);
1 z/ R# j t" _$ M7 c //ADC1选择通道14,规则组采样顺序(1~16),采样时间239.5个周期! s% \+ w7 d( ?* x0 l& s
ADC_DMACmd(ADC1, ENABLE);//使能ADC1模块DMA9 K! O& E" |3 P: l. R
ADC_Cmd(ADC1, ENABLE);//使能ADC1$ b- x; L0 F$ W9 D! }
ADC_ResetCalibration(ADC1); //重置ADC1校准寄存器
4 X- ~$ @; ^6 U8 T# P; R while(ADC_GetResetCalibrationStatus(ADC1));//等待ADC1校准重置完成) O) }; P6 R* K& [0 J: v+ u
ADC_StartCalibration(ADC1);//开始ADC1校准
$ b ^' m- }" s while(ADC_GetCalibrationStatus(ADC1));//等待ADC1校准完成" y. b+ _0 F* S, L ]
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能ADC1软件开始转换0 [0 Q. v0 Z: A7 i
} |
|