STC15系列單片機內部集成了8路10位高速A/D轉換器。STC15系列單片機的A/D轉換口在P1口(P1.7-P1.0),有8路10位高速A/D轉換器,速度到300KHz(30萬次/秒)。8路電壓輸入型A/D,可做溫度檢測、電池電壓檢測、按鍵掃描、頻譜檢測等。
一、A/D轉換器的結構
STC15系列單片機ADC由多路選擇開關、比較器、逐次比較寄存器、10位DAC、轉換結果寄存器(ADC_RES和ADC_RESL)以及ADC_CONTR構成。
STC15系列單片機的ADC是逐次比較型ADC。逐次比較型ADC由一個比較器和D/A轉換器構成,通過逐次比較邏輯,從最高位(MSB)開始,順序地對每一輸入電壓與內置D/A轉換器輸出進行比較,經過多次比較,使轉換所得的數字量逐次逼近輸入模擬量對應值。逐次比較型A/D轉換器具有速度高,功耗低等優點。
從上圖可以看出,通過模擬多路開關,將通過ADC0 ~ 7的模擬量輸入送給比較器。用數/模轉換器(DAC)轉換的模擬量與輸入的模擬量通過比較器進行比較,將比較結果保存到逐次比較寄存器,并通過逐次比較寄存器輸出轉換結果。A/D轉換結束后,最終的轉換結果保存到ADC轉換結果寄存器ADC_RES和ADC_RESL,同時,置位ADC控制寄存器ADC_CONTR中的A/D轉換結束標志位ADC_FLAG,以供程序查詢或發出中斷申請。模擬通道的選擇控制由ADC控制寄存器ADC_CONTR中的CHS2~CHS0確定。ADC的轉換速度由ADC控制寄存器中的SPEED1和SPEED0確定。在使用ADC之前,應先給ADC上電,也就是置位ADC控制寄存器中的ADC_POWER位。
當CLK_DIV.5(PCON2.5)/ADRJ = 0時,A/D轉換結果寄存器格式如下:
當ADRJ=0時,如果取10位結果,則按下面公式計算:
當ADRJ=0時,如果取8位結果,按下面公式計算:
當CLK_DIV.5(PCON2.5)/ADRJ = 1時,A/D轉換結果寄存器格式如下:
當ADRJ=1時,如果取10位結果,則按下面公式計算:
式中,Vin為模擬輸入通道輸入電壓,Vcc為單片機實際工作電壓,用單片機工作電壓作為模擬參考電壓。
二、與A/D轉換相關的寄存器
與STC15系列單片機A/D轉換相關的寄存器列于下表所示。
2.1 P1口模擬功能控制寄存器P1ASF
STC15系列單片機的A/D轉換口在P1口(P1.7-P1.0),有8路10位高速A/D轉換器速度可達到到300KHz(30萬次/秒)。8路電壓輸入型A/D,可做溫度檢測、電池電壓檢測、按鍵掃描、頻譜檢測等。上電復位后P1口為弱上拉型I/O口,用戶可以通過軟件設置將8路中的任何一路設置為A/D轉換,不需作為A/D使用的P1口可繼續作為I/O口使用(建議只作為輸入)。需作為A/D使用的口需先將P1ASF特殊功能寄存器中的相應位置為‘1’,將相應的口設置為模擬功能。P1ASF存器的格式如下:
P1ASF : P1口模擬功能控制寄存器(該寄存器是只寫寄存器,讀無效)
2.2 ADC控制寄存器ADC_CONTR
ADC_CONTR寄存器的格式如下:
ADC_CONTR : ADC控制寄存器
對ADC_CONTR寄存器進行操作,建議直接用MOV賦值語句,不要用‘與’和‘或’語句。
ADC_POWER: ADC 電源控制位。
0:關閉ADC 電源;
1:打開A/D轉換器電源.
建議進入空閑模式和掉電模式前,將ADC電源關閉,即ADC_POWER =0,可降低功耗。
啟動A/D轉換前一定要確認A/D電源已打開,A/D轉換結束后關閉A/D電源可降低功耗,也可不關閉。初次打開內部A/D轉換模擬電源,需適當延時,等內部模擬電源穩定后,再啟動A/D轉換。
建議啟動A/D轉換后,在A/D轉換結束之前,不改變任何I/O口的狀態,有利于高精度A/D轉換,如能將定時器/串行口/中斷系統關閉更好。
SPEED1,SPEED0:模數轉換器轉換速度控制位
ADC_FLAG: 模數轉換器轉換結束標志位,當A/D轉換完成后,ADC_FLAG = 1,要由軟件清0。不管是A/D 轉換完成后由該位申請產生中斷,還是由軟件查詢該標志位A/D轉換是否結束,當A/D轉換完成后,ADC_FLAG = 1,一定要軟件清0。
ADC_START:模數轉換器(ADC)轉換啟動控制位,設置為“1”時,開始轉換,轉換結束后為0。
CHS2/CHS1/CHS0:模擬輸入通道選擇,CHS2/CHS1/CHS0
2.3 ADC轉換結果調整寄存器位——ADRJ
ADC轉換結果調整寄存器位——ADRJ位于寄存器CLK_DIV/PCON中,用于控制ADC轉換結果存放的位置。
ADRJ:ADC轉換結果調整
0:ADC_RES[7:0]存放高8位ADC結果,ADC_RESL[1:0]存放低2位ADC結果
1:ADC_RES[1:0]存放高2位ADC結果,ADC_RESL[7:0]存放低8位ADC結果
2.4 A/D轉換結果寄存器ADC_RES、ADC_RESL
特殊功能寄存器ADC_RES和ADC_RESL寄存器用于保存A/D轉換結果,其格式如下:
CKKO_DIV寄存器的ADRJ位是A/D轉換結果寄存器(ADC_RES,ADC_RESL)的數據格式調整控制位。
當ADRJ=0時,10位是A/D轉換結果的高8位存放在在ADC_RES中,低2位存放在ADC_RESL的低2位中。
2.5 中斷允許寄存器IE
IE : 中斷允許寄存器 (可位尋址)
EA : CPU的中斷開放標志
EA=1,CPU開放中斷,
EA=0,CPU屏蔽所有的中斷申請。
EA的作用是使中斷允許形成多級控制。即各中斷源首先受EA控制;其次還受各中斷源自己的中斷允許控制位控制。
EADC : A/D轉換中斷允許位
EADC=1,允許A/D轉換中斷,
EADC=0,禁止A/D轉換中斷。
2.6 A/D轉換典型應用線路
三、測試程序
3.1 中斷方式
#include "stc15.h"
#include "intrins.h"
#include "delay.h"
#define uchar unsigned char
#define uint unsigned int
#define FOSC 11059200L //系統頻率
#define BAUD 9600 //串口波特率
void UatrInit();
void SendData(uchar dat);
void SendString(char *s);
void AdInit();
uchar num[10] = {'0','1','2','3','4','5','6','7','8','9'};
uint adc_result = 0;
void main()
{
P1M0 = 0x02;
P1M1 = 0x00;
UatrInit();
AdInit();
EA = 1; // CPU開放中斷
while (1);
}
// 初始化串口
void UatrInit()
{
SCON = 0x50; //8位可變波特率 串口工作模式1
T2L = (65536 - (FOSC/4/BAUD)); //設置波特率重裝值
T2H = (65536 - (FOSC/4/BAUD)) >?>8;
AUXR = 0x14; //T2為1T模式, 并啟動定時器2
AUXR |= 0x01; //選擇定時器2為串口1的波特率發生器
ES = 1; //使能串口1中斷
}
// 初始化ADC
void AdInit()
{
P1ASF = 0x01; // P1.0作為模擬功能A/D使用
ADC_RES = 0;
ADC_RESL = 0; // 結果寄存器清零
ADC_CONTR = 0x88; // 打開ADC的電源 540個周期轉換一次 選擇P1.0作為A/D輸入來用
delayus(20);
EADC = 1; // 允許A/D轉換中斷
}
// ADC中斷服務函數
void adc_isr() interrupt 5
{
ADC_CONTR &= !0x10; // 清除ADC中斷標志
adc_result = ADC_RES*4 + ADC_RESL; // 獲取ADC結果,高2位在前
SendData(num[adc_result/1000]); // 千
SendData(num[adc_result%1000/100]); // 百
SendData(num[adc_result%100/10]); // 十
SendData(num[adc_result%10]); // 個
// SendData(ADC_RES);
// SendData(ADC_RESL);
SendString("\\r\\n"); // 換行
ADC_CONTR =