前言
STM32F30x 系列的 12 位 SAR ADC 有很多鮮明的特色性能,比如采樣率可以達到 5MSPS,可支持差分輸入,等等。但是,由于設計的不同,在使用上也有不少不太一樣的地方,我們在使用 STM32F30x 的 ADC 外設的時(shí)候,還是要仔細了解一些使用的細節。
問(wèn)題
某客戶(hù)在其產(chǎn)品的設計中,使用了 STM32F302CCT6?蛻(hù)在使用過(guò)程發(fā)現 ADC 在工作情況下會(huì )有各種各樣奇奇怪怪的問(wèn)題。
調研
1.了解問(wèn)題
客戶(hù)在開(kāi)發(fā)中使用了 STM32F30x 的標準外設庫 STM32F30x_DSP_StdPeriph_Lib_V1.2.3,在其程序設計中,ADC1 是在使用的時(shí)候才打開(kāi)的,在不使用的時(shí)候將 ADC1關(guān)閉。通過(guò)調用 void ADC_DisableCmd(ADC_TypeDef* ADCx)子程序,執行ADC_DisableCmd(ADC1)將 ADC1 關(guān)閉。仔細察看程序,發(fā)現程序中在 ADC1 的“打開(kāi)→關(guān)閉→打開(kāi)→關(guān)閉→…”循環(huán)的關(guān)閉中,執行了兩次ADC_DisableCmd(ADC1)。
2.問(wèn)題分析
通過(guò)學(xué)習 STM32F30x 的參考手冊,可以知道 STM32F30x 在對 ADC 進(jìn)行關(guān)閉的操作與其他系列是不一樣的;其他系列只要將 ADON 位清零就可以停止轉換并使 ADC 進(jìn)入掉電模式,而 STM32F30x 則不一樣,它是通過(guò)置位 ADDIS 位來(lái)關(guān)閉 ADC的。在 void ADC_DisableCmd(ADC_TypeDef* ADCx)程序中也可以看到這一點(diǎn)。
但是,問(wèn)題來(lái)了,ADDIS 位是在什么情況下都可以置位的嗎?
我們來(lái)看一下參考手冊中對關(guān)閉 ADC 的軟件流程的描述:
從描述中,我們可以知道:在置位 ADDIS 之前,必須先檢測 ADSTART 位和 JADSTART 位,確保他們?yōu)榱,也就是說(shuō)沒(méi)有正在進(jìn)行的 A/D 轉換。在 ADC Control register - ADCx_CR 寄存器中對 ADDIS 的描述也注明了:
關(guān)于這一點(diǎn),大多數人都是會(huì )注意到的。但是,注意這個(gè)就夠了嗎?關(guān)于 ADC 的控制位,在參考手冊特別使用一個(gè)小節對向控制位寫(xiě)訪(fǎng)問(wèn)的限制進(jìn)行詳細描述,此小節為“Contraints when writing the ADC control bits”,在這一小節中,有一句話(huà)值得注意:
注意這里的用詞 only if,它的意思是“只有在 ADC 是打開(kāi)狀態(tài),而且沒(méi)有正在等待的關(guān)閉 ADC 的請求的情況下,也就是在A(yíng)DEN=1 且 ADDIS=0 的情況下,才允許軟件對 ADCx_CR 寄存器中的 ADSTART,JADSTART 和 ADDIS 位進(jìn)行操作。在底下的“Note”注意中寫(xiě)道:
這個(gè)注意說(shuō),這些禁止的 ADC 寫(xiě)訪(fǎng)問(wèn)行為是沒(méi)有硬件保護去禁止的,錯誤的操作行為將導致 ADC 進(jìn)入一個(gè)未知的狀態(tài)。要恢復這種狀態(tài),必須將 ADC 徹底關(guān)閉(將 ADCx_CR 中的所有位全清零)。
所以,現在可以知道,ADC1 工作不正常的原因正是因為連續執行了兩次 ADC_DisableCmd(ADC1)。第一次執行ADC_DisableCmd(ADC1)時(shí),當 ADC1 已經(jīng)有效關(guān)閉時(shí),ADEN 和 ADDIS 都被硬件清零,這個(gè)時(shí)候第二次再去寫(xiě) ADDIS 位就是個(gè)錯誤的行為了,將會(huì )導致 ADC1 進(jìn)入未知狀態(tài)。
3. 問(wèn)題解決
在解決問(wèn)題之前,先來(lái)看一下 STM32Cube_FW_F3_V1.2.0 庫中對 ADC 進(jìn)行關(guān)閉的操作。打開(kāi) stm32f3xx_hal_adc_ex.c 文件,找到 static HAL_StatusTypeDef ADC_Disable(ADC_HandleTypeDef* hadc)函數,其程序內容為:
在這個(gè)函數注釋中有個(gè)“Note: forbidden to disable ADC (set bit ADC_CR_ADDIS) if ADC is already disabled.”,再次告訴我們“禁止在 ADC 已經(jīng)被關(guān)閉的情況下再次關(guān)閉 ADC”。然后,程序在運行中先對 ADC 是否已經(jīng)被關(guān)閉進(jìn)行了判斷,如果已經(jīng)被關(guān)閉,則不進(jìn)行關(guān)閉操作;未被關(guān)閉情況下才會(huì )執行關(guān)閉操作。
所以,在使用標準外設庫的時(shí)候,我們也可以參考 Cube 庫中的這種操作來(lái)進(jìn)行改善。
可以考慮對 void ADC_DisableCmd(ADC_TypeDef* ADCx)程序進(jìn)行修改:
當然,不動(dòng) void ADC_DisableCmd(ADC_TypeDef* ADCx)程序也可以,只需在用戶(hù)程序中對這些限制進(jìn)行判斷即可。上面的while 循環(huán)中,也可以加入超時(shí)退出機制。
結論
連續兩次對 ADDIS 控制位進(jìn)行寫(xiě) 1,是錯誤的操作行為,將會(huì )導致 ADC 進(jìn)入未知狀態(tài),工作不正常。
處理
修改程序,避免錯誤的操作行為。
建議
在使用 STM32F30x 系列的 ADC 外設時(shí),必須要對控制位的操作限制有明確的了解。
重要通知 - 請仔細閱讀
意法半導體公司及其子公司(“ST”)保留隨時(shí)對ST 產(chǎn)品和/ 或本文檔進(jìn)行變更、更正、增強、修改和改進(jìn)的權利,恕不另行通知。買(mǎi)方訂貨之前應獲取關(guān)于ST 產(chǎn)品的最新信息。ST 產(chǎn)品的銷(xiāo)售依照訂單確認時(shí)的相關(guān)ST 銷(xiāo)售條款。 買(mǎi)方自行負責對ST 產(chǎn)品的選擇和使用, ST 概不承擔與應用協(xié)助或買(mǎi)方產(chǎn)品設計相關(guān)的任何責任。
ST 不對任何知識產(chǎn)權進(jìn)行任何明示或默示的授權或許可。
轉售的ST 產(chǎn)品如有不同于此處提供的信息的規定,將導致ST 針對該產(chǎn)品授予的任何保證失效。
ST 和ST 徽標是ST 的商標。所有其他產(chǎn)品或服務(wù)名稱(chēng)均為其各自所有者的財產(chǎn)。
本文檔中的信息取代本文檔所有早期版本中提供的信息。