单片机汇编程序实现DA转换,用DAC0832芯片实现

单片机汇编程序实现DA转换,用DAC0832芯片实现,第1张

//看看这个程序,可以参考一下。

//0832各种波型输出演示 //P11P10 = 00时 正弦波 //P11P10 = 01时 锯齿波 //P11P10 = 10时 方波 //P11P10 = 11时 三角波 //需要连接的导线://(1)译码器Y0接DAC0832的CS //(2)P36即XWR插孔接DAC0832的WR //(3)8芯的排线连接8排针的P0口和DAC0832的D7-D0 //(4)串并转换区的+5v接DAC0832的参考电压Vref插孔#include <reg51h> #include <ABSACCH>//absacch是C51中绝对地址访问函数的头文件 #define daport XBYTE[0x8000]//将daport定义为外部I/O口,地址8000H,则74ls138的Y0有效 sbit P1_1=P1^1; sbit P1_0=P1^0; unsigned char i,j,k; bit updown; //三角波的上升/下降 unsigned char code sinn[64]={ 198,204,210,216,222,228,233,237,242,245,249,251,253,255,255,255, 255,254,252,250,247,243,239,235,230,224,219,213,207,201,194,188, 181,175,169,163,158,152,147,143,139,136,133,131,129,128,128,128, 129,130,132,135,138,142,146,151,156,162,168,174,180,186,193,199 }; //正弦的数值在128-255之间,所以输出的正弦幅值在0-5v void main(void) { updown=0; i=128; j=0; while(1) { if (P1_1==1) { if (P1_0==0) //输出占空比为25%的方波 { if (j<=40) i=255;//i=255时输出的电压为+5v else if(j<=160) i=0;//i=0时输出的电压为-5v else j=0; } else //输出-5v—+5v的三角波 { if (updown==0) //上升 { if (i==255) updown=1; else i=i+5;//i++; } else //下降 { if (i==0) updown=0;//if(i==128) updown=0;i=128输出的电压为0 else i=i-5;//i--; } } } else{ if (P1_0==0) //正弦波 { i=sinn[j]; if (j>=63) j=0; } else //锯齿波 { if (i==255) i=128; else i++; } } daport=i; for (k=0;k<2;k++); j++; }}

可变增益说的简单就是调幅。

一般可用乘法DAC,双DA的方案,一个DA的输出作为另一个DA的参考电压,当参考电压变化了,那么最终输出的DA幅度也就变化了。

另外可变固定增益一般用运放做些特定放大倍数的,比如2倍,5倍,10倍等等,然后串联起来,把不用的倍数用继电器短接起来,就可以实现一些比如 1,2,5,10,20,50,100这些倍数关系

是十进制调整指令,用来在进行十进制运算(BCD码)时进行校正的。

单片机将各功能部件集成在一块晶体芯片上,集成度很高,体积自然也是最小的。芯片本身是按工业测控环境要求设计的,内部布线很短,其抗工业噪音性能优于一般通用的CPU。单片机程序指令,常数及表格等固化在ROM中不易破坏,许多信号通道均在一个芯片内,故可靠性高。

扩展资料:

注意事项:

时钟监控有效与省电指令STOP是一对矛盾。只能使用其中之一。 看门狗技术是监测应用程序中的一段定时中断低电压复位技术是监测单片机电源电压,当电压低于某一值时产生复位信号。

由于单片机技术的发展,单片机本身对电源电压范围的要求越来越宽。电源电压从当初的5V降至33V并继续下降到27V、22V、18V。在是否使用低电压复位功能时应根据具体应用情况权衡一下。

参考资料来源:百度百科-单片机

不知道你的程序是什么问题,我就把你程序main中第3行的int ;删掉后,在f、g和h输入的时候输入90 30 30就可以得到da的值为905083,不知道你为什么不可以

另外如果采用double型则scanf输入时用%lf,否则数据无法正确输入

这是一个例程,去掉那些通信显示的,很小。而且想做单片机,就要先看开发文档,看完了就会发现编程序特别简单,而且开发文档里会提供最基本的程序例子的

/

文件名: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转换,用DAC0832芯片实现全部的内容,包括:单片机汇编程序实现DA转换,用DAC0832芯片实现、怎么用51单片机控制DA实现可变增益放大啊急求具体方法与程序啊!!!、求教,单片机中的DA指令是指啥啊等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/zz/9767199.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-05-01
下一篇2023-05-01

发表评论

登录后才能评论

评论列表(0条)

    保存