内容转自他人,至于原创者是谁,我也没找到。应本人觉得内容很好,对本人很有用,所以转载。源代码文件下载地址在下面
功能:STC12C2052AD AD转换C程序 +PWM输出功能 成功使用。
应用:AD检测电压进行过欠压保护(继电器控制)+PWM把直流电压斩波成脉动直流。
板子功能:给手机电池充电。
降压用的LM317,小电流应用应该够了。没时间去买开关管,就用的9013开关。
图纸:
//以下是成功了的程序。如果你需要应用在你自己的项目中,您只需要更改io就能直接应用了
//程序的完整版本下载地址:http://www.51hei.com/ziliao/file/stc12c2052adde.rar
#include <stc12c2052ad.h> //stc单片机专用的头文件
#include <intrins.h>
#define uchar unsigned char
#define uint unsigned int
#define AD_SPEED 0x60 //0110,0000 1 1 270个时钟周期转换一次,
/************河北正定欢迎您!&&&&少占鱼欢迎您!******************************/
//
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;//先关闭输出
delayms(700);
V=40; //这些是我差错的时候添上去的。目的在于弄明白到底AD转换了没有。
voltage=4.0;//实践证明,更换数值没用,说明没AD
LED=0;
CONTRL=1;
voltage=V*VCC/256.00*5.00;
delayms(1000);
PWM=1;
CONTRL=1;//继电器工作,是保护状态
delayms(1000);
M=0;
N=0;
LED=0;
delayms(2000);
M=1;
N=1;
LED=1;
pwm();//产生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脚用作脉宽调节输出
/*********************
PCA 模块工作模式设置 (CCAPMn 寄存器 n= 0-3四种)
7 6 5 4 3 2 1 0
- ECOMn CAPPn CAPNn MATn TOGn PWMn ECCFn
选项: 0x00 无此操作
0x20 16位捕捉模式,由 CEXn上升沿触发
0x10 16位捕捉模式,由CEXn下降沿触发
0x30 16位捕捉模式,由CEXn的跳变触发
0x48 16位软件定时器
0x4c 16位高速输出
0x42 8位PWM输出
每个PCA模块另外还对应两个寄存器:CCAPnH和CCAPnL 。 捕获或者比较时,它们用来
保存16位计数值,当工作于PWM模式时,用来控制占空比
*******************************/
TMOD=0x02;
TH0=0x06;
TL0=0x06;
CR=1; //Start PCA Timer.
TR0=1;
}
//AD转换初始化 ----打开ADC电源
void InitADC()
{
P1=0xff;
ADC_CONTR|=0x80;
delayms(80);
//这两个寄存器用来设置 P1口四种状态,每一位对应一个P1引脚 ,按状态组合操作
/*****************
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 也是。因为STC12C2052AD只有两个P口,所以只有这俩组 STC12C5410AD还多P2M0 P1M0 有三组
P1M0 P1M1 高
0 0 普通I0口 (准双向) P1寄存器位 7 6 5 4 3 2 1 0
0 1 强推挽输出 (20MA电流 )尽量少用 P1.7 P1.6 P1.5 P1.4 P1.3 P1.2 P1.1 P1.0
1 0 仅做输入 A/D转换时可用此模式
1 1 开漏 ,A/D转换时可用此模式
例如:
要设置P1.2为 AD 输入口
则 P1M0=0X02 ;
P1M1=0X02; 开漏即可
当不用AD时,最好 关闭ADC电源 ,恢复为IO口状态
********************************/
P1M0=0x02;//这两个寄存器用来设置 P1口四种状态,每一位对应一个P1引脚 ,按状态组合操作
P1M1=0x02;//设置P1.1为开漏状态
}
// AD转换程序
/******************************************************
注意:这个函数里注释的命令是通用命令,可以针对所有AD通道使用,我这就认准了P1.1一个通道,所以直接
//赋值,省点"流量"!折磨我的问题就出在这个函数里的while等待语句
while (1) //等待A/D转换结束
{
if (ADC_CONTR & 0x10) //0001,0000 测试A/D转换结束否
{ break; }
}
这是能用的,我原来写的是:
while (ADC_CONTR & 0x10==0);
这样写不能用,再说一遍:这样就不能用了!!
至于为嘛,因为 优先级,“==”比&优先级高,
所以加个括号就可以了
while ( (ADC_CONTR & 0x10) == 0 );
不经常用C语言,就会记不住啦!!!
由此得到一个教训;小问题影响效率
经验:经常加一加括号会死啊,似乎也不耗“流量”吧!!
*********************************************/
uint ADC()
{
ADC_DATA = 0; //清除结果
ADC_CONTR = 0x60; //转换速度设置 0x60 最快速度
ADC_CONTR = 0xE0; //1110,0000 清 ADC_FLAG, ADC_START 位和低 3 位
ADC_CONTR =0xe1;
// ADC_CONTR |= 0x01; //选择 A/D 当前通道 P1.1
delayms(1); //使输入电压达到稳定
ADC_CONTR = 0xe9;
// ADC_CONTR |= 0x08; //0000,1000 令 ADCS = 1, 启动A/D转换,
while (1) //等待A/D转换结束
{
if (ADC_CONTR & 0x10) //0001,0000 测试A/D转换结束否
{ break; }
}
ADC_CONTR =0xe1;
//ADC_CONTR &= 0xE7; //1111,0111 清 ADC_FLAG 位, 关闭A/D转换,
return ADC_DATA; //返回 A/D 10 位转换结果
}
void baohu()
{
voltage=V*VCC/256.00*5.00;
if( voltage>5.25)
{
CONTRL=1;//过压保护 ,关断开关管控制端
M=0;
N=1;
LED=1;
}
if(voltage<4.60)
{
CONTRL=1;//保护继电器打开,常闭触点断开保护
N=0;
M=1;
LED=1;
}
if(voltage>4.62&&voltage<5.24)
{
LED=0;
M=1;
N=1;
CONTRL=0;//继电器断开,正常状态
}
}
void delayms(uint k)
{
uint data i,j;
for(i=0;i<k;i++)
{
for(j=0;j<600;j++)
{;}
}
}
来源:https://blog.csdn.net/JIU_LU/article/details/99061624