這一節(jié)內(nèi)容比較簡單,就是電壓采樣,在傳統(tǒng)設(shè)計中應(yīng)用還是比較多的。首先看下支持ADC采樣的管腳,找到你手里模塊的原理圖,我的如下所示:
如上所示,U1的2腳為ADC管腳,而且整個模塊有且僅有這一個電壓采集管腳。CDS1是一個光敏電阻,它和R1組成一個分壓電路。不同強度的光照在CDS1上時,會導(dǎo)致它的阻值變化,最終導(dǎo)致ADC管腳上的電壓發(fā)生變化。
那么,關(guān)于ESP8266的ADC,相關(guān)的性能參數(shù)有哪些?
可以打開手冊:2c-esp8266_sdk_api_guide_cn_v1.5.4,在第26頁,有幾個ADC相關(guān)的函數(shù),我們截取其中一個,看一下:
首先是電壓的輸入范圍,也就是ADC的量程:0~1.0V,相比傳統(tǒng)單片機的0~3.3V小了很多。
然后是分辨率:1/1024V。從這個參數(shù)可以知道,這是個10位的AD,只是不知道為何,量程這么小。
接著是三點注意事項:
1、ADC讀取管腳電壓時,需確保管腳連接了外部電路,且沒有超過量程。
這一點很好理解,根據(jù)輸入電壓設(shè)計相應(yīng)的分壓電路,接過來就行了。
2、讀取電壓之前,需要修改esp_init_data_default.bin文件中的第107byte的值,改為VDD3P3管腳3和4上的真實電源電壓值。
先說修改esp_init_data_default.bin文件,這個其實很簡單,因為這個文件是我們燒錄到ESP8266里面的,所以只要找到文件位置,用修改flash的函數(shù)改一下就行。
接下來這句話可能理解起來有點繞,這個VDD3P3管腳3和4是啥?模塊上沒有這兩個管腳啊~
還記得我在前言里面提到過的,ESP8266是樂鑫的芯片,安信可做的模組封裝。所以本文上面的原理圖截圖其實是模塊的管腳分布,并不是真實的芯片管腳。真實的模塊內(nèi)部的芯片原理圖是什么樣?我這里截取其中一部分,看一下:
懂?很簡單。
3、第107byte的值的單位是0.1V,有效取值范圍是18~38.
這個就很好理解了,第二點已經(jīng)說了,第107byte寫入的是VDD3P3管腳的電壓。而我們常用的供電電壓是3.3V,所以要寫入的值是33。因為單位是0.1V,33*0.1V得到3.3V。
接下來看一下這三個函數(shù),system_adc_read() 剛才已經(jīng)看了,它的功能就是讀取ADC電壓值,很簡單,直接調(diào)用讀取就行。
然后是system_get_vdd33(),我們看一下截圖:
簡單來說,是用來測量VDD3P3管腳上的電壓的,可以理解為獲取當(dāng)前的工作電壓。工作前提必須要確保ADC管腳懸空,同時確保esp_init_data_default.bin的第127byte值為0xFF。
還有一個函數(shù),快速高精度的AD采樣,因為篇幅比較長,這里不截圖了,我總結(jié)一下。先看函數(shù)結(jié)構(gòu):
system_adc_read_fast(uint16 *adc_addr, uint16 adc_num, uint8 adc_clk_div)
注意事項和函數(shù)system_get_vdd33()類似,要限制輸入電壓值、修改107byte的值為VDD3P3,不同的地方在于,使用快速采樣函數(shù)的時候,要關(guān)閉wifi和所有中斷。
參數(shù)1:uint16 *adc_addr,ADC連續(xù)采樣輸出的地址指針
參數(shù)2:uint16 adc_num,ADC連續(xù)采樣的點數(shù),范圍1~65535
參數(shù)3:uint8 adc_clk_div,ADC工作時鐘=80M/ adc_clk_div,輸入范圍8~32,建議值8.
假設(shè)我們要連續(xù)采樣50次,那么可以定義一個50個元素的數(shù)組,把數(shù)組首地址給參數(shù)1,數(shù)組大小給參數(shù)2,參數(shù)3沒有特殊情況的話默認輸入8.
所以,用過帶DMA功能的ADC的童鞋,會發(fā)現(xiàn)用法很相似。
接下來進入演示部分,以之前的串口程序為模版,增加AD采樣功能,得到的AD值通過串口助手打印輸出。代碼比較簡單,直接看一下主函數(shù)部分:
void ICACHE_FLASH_ATTRuser_init(void){ partition_item_t partition_item; uint16 vdd33 = 33; uint32 flash_r_w[1024]; uart_init(BIT_RATE_115200, BIT_RATE_115200); spi_flash_read(0x1fc*4096, flash_r_w, 4096); flash_r_w[107/4] = flash_r_w[107/4] & !(0xff<<((107%4)*8)); flash_r_w[107/4] = flash_r_w[107/4] | 33; spi_flash_erase_sector(0x1fc); spi_flash_write(0x1fc*4096,flash_r_w,4096); system_init_done_cb(system_done);}
前幾行比較簡單,就是串口初始化。
接下來就到了修改esp_init_data_default.bin文件中的地方,該文件的地址為什么是0x1fc?
第二節(jié)講程序燒錄的時候,曾經(jīng)說過每個文件的地址,如圖:
我的模塊是16Mbit的,esp_init_data_default.bin文件的起始地址是0x1fc000,0x1000等于10進制的4096,所以0x1fc000=0x1fc*4096。
而我們要修改的是該文件的第107byte,而讀寫flash必須要4字節(jié)對齊,所以后面對107做了一些換算。理解不了的建議看一下第八節(jié)。
VDD33是前面定義的變量,值為33,對應(yīng)3.3V的供電電壓。
很簡單吧?
系統(tǒng)初始化完成的回調(diào)函數(shù)里,我定義了一個軟件定時器,每隔3秒讀取一次ADC的電壓值,并通過串口打印出來:
void system_done(){ wifi_station_disconnect(); os_timer_disarm(&LED_timer); os_timer_setfn(&LED_timer, (os_timer_func_t *)ADC_OUTPUT, NULL); os_timer_arm(&LED_timer, 3000, 1); }void ADC_OUTPUT(){ static adc_value = 0; adc_value = system_adc_read(); os_printf("adc_value is %d ", adc_value);}
細心的人會發(fā)現(xiàn)定義定時器之前有一行代碼:
wifi_station_disconnect();
這是因為我的模塊之前保存了某個環(huán)境下的wifi賬號、密碼。即便主函數(shù)里沒有要求模塊連接wifi,上電后它還是會自動連接,并打印相關(guān)信息。所以,加入這一行代碼,讓它不再連接。
這就完了?是的,so easy!
程序修改完成,保存、清理、編譯、下載一條龍,然后重新上電。這里借助串口助手來查看效果。設(shè)備上電之后,效果如下所示:
如圖所示,上電后開始輸出ADC采集到的電壓值,前面兩個是200多,后面我用手擋住光敏電阻,導(dǎo)致光敏電阻阻值變大,R1分壓得到的電壓變小,只有40多。
實驗完成。
鏈接:
https://pan.baidu.com/s/1yueZQpULiDklHK22TPqsqA
提取碼:tcfa
-
adc
+關(guān)注
關(guān)注
98文章
6525瀏覽量
545232 -
電壓
+關(guān)注
關(guān)注
45文章
5624瀏覽量
116031 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4344瀏覽量
62812
原文標題:ESP8266_22基于自身ADC的電壓采樣
文章出處:【微信號:gh_dae0718828df,微信公眾號:gh_dae0718828df】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
相關(guān)推薦
評論