利用单片机和DAC0832控制输出方形波,锯齿波,三角波,并通过示波器显示 程序如下

利用单片机和DAC0832控制输出方形波,锯齿波,三角波,并通过示波器显示 程序如下,第1张

单片机和DAC0832控制输让判出方波,锯齿波,三角波,正弦波。

#include<at89x51.h>

//unsigned char TIME0_H=0xec,TIME0_L=0x78     //定时器0的初值设置;全局变量

#include<sinx.h>

#include<0832.h>

void main()

{

TMOD=0X01

TH0=0xff

TL0=0xd9

IT0=1   //设置中断触发方式,下降沿

EA=1

EX0=1

ET0=1

IP=0X01 //键盘中断级别高

TR0=1

while(1)

{

// square()

}

}

********************************************************************

#ifndef __0832_h__

#define __0832_h__

//#define INPUT XBYTE[0xbfff] //即cs 与xfer 轮流低电平坦敏改。

//#define DACR  XBYTE[0x7fff] //单通道输出,单缓冲就行了。

unsigned char i,sqar_num=128//最大值100,默认值50

unsigned char cho=0        //0:正弦波。1:方波。2:三角波。3:锯齿波。

unsigned char num=0

unsigned char TIME0_H=0xff,TIME0_L=0xd9     //定时器0的初值设置拿告;全局变量.对应正弦波,锯齿波50HZ

sbit chg=   P1^0   //三角波100Hz.

sbit freq_u=P1^1

sbit freq_d=P1^2

sbit duty_u=P1^3

sbit duty_d=P1^4

sbit cs    =P3^7

bit flag=0

unsigned int FREQ=50//初始化频率,50HZ

//调节部分——频率

void freq_ud(void)

{

unsigned int temp

if(freq_d==0)

{  FREQ-- }

else if(freq_u==0)

{ FREQ++ }

if(cho==1|cho==3) //锯齿波256次中断一周期,特殊处理下。否则他的频率是100(+\-)n*2Hz.

{

temp=0xffff-3906/FREQ //方波,三角波默认为100hz,切换后频率也为50HZ 65336-10^6/(256*FREQ)

TIME0_H=temp/256

TIME0_L=temp%256

}

else if(cho==0|cho==3){ //正弦波 三角波默认周期50hz  65536-10^6//(512*FREQ)

temp=0xffff-1953/FREQ

TIME0_H=temp/256

TIME0_L=temp%256

}

}

//调节部分——方波的占空比

void duty_ud(void)  //方波也采用512次中断构成一个周期。

{

if(duty_d==0&sqar_num>0)

sqar_num--

else if(duty_u==0&sqar_num<255)

sqar_num++

}

//波形发生函数

void sint(void)

{

if(!flag)

{

cs=0P2=sin_num[num++]cs=1

if(num==0){num=255flag=1}

}

else if(flag)

{

cs=0P2=sin_num[num--]cs=1

if(num==255){num=0flag=0}

}

}

void square(void)

{

if(i++<sqar_num)  {cs=0P2=0XFFcs=1}

else{cs=0P2=0X00cs=1}

}

void triangle(void)

{

cs=0P2=num++cs=1

}

void stw(void)

{

if(~flag)

{

cs=0P2=num++cs=1

if(num==0){num=255flag=1}

}

else if(flag)

{

cs=0P2=num--cs=1

if(num==255){num=1flag=0}

}

}

//按键中断处理程序

void it0() interrupt 0

{

if(chg==0) { if(++cho==4) {cho=0num=0} } //num=0所有数据从新开始,保证波形的完整性

else if(freq_u==0|freq_d==0)

{freq_ud()}

else if (cho==1&(duty_d==0|duty_u==0))

{duty_ud()}

else

}

//定时器中断处理程序。

void intt0() interrupt 1

