找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2894|回复: 0
打印 上一主题 下一主题
收起左侧

51单片机数码管火灾烟雾检测Proteus仿真(带程序)

[复制链接]
跳转到指定楼层
楼主
51单片机数码管火灾报警仿真仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)


单片机源程序如下:
  1. #include <reg51.h>                 //调用单片机头文件
  2. #define uchar unsigned char  //无符号字符型 宏定义        变量范围0~255
  3. #define uint  unsigned int         //无符号整型 宏定义        变量范围0~65535
  4. #include <intrins.h>
  5. #include "eeprom52.h"


  6. //数码管段选定义      0     1    2    3    4    5         6         7          8           9       
  7. uchar code smg_du[]={0x5f,0x44,0x9d,0xd5,0xc6,0xd3,0xdb,0x45,0xdf,0xd7,0x40,0xfd/*-*/};
  8. //数码管位选定义
  9. uchar code smg_we[]={0x7f,0xbf,0xdf,0xef};

  10. uchar dis_smg[8]  = {0xa0,0x84,0x62,0x2a,0x39,0x2c,0x24,0xba};       

  11. sbit CS=P2^3;                //CS定义为P3口的第2位脚,连接ADC0832CS脚  PCB
  12. sbit SCL=P2^0;                //SCL定义为P3口的第3位脚,连接ADC0832SCL脚
  13. sbit DO=P2^1;                //DO定义为P3口的第4位脚,连接ADC0832DO脚

  14. sbit ledy = P1^6;   //温度报警指示灯
  15. sbit ledr = P1^7;   //烟雾报警指示灯 x
  16. sbit dq   = P1^5;        //18b20 IO口的定义
  17. sbit beep = P3^6;   //蜂鸣器IO口定义
  18. uint temperature,s_temp ;  //温度的变量
  19. uchar dengji,s_dengji;     //烟物等级
  20. uchar shoudong;            //手动报警键


  21. bit flag_300ms = 1;
  22. uchar key_can;                 //按键值的变量
  23. uchar menu_1;        //菜单设计的变量

  24. /***********************1ms延时函数*****************************/
  25. void delay_1ms(uint q)
  26. {
  27.         uint i,j;
  28.         for(i=0;i<q;i++)
  29.                 for(j=0;j<120;j++);
  30. }

  31. /***********************小延时函数*****************************/
  32. void delay_uint(uint q)
  33. {
  34.         while(q--);
  35. }


  36. /******************把数据保存到单片机内部eeprom中******************/
  37. void write_eeprom()
  38. {
  39.         SectorErase(0x2000);
  40.         byte_write(0x2000, s_temp);
  41.         byte_write(0x2001, s_dengji);
  42.         byte_write(0x2060, a_a);       
  43. }

  44. /******************把数据从单片机内部eeprom中读出来*****************/
  45. void read_eeprom()
  46. {
  47.         s_temp   = byte_read(0x2000);
  48.         s_dengji = byte_read(0x2001);
  49.         a_a      = byte_read(0x2060);
  50. }

  51. /**************开机自检eeprom初始化*****************/
  52. void init_eeprom()
  53. {
  54.         read_eeprom();                //先读
  55.         if(a_a != 1)                //新的单片机初始单片机内问eeprom
  56.         {
  57.                 s_temp   = 50;
  58.                 s_dengji = 5;
  59.                 a_a = 1;
  60.                 write_eeprom();           //保存数据
  61.         }       
  62. }

  63. /***********************18b20初始化函数*****************************/
  64. void init_18b20()
  65. {
  66.         bit q;
  67.         dq = 1;                                //把总线拿高
  68.         delay_uint(1);            //15us
  69.         dq = 0;                                //给复位脉冲
  70.         delay_uint(80);                //750us
  71.         dq = 1;                                //把总线拿高 等待
  72.         delay_uint(10);                //110us
  73.         q = dq;                                //读取18b20初始化信号
  74.         delay_uint(20);                //200us
  75.         dq = 1;                                //把总线拿高 释放总线
  76. }

  77. /*************写18b20内的数据***************/
  78. void write_18b20(uchar dat)
  79. {
  80.         uchar i;
  81.         for(i=0;i<8;i++)
  82.         {                                         //写数据是低位开始
  83.                 dq = 0;                         //把总线拿低写时间隙开始
  84.                 dq = dat & 0x01; //向18b20总线写数据了
  85.                 delay_uint(5);         // 60us
  86.                 dq = 1;                         //释放总线
  87.                 dat >>= 1;
  88.         }       
  89. }

  90. /*************读取18b20内的数据***************/
  91. uchar read_18b20()
  92. {
  93.         uchar i,value;
  94.         for(i=0;i<8;i++)
  95.         {
  96.                 dq = 0;                         //把总线拿低读时间隙开始
  97.                 value >>= 1;         //读数据是低位开始
  98.                 dq = 1;                         //释放总线
  99.                 if(dq == 1)                 //开始读写数据
  100.                         value |= 0x80;
  101.                 delay_uint(5);         //60us        读一个时间隙最少要保持60us的时间
  102.         }
  103.         return value;                 //返回数据
  104. }

  105. /*************读取温度的值 读出来的是小数***************/
  106. uint read_temp()
  107. {
  108.         uint value;
  109.         uchar low;                           //在读取温度的时候如果中断的太频繁了,就应该把中断给关了,否则会影响到18b20的时序
  110.         init_18b20();                   //初始化18b20
  111.         write_18b20(0xcc);           //跳过64位ROM
  112.         write_18b20(0x44);           //启动一次温度转换命令
  113.         delay_uint(50);                   //500us

  114.         init_18b20();                   //初始化18b20
  115.        
  116.         write_18b20(0xcc);           //跳过64位ROM
  117.         write_18b20(0xbe);           //发出读取暂存器命令
  118.        
  119.         EA = 0;
  120.         low = read_18b20();           //读温度低字节
  121.         value = read_18b20();  //读温度高字节
  122.         EA = 1;
  123.         value <<= 8;                   //把温度的高位左移8位
  124.         value |= low;                   //把读出的温度低位放到value的低八位中
  125.         value *= 0.0625;               //转换到温度值
  126.         return value;                   //返回读出的温度
  127. }


  128. /***********读数模转换数据********************************************************/       
  129. //请先了解ADC0832模数转换的串行协议,再来读本函数,主要是对应时序图来理解,本函数是模拟0832的串行协议进行的

  130. unsigned char ad0832read(bit SGL,bit ODD)
  131. {
  132.         unsigned char i=0,value=0,value1=0;               
  133.                 SCL=0;
  134.                 DO=1;
  135.                 CS=0;                //开始
  136.                 SCL=1;                //第一个上升沿       
  137.                 SCL=0;
  138.                 DO=SGL;
  139.                 SCL=1;          //第二个上升沿
  140.                 SCL=0;
  141.                 DO=ODD;
  142.                 SCL=1;            //第三个上升沿
  143.                 SCL=0;            //第三个下降沿
  144.                 DO=1;
  145.                 for(i=0;i<8;i++)
  146.                 {
  147.                         SCL=1;
  148.                         SCL=0; //开始从第四个下降沿接收数据
  149.                         value<<=1;
  150.                         if(DO)
  151.                                 value++;                                               
  152.                 }
  153.                 for(i=0;i<8;i++)
  154.                 {                        //接收校验数据
  155.                         value1>>=1;
  156.                         if(DO)
  157.                                 value1+=0x80;
  158.                         SCL=1;
  159.                         SCL=0;
  160.                 }
  161.                 CS=1;
  162.                 SCL=1;       
  163.                 if(value==value1)                                //与校验数据比较,正确就返回数据,否则返回0       
  164.                         return value;
  165.         return 0;
  166. }


  167. /***********************数码显示函数*****************************/
  168. void display()
  169. {
  170.         uchar i;
  171.         P0 =0x00;                         //消隐                                           
  172.         P2 = smg_we[i];                          //位选
  173.         P0 = dis_smg[i];                 //段选          
  174.         i ++;
  175.         if(i >= 4)                        //4位数码管显示
  176.                 i = 0;         
  177. }


  178. /*************定时器0初始化程序***************/
  179. void time_init()          
  180. {
  181.         EA   = 1;                   //开总中断
  182.         TMOD = 0X01;          //定时器0、定时器1工作方式1
  183.         ET0  = 1;                  //开定时器0中断
  184.         TR0  = 1;                  //允许定时器0定时
  185. }

  186. /********************独立按键程序*****************/
  187. uchar key_can;         //按键值

  188. void key()         //独立按键程序
  189. {
  190.         static uchar key_new;
  191.         key_can = 20;                   //按键值还原
  192.         P1 |= 0x0f;
  193.         if((P1 & 0x0f) != 0x0f)                //按键按下
  194.         {
  195.                 delay_1ms(1);                     //按键消抖动
  196.                 if(((P1 & 0x0f) != 0x0f) && (key_new == 1))
  197.                 {                                                //确认是按键按下
  198.                         key_new = 0;
  199.                         switch(P1 & 0x0f)
  200.                         {
  201.                                 case 0x0e: key_can = 4; break;           //得到k1键值
  202.                                 case 0x0d: key_can = 3; break;           //得到k2键值
  203.                                 case 0x0b: key_can = 2; break;           //得到k3键值
  204.                                 case 0x07: key_can = 1; break;           //得到k4键值
  205.                         }               
  206.                 }                       
  207.         }
  208.         else            //按键松开
  209.                 key_new = 1;       
  210. }

  211. /****************按键处理数码管显示函数***************/
  212. void key_with()
  213. {
  214.         if(key_can == 4)          //紧急报警键  手动报警
  215.         {
  216.                 if(menu_1 == 0)
  217.                         shoudong = 1;
  218.         }
  219.         if(key_can == 1)         //设置键
  220.         {
  221.                 menu_1 ++;
  222.                 if(menu_1 >= 3)
  223.                 {
  224.                         menu_1 = 0;
  225.                 }
  226.         }
  227.         if(menu_1 == 0)
  228.         {
  229.                 if((key_can == 2) || (key_can == 3))
  230.                         shoudong = 0;          //取消手动报警
  231.         }
  232.         if(menu_1 == 1)                        //设置高温报警
  233.         {
  234.                 if(key_can == 2)
  235.                 {
  236.                         s_temp ++ ;                //高温报警值加1
  237.                         if(s_temp > 99)
  238.                                 s_temp = 99;
  239.                 }
  240.                 if(key_can == 3)
  241.                 {
  242.                         s_temp -- ;                //高温报警值减1
  243.                         if(s_temp <= 10)
  244.                                 s_temp = 10 ;
  245.                 }
  246.                 dis_smg[0] = smg_du[s_temp % 10];                   //取个位显示
  247.                 dis_smg[1] = smg_du[s_temp / 10 % 10];  //取十位显示
  248.                 dis_smg[2] = 0x80;
  249.                 dis_smg[3] = 0x1b;        //显示c
  250.                 write_eeprom();                           //保存数据
  251.         }       
  252.         if(menu_1 == 2)                        //设置烟物报警
  253.         {
  254.                 if(key_can == 2)
  255.                 {
  256.                         s_dengji ++ ;          //烟物报警值加1
  257.                         if(s_dengji >= 9)
  258.                                 s_dengji = 9;
  259.                 }
  260.                 if(key_can == 3)
  261.                 {
  262.                         s_dengji --;          //烟物报警值减1
  263.                         if(s_dengji <= 1)
  264.                                 s_dengji = 1;
  265.                 }
  266.                 dis_smg[0] = smg_du[s_dengji % 10];                   //取个位显示
  267.                 dis_smg[1] = 0x80 ;  
  268.                 dis_smg[2] = 0x80;
  269.                 dis_smg[3] = 0xc7;        //显示q
  270.                 write_eeprom();                           //保存数据
  271.         }       
  272. }  

  273. /****************报警函数***************/
  274. void clock_h_l()
  275. {
  276.         static uchar value;
  277.         if((dengji >= s_dengji) || (temperature >= s_temp) || (shoudong == 1))                //报警
  278.         {
  279.                 value ++;
  280.                 if(value >= 2)
  281.                 {
  282.                         value = 10;
  283.                         beep = ~beep;          //蜂鸣器报警
  284.                         if(dengji >= s_dengji)
  285.                         {
  286.                                 ledr=~ledr;
  287.                         }
  288.                         if(temperature >= s_temp)
  289.                         {
  290.                                 ledy=~ledy;
  291.                         }
  292.                         if(shoudong == 1)
  293.                         {
  294.                                 ledy=~ledy;
  295.                                 ledr=~ledr;
  296.                         }
  297.                 }
  298.         }else
  299.         {
  300.                 if((dengji < s_dengji) && (temperature < s_temp) && (shoudong == 0))          //取消报警
  301.                 {
  302.                         value = 0;
  303.                         beep = 1;                 //取消报警
  304.                         ledr=1;//取消红灯报警
  305.                         ledy=1;//取消黄灯报警
  306.                 }       
  307.         }
  308. }

  309. /***************主函数*****************/
  310. void main()
  311. {
  312.         beep = 0;                                    //开机蜂鸣器叫一声
  313.         delay_1ms(200);
  314.         P0 = P1 = P2 = P3 = 0xff;   //初始化IO口为高电平
  315.         temperature = read_temp();  //读取温度值
  316.         init_eeprom();  //开始初始化保存的数据
  317.         delay_1ms(650);                               
  318.         temperature = read_temp();  //读取温度值
  319.         time_init(); //初始化定时器                
  320.         while(1)  
  321.         {
  322.                 key();                                        //独立按键程序
  323.                 if(key_can < 10)
  324.                 {
  325.                         key_with();                        //按键按下要执行的程序
  326.                 }
  327.                 if(flag_300ms == 1)
  328.                 {               
  329.                         flag_300ms = 0;
  330.                         clock_h_l();
  331.                         temperature = read_temp();  //读取温度值
  332.                         dengji = ad0832read(1,0);       
  333.                         dengji = dengji * 10 / 250;
  334.                         if(menu_1 == 0)
  335.                         {
  336.                                 if(temperature >= 99)
  337.                                         temperature = 99;
  338.                                 dis_smg[3]=smg_du[dengji];            //显示烟物报警等级
  339.                                 dis_smg[2]= 0x80;                    // -               
  340.                                 dis_smg[1]=smg_du[temperature/10%10];        //十位
  341.                                 dis_smg[0]=smg_du[temperature%10];            //个位        ADC0832为8位ADC,数值为0~255,我们将其分开放入l_tmpdate数组中显示
  342.                         }
  343.                 }
  344.                 delay_1ms(1);
  345.         }
  346. }

  347. /*************定时器0中断服务程序***************/
  348. void time0_int() interrupt 1
  349. {       
  350.         static uchar value;
  351.         TH0 = 0xf8;
  352.         TL0 = 0x30;     // 2ms
  353.         value ++;         
  354.         display();          //数码管显示函数
  355.         if(value % 150 == 0)
  356.         {
  357.                 flag_300ms = 1;           //300ms
  358.                 value = 0;
  359.         }
  360. }
复制代码

所有资料51hei提供下载:
4、Proteus 仿真.zip (995.59 KB, 下载次数: 69)


评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏3 分享淘帖 顶 踩
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表