STC12C2052AD單片機(jī)AD轉(zhuǎn)換C語言程序
掃描二維碼
隨時隨地手機(jī)看文章
首先:這個小程序搞了我兩天,加上焊板子共三天,當(dāng)然是業(yè)余時間。也有24個小時了。
我瞪著電腦兩個晚上,我拆了裝,裝了拆(下載程序不在本板子上)。最后,就在今天?。?ldquo;今”讀四聲)18點(diǎn)??勺屛艺业絾栴}所在了!搞了兩個晚上,AD轉(zhuǎn)換控制流程,根本沒錯?。。?!?。?! 錯就錯在一個小地方,打死都想不到的地方。
具體錯在哪,請?jiān)谙挛闹姓遥。。。?/p>
功能:STC12C2052AD AD轉(zhuǎn)換C程序 +PWM輸出功能 成功使用。
應(yīng)用:AD檢測電壓進(jìn)行過欠壓保護(hù)(繼電器控制)+PWM把直流電壓斬波成脈動直流。
板子功能:給手機(jī)電池充電。
降壓用的LM317,小電流應(yīng)用應(yīng)該夠了。沒時間去買開關(guān)管,就用的9013開關(guān)。
圖紙:
//以下是成功了的程序。如果你需要應(yīng)用在你自己的項(xiàng)目中,您只需要更改io就能直接應(yīng)用了
//程序的完整版本下載地址:http://www.51hei.com/ziliao/file/stc12c2052adde.rar
#include <stc12c2052ad.h> //stc單片機(jī)專用的頭文件
#include <intrins.h>
#define uchar unsigned char
#define uint unsigned int
#define AD_SPEED 0x60 //0110,0000 1 1 270個時鐘周期轉(zhuǎn)換一次,
/************河北正定歡迎您!&&&&少占魚歡迎您!******************************/
//
sbit M=P1^5; //過壓指示燈
sbit N=P1^3; //欠壓指示燈
sbit LED=P1^7; //工作正常燈
sbit CONTRL=P3^4; //輸出控制端
sbit PWM=P3^7;
/****************************************************************/
void pwm();
void delayms(uint);
uint ADC();
void InitADC();
void baohu();
float voltage=0.0;
uint V;
float VCC=5.05;
uchar mtab[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};
/***8**************************************************************/
void main()
{
CONTRL=0;//先關(guān)閉輸出
delayms(700);
V=40; //這些是我差錯的時候添上去的。目的在于弄明白到底AD轉(zhuǎn)換了沒有。
voltage=4.0;//實(shí)踐證明,更換數(shù)值沒用,說明沒AD
LED=0;
CONTRL=1;
voltage=V*VCC/256.00*5.00;
delayms(1000);
PWM=1;
CONTRL=1;//繼電器工作,是保護(hù)狀態(tài)
delayms(1000);
M=0;
N=0;
LED=0;
delayms(2000);
M=1;
N=1;
LED=1;
pwm();//產(chǎn)生PWM波形
delayms(7000);
delayms(100);//延時
InitADC();
delayms(20);
V= ADC();
baohu();
while(1)
{
V= ADC();
baohu();
delayms(300);
}
}
//
//
void pwm()
{
//PCA模塊工作于PWM模式 C程序
CMOD = 0x04; //用定時器0溢出做PCA脈沖
CL = 0x00; //PCA定時器低8位 地址:E9H
CH = 0x00; //PCA高8位 地址 F9H
CCON=0x00;
CCAP0L = 0x60; //PWM模式時他倆用來控制占空比
CCAP0H = 0x60; //0xff-0xc0=0x3f 64/256=25% 占空比(溢出)
CCAPM0 = 0x42; //0100,0010 Setup PCA module 0 in PWM mode
// ECOM0=1使能比較 PWM0=1 使能CEX0腳用作脈寬調(diào)節(jié)輸出
/*********************
PCA 模塊工作模式設(shè)置 (CCAPMn 寄存器 n= 0-3四種)
7 6 5 4 3 2 1 0
- ECOMn CAPPn CAPNn MATn TOGn PWMn ECCFn
選項(xiàng): 0x00 無此操作
0x20 16位捕捉模式,由 CEXn上升沿觸發(fā)
0x10 16位捕捉模式,由CEXn下降沿觸發(fā)
0x30 16位捕捉模式,由CEXn的跳變觸發(fā)
0x48 16位軟件定時器
0x4c 16位高速輸出
0x42 8位PWM輸出
每個PCA模塊另外還對應(yīng)兩個寄存器:CCAPnH和CCAPnL 。 捕獲或者比較時,它們用來
保存16位計數(shù)值,當(dāng)工作于PWM模式時,用來控制占空比
*******************************/
TMOD=0x02;
TH0=0x06;
TL0=0x06;
CR=1; //Start PCA Timer.
TR0=1;
}
//AD轉(zhuǎn)換初始化 ----打開ADC電源
void InitADC()
{
P1=0xff;
ADC_CONTR|=0x80;
delayms(80);
//這兩個寄存器用來設(shè)置 P1口四種狀態(tài),每一位對應(yīng)一個P1引腳 ,按狀態(tài)組合操作
/*****************
P1M0 和P1M1 寄存器位 7 6 5 4 3 2 1 0
P1.7 P1.6 P1.5 P1.4 P1.3 P1.2 P1.1 P1.0
同理P3M0 P3M0 也是。因?yàn)?strong>STC12C2052AD只有兩個P口,所以只有這倆組 STC12C5410AD還多P2M0 P1M0 有三組
P1M0 P1M1 高
0 0 普通I0口 (準(zhǔn)雙向) P1寄存器位 7 6 5 4 3 2 1 0
0 1 強(qiáng)推挽輸出 (20MA電流 )盡量少用 P1.7 P1.6 P1.5 P1.4 P1.3 P1.2 P1.1 P1.0
1 0 僅做輸入 A/D轉(zhuǎn)換時可用此模式
1 1 開漏 ,A/D轉(zhuǎn)換時可用此模式
例如:
要設(shè)置P1.2為 AD 輸入口
則 P1M0=0X02 ;
P1M1=0X02; 開漏即可
當(dāng)不用AD時,最好 關(guān)閉ADC電源 ,恢復(fù)為IO口狀態(tài)
********************************/
P1M0=0x02;//這兩個寄存器用來設(shè)置 P1口四種狀態(tài),每一位對應(yīng)一個P1引腳 ,按狀態(tài)組合操作
P1M1=0x02;//設(shè)置P1.1為開漏狀態(tài)
}
// AD轉(zhuǎn)換程序
/******************************************************
注意:這個函數(shù)里注釋的命令是通用命令,可以針對所有AD通道使用,我這就認(rèn)準(zhǔn)了P1.1一個通道,所以直接
//賦值,省點(diǎn)"流量"!折磨我的問題就出在這個函數(shù)里的while等待語句
while (1) //等待A/D轉(zhuǎn)換結(jié)束
{
if (ADC_CONTR & 0x10) //0001,0000 測試A/D轉(zhuǎn)換結(jié)束否
{ break; }
}
這是能用的,我原來寫的是:
while (ADC_CONTR & 0x10==0);
這樣寫不能用,再說一遍:這樣就不能用了?。?
至于為嘛,因?yàn)?優(yōu)先級,“==”比&優(yōu)先級高,
所以加個括號就可以了
while ( (ADC_CONTR & 0x10) == 0 );
不經(jīng)常用C語言,就會記不住啦!??!
由此得到一個教訓(xùn);小問題影響效率
經(jīng)驗(yàn):經(jīng)常加一加括號會死啊,似乎也不耗“流量”吧!!
*********************************************/
uint ADC()
{
ADC_DATA = 0; //清除結(jié)果
ADC_CONTR = 0x60; //轉(zhuǎn)換速度設(shè)置 0x60 最快速度
ADC_CONTR = 0xE0; //1110,0000 清 ADC_FLAG, ADC_START 位和低 3 位
ADC_CONTR =0xe1;
// ADC_CONTR |= 0x01; //選擇 A/D 當(dāng)前通道 P1.1
delayms(1); //使輸入電壓達(dá)到穩(wěn)定
ADC_CONTR = 0xe9;
// ADC_CONTR |= 0x08; //0000,1000 令 ADCS = 1, 啟動A/D轉(zhuǎn)換,
while (1) //等待A/D轉(zhuǎn)換結(jié)束
{
if (ADC_CONTR & 0x10) //0001,0000 測試A/D轉(zhuǎn)換結(jié)束否
{ break; }
}
ADC_CONTR =0xe1;
//ADC_CONTR &= 0xE7; //1111,0111 清 ADC_FLAG 位, 關(guān)閉A/D轉(zhuǎn)換,
return ADC_DATA; //返回 A/D 10 位轉(zhuǎn)換結(jié)果
}
//
void baohu()
{
voltage=V*VCC/256.00*5.00;
if( voltage>5.25)
{ CONTRL=1;//過壓保護(hù) ,關(guān)斷開關(guān)管控制端
M=0;
N=1;
LED=1;
}
if(voltage<4.60)
{
CONTRL=1;//保護(hù)繼電器打開,常閉觸點(diǎn)斷開保護(hù)
N=0;
M=1;
LED=1;
}
if(voltage>4.62&&voltage<5.24)
{
LED=0;
M=1;
N=1;
CONTRL=0;//繼電器斷開,正常狀態(tài)
}
}
void delayms(uint k)
{
uint data i,j;
for(i=0;i