{

//TH0=0x00TL0=0x00sinx()

switch(cho)

{

case 0:{TH0=TIME0_HTL0=TIME0_Lsint()    break}  //正弦波//每半周期256取样。

case 1:{TH0=TIME0_HTL0=TIME0_Lsquare() break} //方波   //为了提高方波的最高频率,只有牺牲占空比的最小可调值。分100份 每次1%。

case 2:{TH0=TIME0_HTL0=TIME0_Ltriangle()break} //三角波

case 3:{TH0=TIME0_HTL0=TIME0_Lstw()    break} //锯齿波

default:

}

}

#endif

*****************************************

//正弦表;每半个周期256个取值,最大限度保证波形不失真。

//各个值通过MATLAB算出,并四设五如取整。具体程序如下

#ifndef __sinx_h__

#define __sinx_h__

unsigned char code sin_num[]={  

0,0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 2, 2, 2,

2, 3, 3, 4, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, 9, 9,

10, 10, 11, 12, 12, 13, 14, 15, 15, 16, 17, 18, 18, 19, 20, 21,

22, 23, 24, 25, 25, 26, 27, 28, 29, 30, 31, 32, 34, 35, 36, 37,

38, 39, 40, 41, 42, 44, 45, 46, 47, 49, 50, 51, 52, 54, 55, 56,

57, 59, 60, 61, 63, 64, 66, 67, 68, 70, 71, 73, 74, 75, 77, 78,

80, 81, 83, 84, 86, 87, 89, 90, 92, 93, 95, 96, 98, 99, 101,102,

104, 106, 107, 109, 110, 112,    113, 115, 116, 118, 120, 121, 123, 124, 126, 128,

129, 131,    132, 134, 135,    137,    139, 140, 142, 143, 145, 146, 148,    149, 151, 153,

154, 156, 157, 159, 160, 162, 163, 165,    166, 168, 169, 171, 172, 174, 175, 177,

178, 180, 181,    182, 184, 185, 187, 188, 189, 191, 192, 194, 195, 196, 198, 199,

200, 201, 203, 204, 205, 206, 208, 209, 210, 211, 213, 214, 215, 216, 217, 218,

219, 220, 221, 223, 224, 225, 226, 227, 228, 229, 230, 230, 231, 232, 233, 234,

235, 236, 237, 237, 238, 239, 240, 240, 241, 242, 243, 243, 244, 245, 245, 246,

246, 247, 247, 248, 248, 249, 249, 250, 250, 251, 251,    251, 252, 252, 253, 253,

253, 253, 254, 254, 254, 254, 254, 255, 255, 255, 255,    255, 255, 255, 255, 255

}

#endif

//MATLAB程序:

//x=linspace(-pi/2,pi/2,255)%如果过采用1位采用,很多值是重的。虽然实际中并不会。

//y=(sin(x)+1)/2.0*255

//%uint32(y)%强制类型转换。

//%fprintf('%.f\n',uint32(y))%控制输出类型

//round(y)%四舍五入函数

#include <reg52.h>

#include <橘搏intrins.h>

#include <math.h>

//------------定义接口-------------//

sbit RS=P2^5

sbit RW=P2^6

sbit E=P2^7

sbit RES=P2^1

#define Lcd_Bus P1

// P1 接 LCM

#define uchar unsigned char

#define FIRST_ADDR 0

//定义字符/汉字显示起始位置

/*------------------检查忙位-----------------------------*/

void chk_busy()

{

RS=0

RW=1

E=1

Lcd_Bus=0xff

while((Lcd_Bus&0x80)==0x80)

E=0

}

/*------------------延时子程序-----------------------------*/

void delay(unsigned int t)

{

unsigned int i,j

for(i=0i<ti++)

for(j=0j<10j++)

}

/*------------------写命令到LCD------------------------------*/

void write_com(unsigned char cmdcode)

{

chk_busy()

RS=0

RW=0

E=1

Lcd_Bus=cmdcode

delay(5)

//------------------在数据写入的时候加入适当的延时

E=0

delay(5)

}

/*-------------------写数据到LCD----------------------------*/

void write_data(unsigned char Dispdata)

{

chk_busy()

RS=1

RW=0

E=1

Lcd_Bus=Dispdata

delay(5)

//------------------在数据写入的时候加入适当的延时

E=0

delay(5)

}

