单片机和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)
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)