很多STM32芯片里往往內置了專用的ADC通道,比方用來測量Vrefint,VBAT的分壓或溫度傳感器的輸出電壓信號。不同系列所內置的模擬信號通道可能有差異。這里以STM32G4系列為例,它內置了對應于Vrefint,VBAT的三分之一分壓和溫度傳感器的輸出電壓的專用模擬通道。
下面的示例就是針對上述3個通道進行AD,并測量相關電壓和片內溫度,最終得到3個結果,分別是VRefint電壓,VBAT的電壓,片內溫度。
實現過程是這樣的,大體分四步:【有點點麻雀雖小五臟俱全的味道】
1、TIMER1 更新事件觸發ADC的轉換;
2、CPU基于EOC中斷獲取ADC結果;
3、對ADC結果進行換算,得到電壓值和溫度值存放在特定內存位置;
其中,TIMER1的CH1輸出PWM波形,其更新事件做ADC的轉換啟動信號。每次的TIMER更新事件觸發ADC,3個通道掃描方式轉換。這里的UART使用片內LPUART,使用它主要是考慮它跟板載虛擬串口直接相連,沒有其它特別用意。
我使用STM32G474Nucleo板來進行下面實驗。其中VDD=3.3v,VBAT與VDD相連。另外,ADC模塊的參考電壓也是3.3v.
使用CubeMx圖形化工具進行配置,先看TIMER配置:
再看看ADC的基本配置:
LPUART的基本配置:
因為要使用ADC中斷和UART的DMA傳輸,記得做ADC的中斷響應使能配置和LPUART的DMA配置,這里只使用UART的TX DMA功能。
使用CubeMx主要配置主要是上面這些。
在組織用戶代碼前,先簡單介紹下片內溫度傳感器的內容。該溫度傳感器針對不同溫度有不同電壓輸出,其輸出電壓跟溫度呈線性關系。ST公司針對片內溫度傳感器在兩個特定溫度【30℃和110℃或30℃和130℃】、基于特定參考電壓【3v或3.3v,不同系列以數據手冊為準】生成了1組校準值并存放于片內特定FLASH位置。
STM32G4系列的校準值是在參考電壓為3v,30℃和110℃條件下的兩個值,在數據手冊里還給出了校準值的片內存放地址。
針對這個溫度傳感器的使用,ST公司在參考手冊里還給出了計算公式。其實,有無這個公式無所謂,我們不難自行推理出來。【TS_DATA代表某時刻測得的傳感器輸出電壓對應的轉換值,TS_CAL1/TS_CAL2分別表示在30℃和110℃條件下基于傳感器輸出電壓的轉換值。】
另外,前面提過,ST公司在手冊里給出了溫度傳感器的兩個溫度下的校準值,但要注意生成校準值的ADC模塊所用參考電壓跟我們實際應用時AD模塊所用的參考基準電壓可能不一致。如果不一致,就必須將ADC值換算成同一基準參考電壓條件下的數據。目前在ST手冊里也特別強調這點了。我把上面一副圖再貼一遍于此【見黃色語句提醒】。
關于這點,我們也不難理解。同一待測信號、同一ADC模塊在不同基準參考電壓下轉換值往往是不一樣的。見下面示意圖加以理解。
完成各項配置后,創建軟件工程。添加必需的用戶代碼:
#define TX_Timeout (9999) #define TS_CAL1_ADDR (0x1FFF75A8) //用于計算溫度傳感器數據 #define TS_CAL2_ADDR (0x1FFF75CA) //用于計算溫度傳感器數據 #define size1 (40) char WDVol[size1],BatVol[size1],InVol[size1]; uint16_t ts_c30,ts_c110; uint16_t ADCResult[3],convCNT; volatile uint32_t Completed,EndofCon_Flag; floatVBATVolt;//存放BBAT電壓最終結果 floatVRefint;//存放Vrefint電壓最終結果 floatTemperature;//存放片外溫度℃最終結果 int main(void) { /* USER CODE BEGIN 1 */ /* USER CODE END 1 */ /* MCU Configuration--------------------------------------------------------*/ /* Reset of all peripherals, Initializes the Flash interface and the Systick. */ HAL_Init(); /* USER CODE BEGIN Init */ /* USER CODE END Init */ /* Configure the system clock */ SystemClock_Config(); /* USER CODE BEGIN SysInit */ /* USER CODE END SysInit */ /* Initialize all configured peripherals */ MX_GPIO_Init(); MX_DMA_Init(); MX_ADC1_Init(); MX_LPUART1_UART_Init(); MX_TIM1_Init(); /* USER CODE BEGIN 2 */ ts_c30=*(uint16_t*)(TS_CAL1_ADDR);//讀取30℃時的ADC校準值 ts_c110 = *(uint16_t *)(TS_CAL2_ADDR);//讀取110℃時的ADC校準值 HAL_ADCEx_Calibration_Start(&hadc1 , ADC_SINGLE_ENDED);//ADC校準 HAL_ADC_Start_IT(&hadc1);//啟動ADC并開啟轉換中斷 HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_1); /* USER CODE END 2 */ /* Infinite loop */ /* USER CODE BEGIN WHILE */ while (1) { /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */ if (EndofCon_Flag!=0) { VBATVolt=(ADCResult[0]/4095.)* 3.3 * 3.; VRefint=(ADCResult[1]/4095.) * 3.3; Temperature = 30.+ (88.*(ADCResult[2]-((ts_c30/1.1))))/(ts_c110 - ts_c30); EndofCon_Flag=0; //HAL_UART_Transmit(&hlpuart1, (uint8_t *)WDVol ,sizeof(WDVol), TX_Timeout); HAL_GPIO_WritePin(GPIOC,GPIO_PIN_3,GPIO_PIN_RESET);//forauxiliarytest HAL_UART_Transmit_DMA(&hlpuart1, (uint8_t *)WDVol ,sizeof(WDVol)); while(Completed==0){} Completed =0; //HAL_UART_Transmit(&hlpuart1, (uint8_t *)InVol ,sizeof(InVol), TX_Timeout); HAL_UART_Transmit_DMA(&hlpuart1,(uint8_t*)InVol,sizeof(InVol)); while(Completed==0) {} Completed =0; //HAL_UART_Transmit(&hlpuart1, (uint8_t *)BatVol ,sizeof(BatVol), TX_Timeout); HAL_UART_Transmit_DMA(&hlpuart1, (uint8_t *)BatVol ,sizeof(BatVol)); while(Completed==0){} Completed =0; HAL_GPIO_WritePin( GPIOC,GPIO_PIN_3,GPIO_PIN_SET); //for auxiliary test } } /* USER CODE END 3 */ } //ADCEOC 中斷處理函數 void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef *hadc) { ADCResult[convCNT]=HAL_ADC_GetValue(&hadc1); //獲取轉換結果并存入數組 convCNT++; if(convCNT==3) { convCNT=0; EndofCon_Flag=0xff; sprintf(WDVol,"Internal PN Temperature: %5.3f ",Temperature); sprintf(InVol,"Internal Reference Volt: %5.3f ",VRefint); sprintf(BatVol,"Current Battery Volt: %5.3f ",VBATVolt); } } //UART DMA傳輸完成處理函數 void HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart) { Completed=0xff; }
基于上面的配置和測試代碼,我們就可以看到最終的結果了。定時器周期性地觸發ADC,每得到3個ADC結果就進行數據處理,然后通過UART以DMA方式傳輸到串口終端。注意VBat電壓是測量結果再乘以3得到的。
針對上面的應用演示,最后給幾點相關應用提醒:
1、針對溫度傳感器做測量時,校準時使用的參考電壓與實際應用不一致時要做換算,換算成相同參考電壓的數據后再做計算。這點前面也提過了。
2、使用TIMER的TRGO觸發ADC,如果選擇類似比較事件、更新事件來觸發ADC時,此時ADC對觸發極性的選擇是無效的,或者說ADC的轉換僅依賴于觸發事件時間點。如果是選擇TIMER的Ocref信號作為觸發源,此時ADC的硬件觸發的極性選擇是有效的,可以是上沿或下沿觸發,甚至是雙沿觸發。這時就得根據需要選擇合適的觸發沿。【可以進一步閱讀本公眾號文章《STM32定時器觸發ADC的時序話題》】
3、這里使用UART的DMA傳輸依次顯示三個結果于串口終端,三個啟動UART DMA傳輸的函數須保留適當時間間隔,即等上次傳輸完成后再啟動下一次傳輸,因為這里每次傳輸使用的是同一DMA通道。否則沒法全部正常輸出。比如上面3次UART DMA傳輸的代碼改成下面這樣子:
HAL_UART_Transmit_DMA(&hlpuart1, (uint8_t *)WDVol ,sizeof(WDVol)); HAL_UART_Transmit_DMA(&hlpuart1,(uint8_t*)InVol,sizeof(InVol)); HAL_UART_Transmit_DMA(&hlpuart1, (uint8_t *)BatVol ,sizeof(BatVol))
這時輸出結果會變成下面的情形,總是只能看到一個結果的輸出,即第一次啟動的DMA傳輸結果。
如果想省事點,直接在相鄰2次DMA傳輸間加上合適延時也行。我這里根據DMA傳輸完成事件來決定執行下一次發送。在DMA傳輸完成中斷里設置Completed變量為非0值表示當前一輪DMA傳輸完成。
4、對于那些在中斷和主程序里都會被訪問的變量,記得將它們冠以volatile。
下圖的三路波形是我調試時輔助使用的。
第一路表示計數器的計數變化,顯然是單向向上計數模式。
第二路是TIMER1通道1的PWM輸出波形。
第三路是我每次基于DMA實現UART發送時拉高拉低的波形。平常管腳電平為高,在實現DMA傳輸過程中拉低。
好,今天的分享就到這里,供君參考。下次再聊。
審核編輯:劉清
-
溫度傳感器
+關注
關注
48文章
3006瀏覽量
157280 -
輸出電壓
+關注
關注
2文章
1360瀏覽量
38863 -
PWM波
+關注
關注
0文章
100瀏覽量
17173 -
VDD
+關注
關注
1文章
316瀏覽量
34550 -
STM32芯片
+關注
關注
0文章
38瀏覽量
4510
原文標題:基于STM32片內信號的ADC應用演示
文章出處:【微信號:stmcu832,微信公眾號:茶話MCU】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
2812片內ADC采樣時間計算
AD7760:24位、2.5 MSPS、100 dB,Σ-Δ ADC,內置片內緩沖 數據手冊

STM32CubeMX | 28 - STM32片內Flash的使用

STM32U5系列片內ADC1和ADC4實現過程(下)

評論