找回密码
 立即注册

QQ登录

只需一步,快速开始

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

单片机红外热释电传感器与温度数码管程序有错误 无法仿真

[复制链接]
跳转到指定楼层
楼主
仿真有错误

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

  5. //数码管段选定义      0     1    2    3    4    5         6         7          8           9        
  6. uchar code smg_du[]={0x05,0x1d,0x46,0x52,0x9c,0x54,0x24,0x58,0x04,0x14,
  7.                                            0x0c,0xa4,0x27,0xc4,0x26,0x2e,0xff};         //断码
  8. //数码管位选定义
  9. uchar code smg_we[]={0x7f,0xdf,0xbf,0xef};

  10. uchar dis_smg[8]  = {0};        

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


  14. sbit dq   = P3^5;        //18b20 IO口的定义
  15. sbit beep = P3^6;   //蜂鸣器IO口定义
  16. uint temperature,s_temp=50 ;  //温度的变量
  17. uchar dengji,s_dengji=5;     //烟物等级
  18. uchar shoudong;            //手动报警键
  19. uchar a_a;

  20. sbit hw    = P3^0;           //红外热释传感器定义
  21. uchar flag_bufang ;    //布防标志位
  22. uchar flag_bufang_en ; //布防标志位使能
  23. uint  flag_value;      //用做定时器的变量
  24. uchar flag_time;       //布防时间
  25. uchar flag_alarm ;     //报警标志位

  26. bit flag_250ms = 1;
  27. uchar key_can;                 //按键值的变量
  28. uchar menu_1;        //菜单设计的变量

  29. /***********************1ms延时函数*****************************/
  30. void delay_1ms(uint q)
  31. {
  32.         uint i,j;
  33.         for(i=0;i<q;i++)
  34.                 for(j=0;j<120;j++);
  35. }

  36. /***********************小延时函数*****************************/
  37. void delay_uint(uint q)
  38. {
  39.         while(q--);
  40. }

  41. /***********************18b20初始化函数*****************************/
  42. void init_18b20()
  43. {
  44.         bit q;
  45.         dq = 1;                                //把总线拿高
  46.         delay_uint(1);            //15us
  47.         dq = 0;                                //给复位脉冲
  48.         delay_uint(80);                //750us
  49.         dq = 1;                                //把总线拿高 等待
  50.         delay_uint(10);                //110us
  51.         q = dq;                                //读取18b20初始化信号
  52.         delay_uint(20);                //200us
  53.         dq = 1;                                //把总线拿高 释放总线
  54. }

  55. /*************写18b20内的数据***************/
  56. void write_18b20(uchar dat)
  57. {
  58.         uchar i;
  59.         for(i=0;i<8;i++)
  60.         {                                         //写数据是低位开始
  61.                 dq = 0;                         //把总线拿低写时间隙开始
  62.                 dq = dat & 0x01; //向18b20总线写数据了
  63.                 delay_uint(5);         // 60us
  64.                 dq = 1;                         //释放总线
  65.                 dat >>= 1;
  66.         }        
  67. }

  68. /*************读取18b20内的数据***************/
  69. uchar read_18b20()
  70. {
  71.         uchar i,value;
  72.         for(i=0;i<8;i++)
  73.         {
  74.                 dq = 0;                         //把总线拿低读时间隙开始
  75.                 value >>= 1;         //读数据是低位开始
  76.                 dq = 1;                         //释放总线
  77.                 if(dq == 1)                 //开始读写数据
  78.                         value |= 0x80;
  79.                 delay_uint(5);         //60us        读一个时间隙最少要保持60us的时间
  80.         }
  81.         return value;                 //返回数据
  82. }

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

  92.         init_18b20();                   //初始化18b20
  93.         
  94.         write_18b20(0xcc);           //跳过64位ROM
  95.         write_18b20(0xbe);           //发出读取暂存器命令
  96.         
  97.         EA = 0;
  98.         low = read_18b20();           //读温度低字节
  99.         value = read_18b20();  //读温度高字节
  100.         EA = 1;
  101.         value <<= 8;                   //把温度的高位左移8位
  102.         value |= low;                   //把读出的温度低位放到value的低八位中
  103.         value *= 0.0625;               //转换到温度值
  104.         return value;                   //返回读出的温度
  105. }


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

  108. unsigned char ad0832read(bit SGL,bit ODD)
  109. {
  110.         unsigned char i=0,value=0,value1=0;               
  111.                 SCL=0;
  112.                 DO=1;
  113.                 CS=0;                //开始
  114.                 SCL=1;                //第一个上升沿        
  115.                 SCL=0;
  116.                 DO=SGL;
  117.                 SCL=1;          //第二个上升沿
  118.                 SCL=0;
  119.                 DO=ODD;
  120.                 SCL=1;            //第三个上升沿
  121.                 for(i=0;i<8;i++)
  122.                 {
  123.                         SCL=0; //开始从第四个下降沿接收数据
  124.                         SCL=1;
  125.                         value<<=1;
  126.                         if(DO)
  127.                                 value++;                                                
  128.                 }
  129.                 for(i=0;i<8;i++)
  130.                 {                        //接收校验数据
  131.                         value1<<=1;
  132.                         if(DO)
  133.                                 value1+=0x80;
  134.                         SCL=1;
  135.                         SCL=0;
  136.                 }
  137.                 CS=1;
  138.                 SCL=1;        
  139.         return 0;
  140. }


  141. /***********************数码显示函数*****************************/
  142. void display()
  143. {
  144.         uchar i;
  145.         P1 = 0xff;                         //消隐                                          
  146.         P2 = smg_we[i];                          //位选
  147.         P1 = dis_smg[i];                 //段选           
  148.         i ++;
  149.         if(i > 4)                        //4位数码管显示
  150.                 i = 0;         
  151. }


  152. /*************定时器0初始化程序***************/
  153. void time_init()         
  154. {
  155.         EA   = 1;                   //开总中断
  156.         TMOD = 0X01;          //定时器0、定时器1工作方式1
  157.         ET0  = 1;                  //开定时器0中断
  158.         TR0  = 1;                  //允许定时器0定时
  159. }

  160. /********************独立按键程序*****************/
  161. uchar key_can;         //按键值

  162. void key()         //独立按键程序
  163. {
  164.         static uchar key_new;
  165.         P2 |= 0x0f;
  166.         if((P2 & 0x0f) != 0x0f)                //按键按下
  167.         {
  168.                 delay_1ms(1);                     //按键消抖动
  169.                 if(((P2 & 0x0f) != 0x0f) && (key_new == 1))
  170.                 {                                                //确认是按键按下
  171.                         key_new = 0;
  172.                         switch(P2 & 0x0f)
  173.                         {
  174.                                 case 0x0e: key_can = 1; break;           //得到k1键值
  175.                                 case 0x0d: key_can = 2; break;           //得到k2键值
  176.                                 case 0x0b: key_can = 3; break;           //得到k3键值
  177.                                 case 0x06: key_can = 4; break;           //得到k4键值
  178.                         }               
  179.                 }                        
  180.         }
  181.         else            //按键松开
  182.                 key_new = 1;        
  183. }

  184. /****************按键处理数码管显示函数***************/
  185. void key_with()
  186. {
  187.         if(key_can == 4)          //紧急报警键  手动报警
  188.         {
  189.                 if(menu_1 == 0)
  190.                         shoudong = 1;
  191.         }
  192.         if(key_can == 1)         //设置键
  193.         {
  194.                 menu_1 ++;
  195.                 if(menu_1 >= 3)
  196.                 {
  197.                         menu_1 = 0;
  198.                 }
  199.         }
  200.         if(menu_1 == 0)
  201.         {
  202.                 if(key_can == 2)     //布防按键
  203.                 {
  204.                         flag_bufang_en = 1;        
  205.                 }
  206.                 if(key_can == 3)         //取消报警  把变量清零
  207.                 {
  208.                         flag_alarm = 0;        
  209.                         shoudong = 0;          //取消手动报警
  210.                         flag_bufang = 0;
  211.                         flag_bufang_en = 0;        
  212.                         flag_value = 0;        
  213.                         beep = 1;
  214.                         dis_smg[2] = 0xfe;
  215.                            flag_time = 0;
  216.                 }                        

  217.         }
  218.         if(menu_1 == 1)                        //设置高温报警
  219.         {
  220.                 if(key_can == 2)
  221.                 {
  222.                         s_temp ++ ;                //高温报警值加1
  223.                         if(s_temp > 99)
  224.                                 s_temp = 99;
  225.                 }
  226.                 if(key_can == 3)
  227.                 {
  228.                         s_temp -- ;                //高温报警值减1
  229.                         if(s_temp <= 10)
  230.                                 s_temp = 10 ;
  231.                 }
  232.                 dis_smg[0] = smg_du[s_temp % 10];                   //取个位显示
  233.                 dis_smg[1] = smg_du[s_temp / 10 % 10];  //取十位显示
  234.                 dis_smg[2] = 0xfe;
  235.                 dis_smg[3] = smg_du[10];        //显示A
  236.         }        
  237.         if(menu_1 == 2)                        //设置烟物报警
  238.         {
  239.                 if(key_can == 2)
  240.                 {
  241.                         s_dengji ++ ;          //烟物报警值加1
  242.                         if(s_dengji >= 9)
  243.                                 s_dengji = 9;
  244.                 }
  245.                 if(key_can == 3)
  246.                 {
  247.                         s_dengji --;          //烟物报警值减1
  248.                         if(s_dengji <= 1)
  249.                                 s_dengji = 1;
  250.                 }
  251.                 dis_smg[0] = smg_du[s_dengji % 10];                   //取个位显示
  252.                 dis_smg[1] = 0xfe ;  
  253.                 dis_smg[2] = 0xfe;
  254.                 dis_smg[3] = smg_du[11];        //显示B
  255.         }        
  256.         key_can = 0;
  257. }  

  258. /****************报警函数***************/
  259. void clock_h_l()
  260. {
  261.         static uchar value;
  262.         if((dengji >= s_dengji) || (temperature >= s_temp)  || (flag_alarm == 1) || (shoudong == 1))                //报警
  263.         {
  264.                 value ++;
  265.                 if(value >= 2)
  266.                 {
  267.                         value = 10;
  268.                         beep = ~beep;          //蜂鸣器报警
  269.                 }
  270.         }else
  271.         {
  272.                         value = 0;
  273.                         beep = 1;                 //取消报警
  274.         }
  275. }

  276. /******************红外报警处理**********************/           
  277. void hongwai_dis()
  278. {
  279.         static uchar value;
  280.         value ++;
  281.         if(value >= 4)
  282.         {
  283.                 value = 0;
  284.                 if(flag_bufang_en == 1) //准备开始布防
  285.                 {
  286.                         flag_time ++ ;
  287.                         if(flag_time >= 15)
  288.                         {
  289.                                 flag_bufang = 1;
  290.                                 flag_bufang_en = 0;
  291.                                 flag_value = 0;         
  292.                         }
  293.                         dis_smg[2] = smg_du[flag_time];
  294.                 }  
  295.                 if(flag_bufang == 1)   //确认布防
  296.                 {
  297.         
  298.                         dis_smg[2] = smg_du[15];
  299.                         if(hw == 1)               //红外有输出
  300.                         {
  301.                                 flag_alarm = 1;        
  302.                         }
  303.                 }                        

  304.         }
  305. }

  306. /***************主函数*****************/
  307. void main()
  308. {
  309.         time_init();                                //初始化定时器
  310.         while(1)
  311.         {
  312.                 key();                                        //独立按键程序
  313.                 if(key_can < 10)
  314.                 {
  315.                         key_with();                        //按键按下要执行的程序
  316.                 }
  317.                 if(flag_250ms == 1)
  318.                 {               
  319.                         flag_250ms = 0;
  320.                         clock_h_l();
  321.                         temperature = read_temp();  //读取温度值
  322.                         dengji = ad0832read(1,0);        
  323.                         dengji = dengji * 10 / 250;
  324.                         dis_smg[1]=smg_du[temperature/10%10];        //十位
  325.                         dis_smg[0]=smg_du[temperature%10];            //个位        ADC0832为8位ADC,数值为0~255,我们将其分开放入l_tmpdate数组中显示
  326.                         dis_smg[3]=smg_du[dengji];            //显示烟物报警等级
  327.                         hongwai_dis();           //红外报警函数
  328.                         if(hw == 1)                //红外热释电指示 有输出就小数点亮
  329.                                 dis_smg[2] = dis_smg[2] & 0XF7;
  330.                         else
  331.                                 dis_smg[2] = dis_smg[2] | 0x0d;                        
  332.                 }
  333.                 delay_1ms(1);
  334.         }
  335. }

  336. /*************定时器0中断服务程序***************/
  337. void time0_int() interrupt 1
  338. {        
  339.         static uchar value;
  340.         TH0 = 0xf8;
  341.         TL0 = 0x30;     // 2ms
  342.         value ++;         
  343.         display();          //数码管显示函数
  344.         if(value % 125 == 0)
  345.         {
  346.                 flag_250ms = 1;           //300ms
  347.                 value = 0;
  348.         }
  349. }
复制代码


51hei图片20220702154054.jpg (760.88 KB, 下载次数: 7)

51hei图片20220702154054.jpg

仿真程序.7z

847.78 KB, 下载次数: 5

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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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