|
- #include<reg52.h>
- #include<intrins.h>
- #define uint unsigned int
- #define uchar unsigned char
- #define AddWr 0x90 //PCF8591地址1001 0000
- sbit SCL=P3^6;
- sbit SDA=P3^7;
- sbit RS=P2^0;
- sbit RW=P2^1;
- sbit E =P2^2;
- sbit sin=P1^0;
- sbit square=P1^1;
- sbit triangle=P1^2;
- sbit juchi=P1^3;
- sbit key1=P1^4;
- sbit key2=P1^5;
- sbit key3=P1^6;
- sbit key4=P1^7;
- bit ack;
- uchar pinlv,shi,ge; //頻率變量//
- uchar change; //波形切換變量//
- uchar num; //波形表輸出個數(shù)變量//
- uint TH0_PL,TL0_PL; //重設(shè)初值//
- uint timer_value; //初值變量//
- uchar fd;
- uchar code table[]="0123456789";
- void delay_ms(unsigned int n) //12MHZ晶振時延時1ms,若用11.0592MHZ,則j=110
- {
- unsigned int i=0,j=0;
- for(i=n;i>0;i--)
- for(j=0;j<123;j++);
- }
- void delay_8591()
- {
- _nop_(); _nop_(); _nop_(); _nop_(); //起始條件建立時間大于4.7us,延時
- }
- uchar code sin_table[]={0x7F,0x98,0xB0,0xC6,0xD9,0xE9,0xF5,0xFC,
- 0xFE,0xFC,0xF5,0xE9,0xD9,0xC6,0xB0,0x98,
- 0x7F,0x66,0x4E,0x38,0x25,0x15,0x09,0x02,
- 0x00,0x02,0x09,0x15,0x25,0x38,0x4E,0x66};
- uchar code square_table[]={0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
- 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
- 0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,
- 0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff};
- uchar code triangle_table[]={0x00,0x10,0x20,0x30,0x40,0x50,0x60,0x70,
- 0x80,0x90,0xa0,0xb0,0xc0,0xd0,0xe0,0xf0,
- 0xff,0xf0,0xe0,0xd0,0xc0,0xb0,0xa0,0x90,
- 0x80,0x70,0x60,0x50,0x40,0x30,0x20,0x10};
- uchar code juchi_table[]={0x00,0x07,0x10,0x17,0x20,0x27,0x30,0x37,
- 0x40,0x47,0x50,0x57,0x60,0x67,0x70,0x77,
- 0x80,0x87,0x90,0x97,0xa0,0xa7,0xb0,0xb7,
- 0xc0,0xc7,0xd0,0xd7,0xe0,0xe7,0xf0,0x00};
- /*******************************************PCF8591DA轉(zhuǎn)換模塊**************************/////////////////////
- /****************************************************************************************///////////
- void start()//開始信號
- {
- SDA=1; //確保SDA線為高電平,開始 之前都確保為高電平
- delay_8591();
- SCL=1; //確保SCL為高電平
- delay_8591();
- SDA=0; //在SCL為高時拉低SDA線,即為起始信號
- delay_8591();
- SCL=0; //將SCL線拉低,為后面SDA的傳數(shù)據(jù)時電平變化做準(zhǔn)備
- }
- void stop() //停止信號
- {
- SDA=0; //確保SDA線為低電平
- delay_8591();
- SCL=1; //確保SCL線為高電平
- delay_8591();
- SDA=1; //在SCL為高時拉高SDA線,即為停止信號,此時SDA和SCL均為高,同時完成IIC初始化過程
- delay_8591();
- }
- void ack_check(bit a)
- {
- if(a==0)
- SDA=0; //*在此發(fā)出應(yīng)答或非應(yīng)答信號 *//
- else
- SDA=1;
- _nop_();
- SCL=1;
- delay_8591();
- SCL=0;
- _nop_();
- _nop_(); //*清時鐘線,鉗住I2C總線以便繼續(xù)接收*//
- SDA=1;
- }
- void send_byte(uchar date)
- {
- uchar i;
- for(i=0;i<8;i++) //*要傳送的數(shù)據(jù)長度為8位*//
- {
- if((date<<i)&0x80)
- SDA=1; //*判斷發(fā)送位*//
- else
- SDA=0;
- _nop_();
- SCL=1; //*置時鐘線為高,通知被控器開始接收數(shù)據(jù)位*/
- delay_8591();
- SCL=0;
- }
- SDA=1; //*8位發(fā)送完后釋放數(shù)據(jù)線,準(zhǔn)備接收應(yīng)答位*/
- _nop_();
- SCL=1;
- if(SDA==1)
- ack=0;
- else ack=1; //*判斷是否接收到應(yīng)答信號*/
- SCL=0;
-
- }
- void write_add()
- {
- start();
- send_byte(AddWr);
- if(ack==0) //判斷有無應(yīng)答信號
- { //有應(yīng)答信號應(yīng)該給低電平
- stop();
- return;
- }
- send_byte(0x40); //01000000 寫入控制位,使能DA輸出
- }
- /*********************************************************顯示模塊*///////////////////////////////////////////////
- /**********************************************************************************************//////////
- void LCD12864_BusyCheck() //讀一下當(dāng)前代碼是否在忙,最高位為0為不忙,1為忙
- {
- unsigned char temp;
- P0=0xff; //釋放整個口,全部為高電平
- RS=0; //RS為0是讀狀態(tài) ,為1是寫數(shù)據(jù)
- RW=1; //高讀低寫
- delay_ms(10);
- while(1) //一直檢測知道不忙為止
- {
- E=1; //E是使能口,由1到0才可寫入數(shù)據(jù)
- temp=P0;
- E=0;
- if((temp&0x80)==0) //如果等于零意味著最高位為0,不忙
- break; //結(jié)束(如果上面代碼不為0,則循環(huán)一直檢測
- }
- }
- void LCD12864_wcmd(unsigned char cmd) //命令函數(shù)
- {
- LCD12864_BusyCheck();
- RS=0; //高數(shù)低命
- RW=0;
- delay_ms(1);
- P0=cmd; //命令把想寫的內(nèi)容送到數(shù)據(jù)口
- delay_ms(1);
- E=1;
- E=0; //產(chǎn)生一個下降沿寫數(shù)據(jù)
- }
- void LCD12864_wdat(unsigned char dat) //寫數(shù)據(jù)函數(shù)
- {
- LCD12864_BusyCheck();
- RS=1; //寫數(shù)據(jù)
- RW=0;
- delay_ms(1);
- P0=dat; //把想寫的內(nèi)容送到數(shù)據(jù)口
- E=1;
- delay_ms(1);
- E=0;
- }
- void LCD12864_SetPos(unsigned char x,unsigned char y) //x是行,y是列
- {
- unsigned char pos;
- switch(x)
- {
- case 0:x=0x80;break; //第一行從0x80開始
- case 1:x=0x90;break;
- case 2:x=0x88;break;
- case 3:x=0x98;break;
- default:break;
- }
- pos=x+y; //x范圍:0-3,y范圍:0-7 (因?yàn)橐恍兄荒軐懓藗漢字)
- LCD12864_wcmd(pos);
- }
- void LCD12864_DisHZ_Str(unsigned char*s)
- {
- while(*s)
- LCD12864_wdat(*s++); //指針++,指針指向下一個
- }
- void LCD12864_init()
- {
- LCD12864_wcmd(0x01); //清屏指令
- delay_ms(3);
- LCD12864_wcmd(0x30);
- delay_ms(5);
- LCD12864_wcmd(0x30);
- delay_ms(5);
- LCD12864_wcmd(0x0c);
- delay_ms(3);
- LCD12864_wcmd(0x01);
- delay_ms(3);
- LCD12864_wcmd(0x80);
- delay_ms(3);
- LCD12864_wcmd(0x06); //游標(biāo)右移,地址自動加一
- delay_ms(3);
- }
- //*******定時器程序*********//
- //*定時次數(shù)=(1/pinlv*32)/(1/12/12)=31250/pinlv*// 更改頻率原理:單片機(jī)12MHZ晶振,所以產(chǎn)生1s的時鐘,也就是1000000us,而輸出波形表內(nèi)有32個數(shù)據(jù),
- // 所以用1000000/(頻率*32),得出數(shù)值作為定時器初值,每隔一個時間輸出一個數(shù)值。
- void timer0_init()
- {
- uchar a; //頻率儲存
- shi=pinlv/10;
- ge=pinlv%10;
- a=shi*10+ge;
- timer_value=31250/a; //定時次數(shù)計算
- timer_value=timer_value-28; //誤差補(bǔ)償
- TMOD=0x01;
- TH0_PL=(65535-timer_value)/256;
- TL0_PL=(65535-timer_value)%256;
- TH0=TH0_PL;
- TL0=TL0_PL;
- TF0=0;
- EA=1;
- ET0=1;
- TR0=1;
- }
- /***************************************按鍵模塊********************************************/
- /******************************************************************************************/
- void key()
- {
- if(key1==0)//按下鍵盤
- {
- delay_ms(10); //消抖動
- if(key1==0);//在確認(rèn)一次
- { pinlv++;//發(fā)光二極管點(diǎn)亮
- if(pinlv==51) //*設(shè)置頻率調(diào)節(jié)范圍1~50
- pinlv=1;
- }
- while(!key1)//松手
- delay_ms(10); ////消抖動
- while(!key1); }
- if(key2==0)//按下鍵盤
- {
- delay_ms(10); //消抖動
- if(key2==0);//在確認(rèn)一次
- {
- pinlv--;//發(fā)光二極管點(diǎn)亮
- if(pinlv==1) //*設(shè)置頻率調(diào)節(jié)范圍1~50
- pinlv=50;
- }
- while(!key2)//松手
- delay_ms(10); ////消抖動
- while(!key2); }
- if(key3==0)//按下鍵盤
- {
- delay_ms(10); //消抖動
- if(key3==0);//在確認(rèn)一次
- {
- fd++;
- if(fd==11) //*設(shè)置幅度調(diào)節(jié)范圍1~10
- fd=1;
- }
- while(!key3)//松手
- delay_ms(10); ////消抖動
- while(!key3); }
- if(key4==0)//按下鍵盤
- {
- delay_ms(10); //消抖動
- if(key4==0);//在確認(rèn)一次
- {
- fd--;
- if(fd==1)
- fd=10;
- }
- while(!key4)//松手
- delay_ms(10); ////消抖動
- while(!key4); }
- }
- void wave_KEY()
- {
- if(square==0)//按下鍵盤
- {
- delay_ms(10); //消抖動
- if(square==0);//在確認(rèn)一次
- {
- change=2;
- }
- while(square)//松手
- delay_ms(10); ////消抖動
- while(square); }
- else if(sin==0)//按下鍵盤
- {
- delay_ms(10); //消抖動
- if(sin==0);//在確認(rèn)一次
- {
- change=1;
- }
- while(sin)//松手
- delay_ms(10); ////消抖動
- while(sin); }
- else if(triangle==0)//按下鍵盤
- {
- delay_ms(10); //消抖動
- if(triangle==0);//在確認(rèn)一次
- {
- change=3;
- }
- while(triangle)//松手
- delay_ms(10); ////消抖動
- while(triangle); }
- else if(juchi==0)//按下鍵盤
- {
- delay_ms(10); //消抖動
- if(juchi==0);//在確認(rèn)一次
- {
- change=4;
- }
- while(juchi)//松手
- delay_ms(10); ////消抖動
- while(juchi); }
- }
- void display() //顯示子程序
- {
- LCD12864_init();
- LCD12864_wcmd(0x0c); //開顯示
- LCD12864_wcmd(0x30);
- LCD12864_SetPos(0,0); //在第一行第一個漢字輸出
- delay_ms(1);
- LCD12864_DisHZ_Str(" 波形發(fā)生器");
- LCD12864_SetPos(1,0); //第二行
- delay_ms(1);
- LCD12864_DisHZ_Str("波形:");
-
- LCD12864_SetPos(2,0); //第三行
- delay_ms(1);
- LCD12864_DisHZ_Str("頻率:");
-
- LCD12864_SetPos(2,7);
- LCD12864_DisHZ_Str("HZ");
- LCD12864_SetPos(3,0); //第四行
- delay_ms(1);
- LCD12864_DisHZ_Str("幅度:");
-
- LCD12864_SetPos(3,7);
- LCD12864_DisHZ_Str("V");
- }
- void state_init() //*初始波形數(shù)據(jù)設(shè)置(正弦波,頻率5hz,幅度5v*//
- {
- pinlv=5;
- change=1;
- fd=10;
- write_add();
- }
- /****************************************主程序*********************************//////
- /************************************************************************///////////////
- void main()
- {
- state_init();
- P0=0xff;
- display();
- while(1)
- {
- if(change==1) // 此處數(shù)據(jù)是隨按鍵改變的數(shù)據(jù)程序,因此放入while循環(huán)中,隨波形轉(zhuǎn)換而改變顯示
- {
- LCD12864_SetPos(1,4);
- LCD12864_DisHZ_Str("sin");
- }
- if(change==2)
- {
- LCD12864_SetPos(1,4);
- LCD12864_DisHZ_Str("方波");
- }
- if(change==3) {
- LCD12864_SetPos(1,4);
- LCD12864_DisHZ_Str("角波");
- }
- if(change==4) {
- LCD12864_SetPos(1,4);
- LCD12864_DisHZ_Str("鋸齒");
- }
- LCD12864_SetPos(2,4);
- shi=pinlv/10;
- ge=pinlv%10;
- if(shi==0)
- LCD12864_wdat(' ');
- else
- LCD12864_wdat(table[shi]);
- if(shi==0&&ge==0)
- LCD12864_wdat(' ');
- else
- LCD12864_wdat(table[ge]);
- LCD12864_SetPos(3,4);
- LCD12864_wdat(table[fd*5/10]);
- LCD12864_DisHZ_Str(".");
- LCD12864_wdat(table[fd*5%10]);
- wave_KEY();
- key();
- delay_ms(10);
- timer0_init();
- delay_ms(10);
- }
- }
- void timer0() interrupt 1 //改變頻率
- {
- TH0=TH0_PL;
- TL0=TL0_PL;
- if(change==1)
- send_byte(sin_table[num]*0.1*fd); //單片機(jī)輸出為5V,因此FD變化1~10,在乘以0.1后,可編程0~5的調(diào)節(jié)
- else if(change==2)
- send_byte(square_table[num]*0.1*fd); //假如不加幅度程序,則輸出頻率可靈活調(diào)制1~40hz!!,即去掉*0.1*fd
- else if(change==3)
- send_byte(triangle_table[num]*0.1*fd);
- else if(change==4)
- send_byte(juchi_table[num]*0.1*fd);
- if(num>=31)
- num=0;
- else
- num++;
- }
復(fù)制代碼 |
|