你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

【实战经验】STM32F30x禁止ADC已关闭情况下再次关闭ADC

[复制链接]
zero99 发布时间:2017-4-5 17:26
STM32F30x 禁止ADC 已关闭情况下再次关闭ADC
' t  k9 h+ M" m" a+ t9 Q3 I
5 q* U! p% y: n5 i
前言4 `4 ?; p$ z3 R9 |3 W
STM32F30x 系列的12 位SAR ADC 有很多鲜明的特色性能,比如采样率可以达到5 MSPS,可支持差分输入,等等。但是,由于设计的不同,在使用上也有不少不太一样的地方,我们在使用STM32F30x 的ADC 外设的时候,还是要仔细了解一些使用的细节。
! y6 x: W+ E3 ?' g. F. z4 G
$ t! @# F; B- J7 j
) J$ w$ |+ f+ q( g问题
  y! A  x! w, I9 U& m某客户在其产品的设计中,使用了STM32F302CCT6。客户在使用过程发现ADC 在工作情况下会有各种各样奇奇怪怪的问题。
. Y# B7 b8 y1 A; ^0 I
: t, }( Y$ J# {7 t0 t& X" w+ U; @, k2 u) ^/ U2 Y5 s8 M3 I
调研
( P/ G* x4 w0 T6 z1 n: \1.了解问题4 G* @  ?" k4 Z( \. o+ G7 A9 P
客户在开发中使用了STM32F30x 的标准外设库STM32F30x_DSP_StdPeriph_Lib_V1.2.3,在其程序设计中,ADC1 是在使用的时候才打开的,在不使用的时候将ADC1 关闭。通过调用void ADC_DisableCmd(ADC_TypeDef* ADCx)子程序,执行ADC_DisableCmd(ADC1)将ADC1 关闭。- V! o6 {$ o# P) h( K
. ]' v8 m& i* D, B9 \9 V. x- Y

+ Q+ c6 A% I; t3 F5 q$ V仔细察看程序,发现程序中在ADC1 的“打开→关闭→打开→关闭→…”循环的关闭中,执行了两次ADC_DisableCmd(ADC1)。& Y: h' T6 j2 r* L  k8 H
! ?: j& u/ K7 Z

0 ~! K% H9 ]/ E3 F9 {3 Q- ?2.问题分析
0 m$ M3 h) @  O' ]通过学习STM32F30x 的参考手册,可以知道STM32F30x 在对ADC 进行关闭的操作与其他系列是不一样的;其他系列只要将ADON 位清零就可以停止转换并使ADC 进入掉电模式,而STM32F30x 则不一样,它是通过置位ADDIS 位来关闭ADC的。在void ADC_DisableCmd(ADC_TypeDef* ADCx)程序中也可以看到这一点。
3 S2 P* y5 V% |# S: p+ Q
21.jpg
2 t0 u% A: _% Q; `
" u$ T/ R; _0 H5 T" ^但是,问题来了,ADDIS 位是在什么情况下都可以置位的吗?0 a3 \5 D- P: i; C
我们来看一下参考手册中对关闭ADC 的软件流程的描述:
7 t; j6 b1 I6 e3 Z
22.png
( T9 m3 J, v; g' Z从描述中,我们可以知道:在置位ADDIS 之前,必须先检测ADSTART 位和JADSTART 位,确保他们为零,也就是说没有正在进行的A/D 转换。在ADC Control register - ADCx_CR 寄存器中对ADDIS 的描述也注明了:5 Z& v# A( X3 I- Q7 j
23.png
8 P7 D' x1 o1 ?/ H  w关于这一点,大多数人都是会注意到的。但是,注意这个就够了吗?关于ADC 的控制位,在参考手册特别使用一个小节对向控制位写访问的限制进行详细描述,此小节为“Contraints when writing the ADC control bits”,在这一小节中,有一句话值得注意:. l9 T8 w! s+ P1 d( `! S3 E. w: d% c
24.png ( [" F/ `" P$ V7 i
注意这里的用词only if,它的意思是“只有在ADC 是打开状态,而且没有正在等待的关闭ADC 的请求的情况下,也就是在ADEN=1 且ADDIS=0 的情况下,才允许软件对ADCx_CR 寄存器中的ADSTART,JADSTART 和ADDIS 位进行操作。在底下的“Note”注意中写道:9 A$ Q# m+ k0 O( T, ^3 X
25.png
' A+ q; l/ L" {0 y0 B这个注意说,这些禁止的ADC 写访问行为是没有硬件保护去禁止的,错误的操作行为将导致ADC 进入一个未知的状态。要恢复这种状态,必须将ADC 彻底关闭(将ADCx_CR 中的所有位全清零)。
, |) Q4 f8 b  G1 i; Y. B- u; D! ~$ i! e. o

# B. C8 i$ E# s  t8 i所以,现在可以知道,ADC1 工作不正常的原因正是因为连续执行了两次ADC_DisableCmd(ADC1)。第一次执行ADC_DisableCmd(ADC1)时,当ADC1 已经有效关闭时,ADEN 和ADDIS 都被硬件清零,这个时候第二次再去写ADDIS 位就是个错误的行为了,将会导致ADC1 进入未知状态。- Q- ~5 z, ^. q% W

# t- m1 U7 s0 B3 w2 }" E5 y0 {  p; a& h: e+ V8 s! G, Y7 A; m" C/ r6 I
3.问题解决
' V4 O$ S8 R" D' m& x在解决问题之前,先来看一下STM32Cube_FW_F3_V1.2.0 库中对ADC 进行关闭的操作。打开stm32f3xx_hal_adc_ex.c 文件,找到static HAL_StatusTypeDef ADC_Disable(ADC_HandleTypeDef* hadc)函数,其程序内容为:
4 J7 [/ N$ J  }+ p
26.png & m, O; s, Q/ h. ?/ g6 _! N& e6 d
在这个函数注释中有个“Note: forbidden to disable ADC (set bit ADC_CR_ADDIS) if ADC is already disabled.”,再次告诉我们“禁止在ADC 已经被关闭的情况下再次关闭ADC”。然后,程序在运行中先对ADC 是否已经被关闭进行了判断,如果已经被关闭,则不进行关闭操作;未被关闭情况下才会执行关闭操作。
1 T& X- @" R& D+ M3 M7 M
# f3 t* s/ I! f4 n所以,在使用标准外设库的时候,我们也可以参考Cube 库中的这种操作来进行改善。
0 @; ^, g- c+ T# W, |9 \9 b! J# ]1 o1 @可以考虑对void ADC_DisableCmd(ADC_TypeDef* ADCx)程序进行修改:. h2 t/ b% P3 ^; [7 s! Y
27.png ! H4 J; ^" x0 n/ l3 F
当然,不动void ADC_DisableCmd(ADC_TypeDef* ADCx)程序也可以,只需在用户程序中对这些限制进行判断即可。上面的
7 L" N% G( R( s# u, P. Swhile 循环中,也可以加入超时退出机制。
. o: f! ]' a4 ?; C9 ?" R9 k. @6 E% h
结论5 l5 u# H4 W& U* U
连续两次对ADDIS 控制位进行写1,是错误的操作行为,将会导致ADC 进入未知状态,工作不正常。2 J8 H% w# D8 H+ U, T% x: d

- K$ O( h5 w3 z+ \7 b处理6 q4 R- I6 X& x% R: m' c. Z- \; c
修改程序,避免错误的操作行为。
  ?8 `, v, F8 _3 A
2 f3 ~4 h2 J" C+ i6 l建议& u& ]! r; T. S7 w: X
在使用STM32F30x 系列的ADC 外设时,必须要对控制位的操作限制有明确的了解。在STM32F30x 的“快速ADC 模块”培训资料中,也有对此进行描述,如下:# v6 F; l1 ?# j' T# S$ V/ @' |
28.png
- B  x7 {3 \+ }. I最重要还是要看参考手册,多注意一些小细节。& @0 D% Q. r1 g  I; |9 M, b. n
& ]3 ?7 ~4 R9 T  {
2 x' W5 J8 o* V
4 F6 d7 O1 P# C6 L+ L

$ ]" H) U& p) e7 U. F  J+ m$ m5 a2 d3 W+ o7 n! F6 e! |' J
STM32F30x禁止ADC已关闭情况下再次关闭ADC.pdf (283.08 KB, 下载次数: 25)
收藏 评论2 发布时间:2017-4-5 17:26

举报

2个回答
霹雳之火 回答时间:2017-4-5 18:41:03
沙发 沙发 管理员也在学习吗 可以作为一个系列了
弱电电人 回答时间:2018-6-13 15:00:58
谢谢
关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版