本設計程序代碼實現室內CO濃度值的在線監測,并可實現報警設定功能。可以互相學習 。,
單片機源程序如下:
- #include <reg52.h> //調用單片機頭文件
- #define uchar unsigned char //無符號字符型 宏定義 變量范圍0~255
- #define uint unsigned int //無符號整型 宏定義 變量范圍0~65535
- #include <intrins.h>
- //#include "lcd1602.h"
- sbit CS=P1^3; //CS定義為P2口的第4位腳,連接ADC0832CS腳
- sbit SCL=P1^0; //SCL定義為P2口的第3位腳,連接ADC0832SCL腳
- sbit DO=P1^1; //DO定義為P2口的第4位腳,連接ADC0832DO腳
- sbit beep = P2^0; //蜂鳴器IO口定義
- long dengji,s_dengji = 50; //濃度等級
- bit flag_300ms ;
- uchar key_can; //按鍵值的變量
- uchar menu_1; //菜單設計的變量
- uchar flag_clock;
- //這三個引腳參考資料
- sbit rs=P1^2; //1602數據/命令選擇引腳 H:數據 L:命令
- //sbit rw=P2^6; //1602讀寫引腳 H:數據寄存器 L:指令寄存器
- sbit e =P1^4; //1602使能引腳 下降沿觸發
- uchar code table_num[]="0123456789abcdefg";
- /********************************************************************
- * 名稱 : delay_uint()
- * 功能 : 小延時。
- * 輸入 : 無
- * 輸出 : 無
- ***********************************************************************/
- void delay_uint(uint q)
- {
- while(q--);
- }
- /********************************************************************
- * 名稱 : write_com(uchar com)
- * 功能 : 1602命令函數
- * 輸入 : 輸入的命令值
- * 輸出 : 無
- ***********************************************************************/
- void write_com(uchar com)
- {
- e=0;
- rs=0;
- // rw=0;
- P0=com;
- delay_uint(3);
- e=1;
- delay_uint(25);
- e=0;
- }
- /********************************************************************
- * 名稱 : write_data(uchar dat)
- * 功能 : 1602寫數據函數
- * 輸入 : 需要寫入1602的數據
- * 輸出 : 無
- ***********************************************************************/
- void write_data(uchar dat)
- {
- e=0;
- rs=1;
- // rw=0;
- P0=dat;
- delay_uint(3);
- e=1;
- delay_uint(25);
- e=0;
- }
- /********************************************************************
- * 名稱 : write_sfm2(uchar hang,uchar add,uchar date)
- * 功能 : 顯示2位十進制數,如果要讓第一行,第五個字符開始顯示"23" ,調用該函數如下
- write_sfm1(1,5,23)
- * 輸入 : 行,列,需要輸入1602的數據
- * 輸出 : 無
- ***********************************************************************/
- void write_sfm2(uchar hang,uchar add,uint date)
- {
- if(hang==1)
- write_com(0x80+add);
- else
- write_com(0x80+0x40+add);
- if(date >= 100)
- {
- write_data(0x30+date/100%10);
- write_data(0x30+date/10%10);
- }
- else
- {
- write_data(' ');
- write_data(0x30+date/10%10);
- }
- write_data(0x30+date%10);
- }
- /********************************************************************
- * 名稱 : write_string(uchar hang,uchar add,uchar *p)
- * 功能 : 改變液晶中某位的值,如果要讓第一行,第五個字符開始顯示"ab cd ef" ,調用該函數如下
- write_string(1,5,"ab cd ef;")
- * 輸入 : 行,列,需要輸入1602的數據
- * 輸出 : 無
- ***********************************************************************/
- void write_string(uchar hang,uchar add,uchar *p)
- {
- if(hang==1)
- write_com(0x80+add);
- else
- write_com(0x80+0x40+add);
- while(1)
- {
- if(*p == '\0') break;
- write_data(*p);
- p++;
- }
- }
- /********************************************************************
- * 名稱 : clear_1602()
- * 功能 : 清除1602顯示
- * 輸入 : 無
- * 輸出 : 無
- ***********************************************************************/
- void clear_1602()
- {
- write_string(1,0," ");
- write_string(2,0," ");
- }
- /***********************lcd1602上顯示特定的字符************************/
- void write_zifu(uchar hang,uchar add,uchar date)
- {
- if(hang==1)
- write_com(0x80+add);
- else
- write_com(0x80+0x40+add);
- write_data(date);
- }
- /********************************************************************
- * 名稱 : init_1602()
- * 功能 : 初始化1602液晶
- * 輸入 : 無
- * 輸出 : 無
- ***********************************************************************/
- void init_1602()
- {
- write_com(0x38);
- write_com(0x0c);
- write_com(0x06);
- write_string(1,0," thickne: 00 ");
- write_string(2,0," Thresho: 80 ");
- write_sfm2(2,9,s_dengji); //顯示濃度等級
- }
- uchar a_a;
- /********STC89C51扇區分布*******
- 第一扇區:1000H--11FF
- 第二扇區:1200H--13FF
- 第三扇區:1400H--15FF
- 第四扇區:1600H--17FF
- 第五扇區:1800H--19FF
- 第六扇區:1A00H--1BFF
- 第七扇區:1C00H--1DFF
- 第八扇區:1E00H--1FFF
- *****************/
- /********STC89C52扇區分布*******
- 第一扇區:2000H--21FF
- 第二扇區:2200H--23FF
- 第三扇區:2400H--25FF
- 第四扇區:2600H--27FF
- 第五扇區:2800H--29FF
- 第六扇區:2A00H--2BFF
- 第七扇區:2C00H--2DFF
- 第八扇區:2E00H--2FFF
- *****************/
- #define RdCommand 0x01 //定義ISP的操作命令
- #define PrgCommand 0x02
- #define EraseCommand 0x03
- #define Error 1
- #define Ok 0
- #define WaitTime 0x01 //定義CPU的等待時間
- sfr ISP_DATA=0xe2; //寄存器申明
- sfr ISP_ADDRH=0xe3;
- sfr ISP_ADDRL=0xe4;
- sfr ISP_CMD=0xe5;
- sfr ISP_TRIG=0xe6;
- sfr ISP_CONTR=0xe7;
- /* ================ 打開 ISP,IAP 功能 ================= */
- void ISP_IAP_enable(void)
- {
- EA = 0; /* 關中斷 */
- ISP_CONTR = ISP_CONTR & 0x18; /* 0001,1000 */
- ISP_CONTR = ISP_CONTR | WaitTime; /* 寫入硬件延時 */
- ISP_CONTR = ISP_CONTR | 0x80; /* ISPEN=1 */
- }
- /* =============== 關閉 ISP,IAP 功能 ================== */
- void ISP_IAP_disable(void)
- {
- ISP_CONTR = ISP_CONTR & 0x7f; /* ISPEN = 0 */
- ISP_TRIG = 0x00;
- EA = 1; /* 開中斷 */
- }
- /* ================ 公用的觸發代碼 ==================== */
- void ISPgoon(void)
- {
- ISP_IAP_enable(); /* 打開 ISP,IAP 功能 */
- ISP_TRIG = 0x46; /* 觸發ISP_IAP命令字節1 */
- ISP_TRIG = 0xb9; /* 觸發ISP_IAP命令字節2 */
- _nop_();
- }
- /* ==================== 字節讀 ======================== */
- unsigned char byte_read(unsigned int byte_addr)
- {
- EA = 0;
- ISP_ADDRH = (unsigned char)(byte_addr >> 8);/* 地址賦值 */
- ISP_ADDRL = (unsigned char)(byte_addr & 0x00ff);
- ISP_CMD = ISP_CMD & 0xf8; /* 清除低3位 */
- ISP_CMD = ISP_CMD | RdCommand; /* 寫入讀命令 */
- ISPgoon(); /* 觸發執行 */
- ISP_IAP_disable(); /* 關閉ISP,IAP功能 */
- EA = 1;
- return (ISP_DATA); /* 返回讀到的數據 */
- }
- /* ================== 扇區擦除 ======================== */
- void SectorErase(unsigned int sector_addr)
- {
- unsigned int iSectorAddr;
- iSectorAddr = (sector_addr & 0xfe00); /* 取扇區地址 */
- ISP_ADDRH = (unsigned char)(iSectorAddr >> 8);
- ISP_ADDRL = 0x00;
- ISP_CMD = ISP_CMD & 0xf8; /* 清空低3位 */
- ISP_CMD = ISP_CMD | EraseCommand; /* 擦除命令3 */
- ISPgoon(); /* 觸發執行 */
- ISP_IAP_disable(); /* 關閉ISP,IAP功能 */
- }
- /* ==================== 字節寫 ======================== */
- void byte_write(unsigned int byte_addr, unsigned char original_data)
- {
- EA = 0;
- // SectorErase(byte_addr);
- ISP_ADDRH = (unsigned char)(byte_addr >> 8); /* 取地址 */
- ISP_ADDRL = (unsigned char)(byte_addr & 0x00ff);
- ISP_CMD = ISP_CMD & 0xf8; /* 清低3位 */
- ISP_CMD = ISP_CMD | PrgCommand; /* 寫命令2 */
- ISP_DATA = original_data; /* 寫入數據準備 */
- ISPgoon(); /* 觸發執行 */
- ISP_IAP_disable(); /* 關閉IAP功能 */
- EA =1;
- }
- /***********************1ms延時函數*****************************/
- void delay_1ms(uint q)
- {
- uint i,j;
- for(i=0;i<q;i++)
- for(j=0;j<120;j++);
- }
- /******************把數據保存到單片機內部eeprom中******************/
- void write_eeprom()
- {
- SectorErase(0x2000);
- // byte_write(0x2000, s_dengji);
- byte_write(0x2001, s_dengji);
- byte_write(0x2060, a_a);
- }
- /******************把數據從單片機內部eeprom中讀出來*****************/
- void read_eeprom()
- {
- // s_dengji = byte_read(0x2000);
- s_dengji = byte_read(0x2001);
- a_a = byte_read(0x2060);
- }
- /**************開機自檢eeprom初始化*****************/
- void init_eeprom()
- {
- read_eeprom(); //先讀
- if(a_a != 2) //新的單片機初始單片機內問eeprom
- {
- s_dengji = 80;
- a_a = 2;
- write_eeprom();
- }
- }
- /***********讀數模轉換數據********************************************************/
- //請先了解ADC0832模數轉換的串行協議,再來讀本函數,主要是對應時序圖來理解,本函數是模擬0832的串行協議進行的
- // 1 0 0 通道
- // 1 1 1 通道
- unsigned char ad0832read(bit SGL,bit ODD)
- {
- unsigned char i=0,value=0,value1=0;
- SCL=0;
- DO=1;
- CS=0; //開始
- SCL=1; //第一個上升沿
- SCL=0;
- DO=SGL;
- SCL=1; //第二個上升沿
- SCL=0;
- DO=ODD;
- SCL=1; //第三個上升沿
- SCL=0; //第三個下降沿
- DO=1;
- for(i=0;i<8;i++)
- {
- SCL=1;
- SCL=0; //開始從第四個下降沿接收數據
- value<<=1;
- if(DO)
- value++;
- }
- for(i=0;i<8;i++)
- { //接收校驗數據
- value1>>=1;
- if(DO)
- value1+=0x80;
- SCL=1;
- SCL=0;
- }
- CS=1;
- SCL=1;
- if(value==value1) //與校驗數據比較,正確就返回數據,否則返回0
- return value;
- return 0;
- }
- /*************定時器0初始化程序***************/
- void time_init()
- {
- EA = 1; //開總中斷
- TMOD = 0X01; //定時器0、定時器1工作方式1
- ET0 = 1; //開定時器0中斷
- TR0 = 1; //允許定時器0定時
- }
- /****************按鍵處理顯示函數***************/
- void key_with()
- {
- if(key_can == 1)
- {
- s_dengji ++ ; //濃度設置數加1
- if(s_dengji > 999)
- s_dengji = 999;
- }
- if(key_can == 2)
- {
- s_dengji -= 1; //濃度設置數減1
- if(s_dengji <= 1)
- s_dengji = 1 ;
- }
- write_sfm2(2,9,s_dengji); //顯示濃度等級
- write_eeprom(); //保存數據
-
- }
- /********************獨立按鍵程序*****************/
- uchar key_can; //按鍵值
- sbit key1=P1^5;
- sbit key2=P3^3;
- void key() //獨立按鍵程序
- {
- static uchar key_new;
- key_can = 20; //按鍵值還原
- key1 = 1;
- key2 = 1;
- if((key1==0)||(key2==0)) //按鍵按下
- {
- delay_1ms(1); //按鍵消抖動
- if(((key1==0)||(key2==0)) && (key_new == 1))
- { //確認是按鍵按下
- key_new = 0;
- if(key1==0) key_can = 1;
- if(key2==0) key_can = 2;
- }
- }
- else
- key_new = 1;
- }
- /****************報警函數***************/
- void clock_h_l()
- {
- static uchar value;
- if(dengji >= s_dengji ) //報警
- {
- value ++;
- if(value >= 2)
- {
- value = 10;
- beep = ~beep; //蜂鳴器報警
- }
- }else
- {
- if(dengji < s_dengji) //取消報警
- {
- value = 0;
- beep = 1;
- }
- }
- }
- /***************主函數*****************/
- void main()
- {
- beep = 0; //開機叫一聲
- delay_1ms(150);
- P0 = P1 = P2 = P3 = 0xff; //單片機IO口初始化為1
- init_eeprom(); //讀eeprom數據
- time_init(); //初始化定時器
- init_1602();
- while(1)
- {
- key(); //獨立按鍵程序
- if(key_can < 10)
- {
- key_with(); //按鍵按下要執行的程序
- }
- if(flag_300ms == 1)
- {
- flag_300ms = 0;
- clock_h_l(); //報警函數
- dengji = ad0832read(1,0);
- dengji = dengji * 450 / 255.0;
- dengji = dengji - 100; //首先減去零點漂移,一般是1V
- if(dengji < 0)
- dengji = 0;
- dengji = dengji * 2; //將mV轉變成mg/L,系數需要校準
- //電壓每升高0.1V,實際被測氣體的濃度增加20ppm
- //1ppm=1mg/kg=1mg/L=1×10-6 常用來表示氣體濃度,或者溶液濃度。
- write_sfm2(1,9,dengji); //顯示濃度
- }
- }
- }
- /*************定時器0中斷服務程序***************/
- void time0_int() interrupt 1
- {
- static uchar value;
- TH0 = 0x3c;
- TL0 = 0xb0; // 50ms
- value ++;
- if(value % 6 == 0)
- {
- flag_300ms = 1; //300ms
- value = 0;
- }
- }
復制代碼
所有資料51hei提供下載:
keil4版.rar
(50.05 KB, 下載次數: 71)
2018-4-27 16:43 上傳
點擊文件名下載附件
源碼
|