
A=02AH,CY=1。执行DA A,运行结果 A=90H, CY的值变为多少
DAA时,是加上了66H,调整为 90H,并产生进位。
如果DAA影响标志位,CY就是1。
但是老师说模拟程序不一定对 而且书上只介绍了 该程序不影响溢出位
如果DAA不影响标志位,CY就应该不变,即为原来的:CY=1。
这是一个例程,去掉那些通信显示的,很小。而且想做单片机,就要先看开发文档,看完了就会发现编程序特别简单,而且开发文档里会提供最基本的程序例子的
/
文件名:AD_CAIYANGC
功能:使用AD采集电压显示在LCD
说明:转自网络,本人验证通过
/
#include<STC12C5A60S2H>
#define uint unsigned int
#define uchar unsigned char
sbit CS=P2^0; //LCD12864串行通信片选
sbit SID=P2^1; //LCD12864串行通信数据口
sbit SCLK=P2^2; //LCD12864串行通信同步时钟信号
sbit PSB=P2^5; //LCD12864并/串选择:H并行 L串行
unsigned int temp1,sh1,ge1,n1,m1;
unsigned char ad_result_data[10]; //AD转换高八位
unsigned char ad_result_low2[10]; //AD转换低八位
unsigned char ad_result_total[10]; //AD转换总十位
unsigned char ad_average_result; //AD转换十次的平均值
unsigned char Ain,Vin;
unsigned char b,t,R;
char tp=0;
unsigned char code ma1[6]={0xb5,0xe7,0xd1,0xb9,0xa1,0xc3}; //电压:
unsigned char code ma2[]={""};
uchar code disp1[]={"提示: 按1 键进入"};
uchar code disp2[]={"功能选择界面 "};
unsigned char code num0[]={0xa3,0xb0};
unsigned char code num1[]={0xa3,0xb1};
unsigned char code num2[]={0xa3,0xb2};
unsigned char code num3[]={0xa3,0xb3};
unsigned char code num4[]={0xa3,0xb4};
unsigned char code num5[]={0xa3,0xb5};
unsigned char code num6[]={0xa3,0xb6};
unsigned char code num7[]={0xa3,0xb7};
unsigned char code num8[]={0xa3,0xb8};
unsigned char code num9[]={0xa3,0xb9};
//-------模块延时程序---------------------------- 1ms
void delay1ms(uint delay1ms) //STC11F60XE,221184M,延时1ms
{
uint i,j;
for(;delay1ms>0;delay1ms--)
for(i=0;i<7;i++)
for(j=0;j<210;j++);
}
void delay(uint delay) //STC11F60XE,221184M,延时170us
{
uint i,j;
for(;delay>0;delay--)
for(i=0;i<124;i++);
for(j=0;j<124;j++);
}
/
AD转换程序
/
void AD_initiate() //初始化函数
{
ES=0;
TMOD=0x21; //定时计数器方式控制寄存器,"自动重装,16位计数器"
SCON=0x50; //串行控制寄存器,方便在串口助手那观察
TH1=0xfa;
TL1=0xfa;
TR1=1;
}
void ADC_Power_On() //AD转换电
{
ADC_CONTR|=0x80;
delay(5); //必要的延时
}
void get_ad_result() //取AD结果函数,它是十位AD转换,每十次平均,最后取低八位作为AD采样数据
{
uint i,q=0;
for(i=0;i<10;i++)
{
tp=0;
ADC_RES=0; //高八位数据清零,STC12C5A60S2 AD数据寄存名与STC12C54××系列不同
ADC_RESL=0; //低两位清零
ADC_CONTR|=0x08; //启动AD转换
while(!tp) //判断AD转换是否完成
{
tp=0x10;
tp&=ADC_CONTR;
}
ADC_CONTR&=0xe7;
ad_average_result=ADC_RES;
q=q+ad_average_result;
}
ad_average_result=q/10;
//ad_average_result=ad_average_result45000/1024;
}
/AD转换结束/
void send_ad_result() //取AD结果函数发送到串口,方便调试
{
SBUF=n1;
while(TI==0) ;
TI=0;
delay1ms(100);
//SBUF=R>>4;
}
//---------------------电压采样程序-------------------------
void caiyangP10() //测电压
{
P1M0|=0x01; //设P1_0为开漏模式 如: P1_0= #00000000B
P1M1|=0x01;
ADC_CONTR=0xe0; //设置P10为输入AD转换口
delay(2);
get_ad_result(); //取转换数据
Vin=ad_average_result;
R=Vin;
}
/-----------写控制字到LCD12864------------/
void write_cmd(uchar cmd)
{
uchar i;
uchar i_data;
i_data=0xf8; //命令控制字:11111000写指令 11111010写数据 11111100读状态 11111110读数据
CS=1; //片选置高,才能进行读写 *** 作
SCLK=0;
/----------写命令控制字-----------------/
for(i=0;i<8;i++) //循环八次,每次读取一位数据
{
SID=(bit)(i_data&0x80); //bit表示取其最高位
SCLK=0;
SCLK=1; //正跳变写入指令
i_data=i_data<<1; //左移一位
}
/---------------------------------------/
/----------写指令高四位-----------------/
i_data=cmd;
i_data=i_data&0xf0; //把低四位置0
for(i=0;i<8;i++) //循环八次,每次读取一位数据
{
SID=(bit)(i_data&0x80); //bit表示取其最高位
SCLK=0;
SCLK=1; //正跳变写入指令
i_data=i_data<<1; //左移一位
}
/---------------------------------------/
/----------写指令低四位-----------------/
i_data=cmd;
i_data=i_data<<4; //左移四位,把低四位的数据移到高四位,再把低四位置0
for(i=0;i<8;i++) //循环八次,每次读取一位数据
{
SID=(bit)(i_data&0x80); //bit表示取其最高位
SCLK=0;
SCLK=1; //正跳变写入指令
i_data=i_data<<1; //左移一位
}
/-----------------------------------------/
CS=0; //把片选置低
delay1ms(5); //延时是因为没有进行忙检测,适当的延时可以不进行忙检测
}
/-----------------------------------------/
/------------写数据到LCD12864-------------/
void write_dat(uchar dat)
{
uchar i;
uchar i_data;
i_data=0xfa;
CS=1;
for(i=0;i<8;i++)
{
SID=(bit)(i_data&0x80);
SCLK=0;
SCLK=1;
i_data=i_data<<1;
}
i_data=dat;
i_data=i_data&0xf0;
for(i=0;i<8;i++)
{
SID=(bit)(i_data&0x80);
SCLK=0;
SCLK=1;
i_data=i_data<<1;
}
i_data=dat;
i_data=i_data<<4;
for(i=0;i<8;i++)
{
SID=(bit)(i_data&0x80);
SCLK=0;
SCLK=1;
i_data=i_data<<1;
}
CS=0;
delay1ms(5);
}
/-----------------------------------------/
/--------------显示坐标-------------------/
void lcd_pos(uchar x,uchar y) //汉字显示坐标,x为哪一行,y为哪一列
{
uchar pos;
if(x==0)
x=0x80; //第一行
else if(x==1)
x=0x90; //第二行
else if(x==2)
x=0x88; //第三行
else if(x==3)
x=0x98; //第四行
pos=x+y; //显示哪一行(总共有4行)哪一竖(总共有8竖,每16列为1竖)
write_cmd(pos);
}
/-----------------------------------------/
/--------------显示8个汉字-------------------/
void disp_hanzi(uchar code chn)
{
uchar i;
write_cmd(0x30); //基本指令 *** 作方式
for(i=0;i<16;i++) //16列8个汉字=128(刚好)
write_dat(chn[i]);
}
/-----------------------------------------/
/--------------显示数字-------------------/
void disp_num(uchar code chn)
{
uchar i;
write_cmd(0x30); //基本指令 *** 作方式
for(i=0;i<2;i++) //1个数字
write_dat(chn[i]);
}
void disp_number(uchar num)
{
switch(num)
{
case 0: disp_num(num0);break;
case 1: disp_num(num1);break;
case 2: disp_num(num2);break;
case 3: disp_num(num3);break;
case 4: disp_num(num4);break;
case 5: disp_num(num5);break;
case 6: disp_num(num6);break;
case 7: disp_num(num7);break;
case 8: disp_num(num8);break;
case 9: disp_num(num9);break;
default: break;
}
}
/----------- --LCD初始化------------------/
void lcd_init()
{
PSB=0;
write_cmd(0x30); //基本指令
write_cmd(0x02); //地址归位
write_cmd(0x06); //游标右移
write_cmd(0x0c); //整体显示
write_cmd(0x01); //清屏
}
/-----------------------------------------/
void displayP10()
{
float ad1;
//unsigned int temp1,sh1,ge1,n1,m1;
//uchar code dis2[]={0x01,0x02,0x00};
//ad1=x78125; //电压修正
uchar i;
ad1=Vin39608; //具体线性参数由输入电压值调整,该值的测量范围为0-1000V,5V左右的测量比较准确,
//两端的最大误差为70mv,其他一般在40mv以内
temp1=(int)ad1;
sh1=temp1/1000; //十位
ge1=(temp1%1000)/100; //个位
n1=((temp1%1000)%100)/10; //小数点后一位
m1=((temp1%1000)%100)%10; //小数点后二位
//write_cmd(0x01);
write_cmd(0x30); //基本指令 *** 作方式
lcd_pos(0,0);
for(i=0;i<6;i++) write_dat(ma1[i]);
lcd_pos(0,3);
disp_number(sh1);
lcd_pos(0,4);
disp_number(ge1);
lcd_pos(0,5);
for(i=0;i<2;i++) write_dat(ma2[i]);
lcd_pos(0,6);
disp_number(n1);
lcd_pos(0,7);
disp_number(m1);
/lcd_pos(2,0);
disp_hanzi(disp1);
lcd_pos(3,0);
disp_hanzi(disp2);/
}
void main()
{
EA=1;
AD_initiate(); //初始化
ADC_Power_On(); //开AD电源
//init();
lcd_init();
delay(10);
while(1)
{
caiyangP10(); //测电压
send_ad_result();
//Vin=Vin4007;
displayP10();
delay(10);
}
}
DA,AD都在里面。。DA芯片是PCF8591ADDA
IIC协议 PCF8591ADDA转换
#include<reg52h> //包含头文件,一般情况不需要改动,头文件包含特殊功能寄存器的定义
#include <intrinsh> //包含NOP空指令函数_nop_();
#define AddWr 0x90 //写数据地址
#define AddRd 0x91 //读数据地址
sbit Sda=P1^2; //定义总线连接端口
sbit Scl=P1^1;
bit ADFlag; //定义AD采样标志位
unsigned char code Datatab[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};//7段数共阴码管段码表
data unsigned char Display[8];//定义临时存放数码管数值
/------------------------------------------------
延时程序
------------------------------------------------/
void mDelay(unsigned char j)
{
unsigned int i;
for(;j>0;j--)
{
for(i=0;i<125;i++)
{;}
}
}
/------------------------------------------------
初始化定时器1
------------------------------------------------/
void Init_Timer1(void)
{
TMOD |= 0x10;
TH1=0xff; / Init value /
TL1=0x00;
//PT1=1; / 优先级 /
EA=1; / interupt enable /
ET1=1; / enable timer1 interrupt /
TR1=1;
}
/------------------------------------------------
启动IIC总线
------------------------------------------------/
void Start(void)
{
Sda=1;
_nop_();
Scl=1;
_nop_();
Sda=0;
_nop_();
Scl=0;
}
/------------------------------------------------
停止IIC总线
------------------------------------------------/
void Stop(void)
{
Sda=0;
_nop_();
Scl=1;
_nop_();
Sda=1;
_nop_();
Scl=0;
}
/------------------------------------------------
应答IIC总线
------------------------------------------------/
void Ack(void)
{
Sda=0;
_nop_();
Scl=1;
_nop_();
Scl=0;
_nop_();
}
/------------------------------------------------
非应答IIC总线
------------------------------------------------/
void NoAck(void)
{
Sda=1;
_nop_();
Scl=1;
_nop_();
Scl=0;
_nop_();
}
/------------------------------------------------
发送一个字节
------------------------------------------------/
void Send(unsigned char Data)
{
unsigned char BitCounter=8;
unsigned char temp;
do
{
temp=Data;
Scl=0;
_nop_();
if((temp&0x80)==0x80)
Sda=1;
else
Sda=0;
Scl=1;
temp=Data<<1;
Data=temp;
BitCounter--;
}
while(BitCounter);
Scl=0;
}
/------------------------------------------------
读入一个字节并返回
------------------------------------------------/
unsigned char Read(void)
{
unsigned char temp=0;
unsigned char temp1=0;
unsigned char BitCounter=8;
Sda=1;
do
{
Scl=0;
_nop_();
Scl=1;
_nop_();
if(Sda)
temp=temp|0x01;
else
temp=temp&0xfe;
if(BitCounter-1)
{
temp1=temp<<1;
temp=temp1;
}
BitCounter--;
}
while(BitCounter);
return(temp);
}
/------------------------------------------------
写入DA数模转换值
------------------------------------------------/
void DAC(unsigned char Data)
{
Start();
Send(AddWr); //写入芯片地址
Ack();
Send(0x40); //写入控制位,使能DAC输出
Ack();
Send(Data); //写数据
Ack();
Stop();
}
/------------------------------------------------
读取AD模数转换的值,有返回值
------------------------------------------------/
unsigned char ReadADC(unsigned char Chl)
{
unsigned char Data;
Start(); //写入芯片地址
Send(AddWr);
Ack();
Send(0x40|Chl);//写入选择的通道,本程序只用单端输入,差分部分需要自行添加
//Chl的值分别为0、1、2、3,分别代表1-4通道
Ack();
Start();
Send(AddRd); //读入地址
Ack();
Data=Read(); //读数据
Scl=0;
NoAck();
Stop();
return Data; //返回值
}
/------------------------------------------------
主程序
------------------------------------------------/
void main()
{
unsigned char num; //DA数模输出变量
unsigned char ADtemp; //定义中间变量
Init_Timer1();
while(1)
{
DAC(num); //DA输出,可以用LED模拟电压变化
num++; //累加,到256后溢出变为0,往复循环。显示在LED上亮度逐渐变化
mDelay(20); //延时用于清晰看出变化
if(ADFlag) //定时采集输入模拟量
{
ADFlag=0;
ADtemp=ReadADC(0);
Display[0]=Datatab[(ReadADC(0))/50]|0x80;//处理0通道电压显示
Display[1]=Datatab[((ReadADC(0))%50)/10];
ADtemp=ReadADC(1);
Display[2]=Datatab[((ReadADC(1))/50)]|0x80;//处理1通道电压显示 此通道暂时屏蔽,可以自行添加
Display[3]=Datatab[((ReadADC(1))%50)/10];
ADtemp=ReadADC(2);
Display[4]=Datatab[((ReadADC(2))/50)]|0x80;//处理1通道电压显示 此通道暂时屏蔽,可以自行添加
Display[5]=Datatab[((ReadADC(2))%50)/10];
ADtemp=ReadADC(3);
Display[6]=Datatab[((ReadADC(3))/50)]|0x80;//处理1通道电压显示 此通道暂时屏蔽,可以自行添加
Display[7]=Datatab[((ReadADC(3))%50)/10];
}
}
}
/------------------------------------------------
定时器中断程序
------------------------------------------------/
void Timer1_isr(void) interrupt 3 using 1//定时器1执行数码管动态扫描
{
static unsigned int count,j;
TH1=0xfb; //重新赋值
TL1=0x00;
j++;
if(j==300)
{j=0;ADFlag=1;} //定时置位AD采样标志位
P0=Display[count];//用于动态扫描数码管
P2=count;
count++;
if(count==8) //表示扫描8个数码管
count=0;
}
MIAN: MOV R1,#10
MOV R2,#40H
MOV R0,#7CH
MOVX @R0,A
LOOP: NOP
LOOP1: NOP
JB P32,LOOP1
MOVX A,@R0
MOV @ R2,A
INC R2
MOV R0,#7DH
MOVX A,@R0
MOV @R2,A
INC R2
DJNZ R1,LOOP
不知道你具体想要干什么,只能简单说下:根据你的设计需求选取一个DA芯片(问老师),每个芯片都有配套的手册,上面写了使用方法,包括输入输出管脚,输入输出信号的格式,每个管脚的外围电路怎么连,最重要的提供芯片的内部配置(就是配置DA芯片的寄存器)。自己编写硬件描述语言配准DA芯片,FPGA连接DA。
比如我现在用的CH7301是DVI接口的DA芯片,使用VHDL语言,根据IIC总线协议配置 CH7301的工作方式。 还是比较麻烦的。
以上就是关于51单片机中 执行 DA A程序的时候 如果开始CY值是1 但是DA A的时候没有进位那么CY值是置为0 还是保持1全部的内容,包括:51单片机中 执行 DA A程序的时候 如果开始CY值是1 但是DA A的时候没有进位那么CY值是置为0 还是保持1、求助stc12c5a60s2单片机用软件实现DA的程序! 求高手指导!! 可追加100分财富!、单片机 通过串口控制DA变换器的输出程序等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)