锯齿波生成原理:PCF8591中所存数据不断增大,直至到达最值(十进制255),随后令PCF8591中所存数据变为0,从而到达锯齿波的效果
实现:输出峰峰值为0-3V的爬升锯齿波
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
单片机源程序如下:
- #include <reg52.h>
- #include <intrins.h> //包含函数_nop_()定义的头文件
- #define AddWr 0x90 //PCF8591写地址
- #define u8 unsigned char
- #define u16 unsigned int
- u8 code juchi[64]=
- {
- 0,4,8,12,16,20,24,28,32,36,40,45,49,53,57,61,65,69,73,77,81,85,89,93,97,101,105,109,113,117,121,125,130,134,138,142,
- 146,150,154,158,162,166,170,174,178,182,186,190,194,198,202,206,210,215,219,223,227,231,235,239,243,247,251,255
- }; //锯齿波取码
- sbit Sda=P1^2; //定义IIC总线引脚
- sbit Scl=P1^1;
- u16 num;
- void Start(void)
- {
- Sda=1;
- _nop_();
- Scl=1;
- _nop_();
- Sda=0; //sda负跳变
- _nop_();
- Scl=0;
- }
- void Stop(void)
- {
- Sda=0;
- _nop_();
- Scl=1;
- _nop_();
- Sda=1; //sda正跳变
- _nop_();
- Scl=0;
- }
-
- void Send(u8 Data)
- {
- u8 BitCounter=8; //定义发送位数为8
- do
- {
- Scl=0;
- _nop_();
- if((Data&0x80)==0x80) //逐位判别1/0后送至SDA
- Sda=1; //scl低电平期间数据送至sda线
- else
- Sda=0;
- Scl=1; //scl变高输出数据(写入pcf8591)
- Data=Data<<1;
- BitCounter--;
- }while(BitCounter);
- Scl=0;
- }
-
- void Ack(void)
- {
- Sda=0; //scl高电平期间sda输出低电平
- _nop_();
- Scl=1;
- _nop_();
- Scl=0;
- _nop_();
- }
-
- void DACconversion(u8 sla,u8 c, u8 Val) //8591输出一个数据
- {
- Start();//启动总线
- Send(sla);//发送器件地址
- Ack();
- Send(c);//发送控制字节
- Ack();
- Send(Val);//发送DAC的数值
- Ack();
- Stop();//结束总线
- }
-
- void main()
- {
- while(1)
- {
- for(num=0;num<64;num++)
- {
- DACconversion(AddWr,0x40,juchi[num]);
- }
- if(num==64)
- {
- num=0;
- }
- }
- }
复制代码
Keil代码与Proteus8.13版本的仿真下载(注意不能兼容其他版本 只能用8.13打开):
DA锯齿波发生器.7z
(32.08 KB, 下载次数: 22)
|