/*------------------初始化LCD屏--------------------------*/

void lcdreset()

{

delay(2000)

write_com(0x30)

delay(10)

//选择基本指令集

write_com(0x30)

//选择8bit数据流

delay(5)

write_com(0x0c)

//开显示(无游标、不反白)

delay(10)

write_com(0x01)

//清除显示,并且设定唯缺地址指针为00H

delay(500)

write_com(0x06)

//指定在资料的读取及写入时,设定游标的移动方向及指定显示的移位

delay(0)

}

/*------------------显示字符串--------------------------*/

void hzkdis(unsigned char code*s)

{

while(*s>0)

{

write_data(*s)

s++

delay(50)

}

}

/*------------------首屏显示--------------------------*/

void ceshi()

{

write_com(0x01)

//清除显示,并且设定地址指针为00H

delay(5)

write_com(0x80)

//第一行(如果是地址是:80H,即LCD的第一行的第一个位置显示)

hzkdis("中国计量学院")

write_com(0x90)

//第二行(如果是地址是:90H,即LCD的第二行的第一个位置显指伍辩示)

hzkdis("光电学院")

write_com(0x88)

//第三行(如果是地址是:88H,即LCD的第二行的第一个位置显示)

hzkdis("06光信2")

write_com(0x98)

//第四行(如果是地址是:98H,即LCD的第二行的第一个位置显示)

hzkdis("测试程序")

}

//------------------清整个GDRAM空间----------------------------

void clrgdram()

{

unsigned char x,y

for(y=0y<64y++)

for(x=0x<16x++)

{

write_com(0x34)

write_com(y+0x80)

//行地址

write_com(x+0x80)

//列地址

write_com(0x30)

write_data(0x00)

write_data(0x00)

}

}

//------------------------------------------------------------

void clrscreen()

{

write_com(0x01)

delay(10)

}

unsigned char ReadByte(void)

{

unsigned char byReturnValue

chk_busy()

Lcd_Bus=0xff

RS=1

RW=1

E=0

E=1

byReturnValue=Lcd_Bus

E=0

return byReturnValue

}

//增加画点子程序

void DrawPoint(unsigned char X,unsigned char Y,unsigned char Color)

{

unsigned char Row,Tier,Tier_bit

unsigned char ReadOldH,ReadOldL

write_com(0x34)

write_com(0x36)

Tier=X>>4

Tier_bit=X&0x0f

if(Y<32)

{

Row=Y

}

else

{

Row=Y-32

Tier+=8

}

write_com(Row+0x80)

write_com(Tier+0x80)

ReadByte()

ReadOldH=ReadByte()

ReadOldL=ReadByte()

write_com(Row+0x80)

write_com(Tier+0x80)

if(Tier_bit<8)

{

switch(Color)

{

case 0 :

ReadOldH&=(~(0x01<<(7-Tier_bit)))

break

case 1 :

ReadOldH|=(0x01<<(7-Tier_bit))

break

case 2 :

ReadOldH^=(0x01<<(7-Tier_bit))

break

default :

break

}

write_data(ReadOldH)

write_data(ReadOldL)

}

else

{

switch(Color)

{

case 0 :

ReadOldL&=(~(0x01<<(15-Tier_bit)))

break

case 1 :

ReadOldL|=(0x01<<(15-Tier_bit))

break

case 2 :

ReadOldL^=(0x01<<(15-Tier_bit))

break

default :

break

}

write_data(ReadOldH)

write_data(ReadOldL)

}

write_com(0x30)

}

void main(void)

{

uchar i,j,colour=1

// uint ADzhi

RW=0

lcdreset()

// ceshi()

clrgdram()

// delay(5000)

clrscreen()

for(i=4i<124i++)

{

j=36-28*sin((i-8)*3.14)

//正弦波显示

DrawPoint(i,j,colour)

}

delay(5000)

while(1)

}


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

原文地址: http://outofmemory.cn/yw/12258254.html

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

发表评论

登录后才能评论

评论列表(0条)

保存