单片机P0,P1口读取16位数字量,然后DA输出,请问这样的程序有问题么

单片机P0,P1口读取16位数字量,然后DA输出,请问这样的程序有问题么,第1张

这样用可真够浪费的呵呵,一个是浪费PLC输出模块,一个是你还要在调试单片机程序。如果PLC的DA模块分辨率够用的话还是建议使用PLC的DA控制,稳定性也好。如果是因为PLC的DA分辨率不够或是成本问题(其实这样成本也比配个DA模块省不了多少了,估计是分辨率问题)

建议你的DA选并行的,通过个光耦和PLC的输出点直接相连,省事多了,如果没有并行DA其实自己搭个也是不错的方法~

256点正弦波,用8位即可,用不着12位数。

非得用 12 位数,就把最低四位(或最高四位),都置为零,即可。

程序,可以参考:

>

FPGA的精髓就是时序控制啊,但这也是它的难点之一,要是不掌握好时序,那就不要想好好控制外围了。

FPGA不像单片机这类控制器,因此时序对它来说是非常重要的(当然,对单片机也重要),但是由于结构和执行机理的不同,注定这两类控制器有着本质的区别。

至于你说的为什么要看时序,任何DA模块都有它的工作过程,要是不按它规定的工程控制它,它肯定就不能正常工作啊,这里的工作就是指它的时序了。

由于FPGA和单片机的不同,所以在使用FPGA时,一再强调要严格按照时序来控制。

不知道你具体想要干什么,只能简单说下:根据你的设计需求选取一个DA芯片(问老师),每个芯片都有配套的手册,上面写了使用方法,包括输入输出管脚,输入输出信号的格式,每个管脚的外围电路怎么连,最重要的提供芯片的内部配置(就是配置DA芯片的寄存器)。自己编写硬件描述语言配准DA芯片,FPGA连接DA。

比如我现在用的CH7301是DVI接口的DA芯片,使用VHDL语言,根据IIC总线协议配置 CH7301的工作方式。 还是比较麻烦的。

统计DA_WORD数据区中正数、0、负数的个数,数据定义如下:

DA_WORD DW -1,3,5,0,-5,-7,4,0,-8……

COUNT EQU $-DA_WORD

NUM DB 0 ;存放正数的个数

DB 0 ;存放0的个数

DB 0 ;存放负数的个数

DATAS SEGMENT

ORG 1000H

DA_WORD DW 0FFFFH,3,5,0,0FFFBH,0FFF9H,4,0,0FFF8H

DOUNT EQU $-DA_WORD

ORG 1020H

NUM DB 0 ;存放正数的个数

ZE DB 0 ;存放0的个数

PLUS DB 0

DATAS ENDS

STACKS SEGMENT

DW 100 DUP(0)

STACKS ENDS

CODES SEGMENT

ASSUME CS:CODES,DS:DATAS,SS:STACKS

START:

MOV AX,DATAS

MOV DS,AX

MOV AX,DOUNT

MOV CL,1

SHR AX,CL

MOV CX,AX

MOV SI,OFFSET DA_WORD

AG: MOV AX,[SI]

CMP AX,0

JNZ NOZ

INC [ZE]

JMP NEXT

NOZ:TEST AX,8000H

JNZ ISPLUS

INC [NUM]

JMP NEXT

ISPLUS:

INC [PLUS]

NEXT:

ADD SI,2

LOOP AG

INT 3

MOV AH,4CH

INT 21H

CODES ENDS

END START

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

/

文件名: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);

}

}

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

以上就是关于单片机P0,P1口读取16位数字量,然后DA输出,请问这样的程序有问题么全部的内容,包括:单片机P0,P1口读取16位数字量,然后DA输出,请问这样的程序有问题么、单片机产生12位DA,256点正弦波程序、如何写fpga调试da模块的程序时为什么要看时序等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

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

原文地址: http://outofmemory.cn/zz/9356836.html

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

发表评论

登录后才能评论

评论列表(0条)

保存