|
以下程序為使用STC12C5A60S2單片機,采用ADC中斷掃描模式,并且初始化定時器P進行WM輸出,用串口輸出溫度數值:
難點:ADC中斷與PWM輸出靠定時器0發生并且串口通訊波特率靠定時器1發生,這幾個中斷在代碼編寫與調試過程中,發現這幾個程序獨立應用的代碼直接移植過來,并不能正常使用,這幾個中斷的優先級順序和初始化順序會產生相互干擾,因此經過查找相關資料,最終解決了這個問題,下面為詳細代碼(注釋是手打上去的,不知道為啥從keil往這里粘貼會出現問題)
void PWM_init() //PWM初始化函數,初始化占空比為0
{
CMOD=0x02; //0000 0010 空閑時不計數,不產生中斷,時鐘源fosc/2,因此輸出占用率fosc/512
CL=0x00;
CH=0x00;
CCAPM0=0x42; //0100 001,8位PWM
CCAP0L=0x00;
CCAP0H=0x00; //PWM實現方法,因為是8位,CL基礎計算器從0xFF遞減到0x00,溢出后將CCAPOH的值加載到CCAPOL,若CL遞減過程中,若大于CCAPOL,則輸出高電平,否則為低電平
CCAPM1=0x42;
CCAP1H=0x00;
CCAP1L=0x00;
CR = 1; //開啟PCA計數器
}
void PWM_set(uchar x,uchar y) //占空比設置函數 引腳為P1.3和P1.4
{
CCAP0H=x*2.55; /占空比公式D=(256-CCAPnH)/256(8位PWM模式)
CCAP0L=x*2.55; //乘2.55轉換為0--100%
CCAP1H=y*2.55;
CCAP1L=y*2.55;
}
void Timer0_Init(void)
{
TH0=(65536-TIME)/256; //開定時器0
TL0=(65536-TIME)%256;
EA=1; //開全局中斷
ET0=1; //允許定時器0中斷
TR0=1;
IE = 0xa0;
}
void timer0() interrupt 1 //interrupt 1: ¶¨ê±Æ÷0£¬interrupt3£o¶¨ê±Æ÷3
{
TH0=(65536-TIME)/256; //高八位,(需要表示Xms的定時,計數器由65536-X數到65536,由于16位,只能分高低位)
TL0=(65536-TIME)%256; /低八位
num++;
if(num==500)
{
flagTime ++;
num = 0;
}
Init_ADC();
}
/*------------------------------------------------
串口初始化
------------------------------------------------*/
void Init_UART (void)
{
SCON = 0x50; //8位數據,可變波特率
TMOD |= 0x20; //設定定時器1為16位自動重裝方式
TL1 = 0xC0; //設定定時器高8位初值
TH1 = 0xFA; //設定定時器低8位初值
TR1 = 1; //啟動定時器
EA = 1;
}
/*------------------------------------------------
發送一個字節
------------------------------------------------*/
void SendByte(unsigned char dat)
{
SBUF = dat;
while(!TI);
TI = 0;
}
/*------------------------------------------------
發送一個字符串
------------------------------------------------*/
void SendStr(unsigned char *s)
{
while(*s!='\0')// \0表示字符串結束標志
/通過檢測是否是字符串末尾
{
SendByte(*s);
s++;
}
}
/*----------------------------
ADC interrupt service routine
----------------------------*/
void adc_isr() interrupt 5 using 1
{
//V_5REF=V_1REF*256/da_ref;
if(ADC_Chanul_Turn%2==0) //外部基準電壓
{
temp1=ADC_RES; //獲取轉換結果
ADC_1Data=((double)temp1/256)*5; //取八位計算基準電壓Data
adc1 = ADC_1Data;
}
if(ADC_Chanul_Turn%2==1)
{
temp0=ADC_RES; //獲取轉換結果
ADC_0Data=((double)temp0/256)*5; //取八位計算實際值Data
adc0 =ADC_0Data;
}
ADC_CONTR = ADC_POWER | ADC_SPEEDH | ADC_START | ADC_Chanul_Turn;
ADC_Chanul_Turn++;
if(ADC_Chanul_Turn==2)
ADC_Chanul_Turn=0;
}
/*----------------------------
Initial ADC sfr
----------------------------*/
void Init_ADC()
{
P1ASF = 0x03; //Set all P1 as analog input port
ADC_RES = 0; //Clear previous result
// ADC_CONTR = ADC_POWER | ADC_SPEEDLL | ADC_START | ch;
if(ADC_Chanul_Turn%2==0)//外部基準電壓
{
ADC_CONTR=0xF0; //AD轉換控制器,1111,0000 P10口
_nop_();
_nop_();
_nop_();
_nop_();
ADC_CONTR=0xE8; //1110,1000 (清0flag,置位start)
}
if(ADC_Chanul_Turn%2==1)
{
ADC_CONTR=0xF1; // 1111,0001 P11口
_nop_();
_nop_();
_nop_();
_nop_();
ADC_CONTR=0xE9; //1110,1001
}
delay_1ms(20); //ADC power-on delay and Start A/D conversion
}
|
評分
-
查看全部評分
|