以下代碼和文字說明,參考了STC公司提供的技術(shù)手冊(STC15.pdf)。具體代碼如下:
本文引用地址:http://cafeforensic.com/article/201611/322096.htm#include
#include"init.h"
#include"led.h"
#include"buzzer.h"
#include"delayms.h"
#include"uart.h"
//ADC_CONTR寄存器的位取值
#define ADC_POWER0x80//打開ADC轉(zhuǎn)化器電源
#define ADC_FLAG0x10//ADC完成標(biāo)志位
#define ADC_START0x08//ADC開始控制位
#define ADC_SPEEDLL0x00//540個時鐘周期轉(zhuǎn)換一次
#define ADC_SPEEDL0x20//360個時鐘周期轉(zhuǎn)換一次
#define ADC_SPEEDH0x40//180個時鐘周期轉(zhuǎn)換一次
#define ADC_SPEEDHH0x60// 90個時鐘周期轉(zhuǎn)換一次
#define ADC_CH10x01//選擇通道1
#define ADC_CH20x02//選擇通道2
#define ADC_CH30x03//選擇通道3
#define ADC_CH40x04//選擇通道4
//初始化ADC
void init_ADC(void)
{
P1ASF= 0x1e; //P11-P14口作為模擬功能A/D使用
ADC_RES= 0;//清除結(jié)果寄存器
ADC_RESL= 0;
PCON&= 0xdf;//ADCRES存放高8位ADC結(jié)果,ADCRESL[1:0]存放低2位結(jié)果
ADC_CONTR = ADC_POWER|ADC_SPEEDLL|ADC_START|ADC_CH2;//啟動轉(zhuǎn)換,通道2
delayms(1);//ADC上電延時
EA= 1;//CPU開啟總中斷
EADC = 1;//允許ADC中斷
}
void main(void)
{
init_IO(); //初始化
init_sym();
init_uart();
init_ADC();
while(1)
{
while(!(ADC_CONTR & ADC_FLAG));//等待ADC完成
ADC_CONTR &= !ADC_FLAG;//清除ADC中斷
sendData(ADC_RES);
sendData(ADC_RESL);
led_ye_on();
ADC_RES= 0;//獲取ADC結(jié)果后,清除結(jié)果寄存器
ADC_RESL= 0;
//AD轉(zhuǎn)換結(jié)束后會自動停止轉(zhuǎn)換,所以這里再次開啟ADC
//并且一定要重新開啟ADC電源、設(shè)置ADC通道和ADC轉(zhuǎn)換速率
ADC_CONTR = ADC_POWER|ADC_SPEEDLL|ADC_START|ADC_CH2;//啟動轉(zhuǎn)換,通道2
delayms(1000);
}
}
自帶ADC的使用方法請參照以上代碼。
需要注意的問題:
1、在啟動AD轉(zhuǎn)換前,給AD轉(zhuǎn)換器上電,并且上電后需要延時一下;
2、對ADC控制寄存器(ADC_CONTR)的操作最好是直接賦值,不要用‘&’和‘|’;
3、在ADC轉(zhuǎn)換過程中不要改變I/O口的狀態(tài),即使試圖改變也會失??;
4、ADC實現(xiàn)一次轉(zhuǎn)換結(jié)束后,AD轉(zhuǎn)換將會自動關(guān)閉,如果要實現(xiàn)連續(xù)轉(zhuǎn)換,則需要再次啟動AD轉(zhuǎn)換(將ADC_CONTR 寄存器的ADC_START位置1)。因為需要對ADC_CONTR寄存器直接賦值,所以 要對ADC_CONTR寄存器再次賦值(該值和ADC初始化中的值一致)。實踐也證明了,如果直接利用或語句將ADC_START置位(ADC_CONTR |= ADC_START;),則AD轉(zhuǎn)換失敗,通過調(diào)試測試,初步判斷是ADC_CONTR寄存器的ADC_POWER位不等于1,也就是說啟動AD轉(zhuǎn)換前,AD轉(zhuǎn)換器沒有上電,這樣AD轉(zhuǎn)換自然不會成功。
5、如果需要改變I/O口的狀態(tài),最好是在AD轉(zhuǎn)換開始之前進行。
以上的內(nèi)容純屬個人學(xué)習(xí)體會,如有不當(dāng)之處,請指正!
評論