智能火灾报警系统仿真源程序资料

合集下载
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

智能火灾报警系统仿真源程序
#include <reg52.h>
#include eepom52.h
#define uchar unsigned char
#define uint unsigned int
#include <intrins.h>
//数码管段选定义0 1 2 3 4 5 6 7 8 9 uchar code smg_du[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90, 0x88,0x83,0xc6,0xa1,0x86,0x8e,0xff};
//断码
//数码管位选定义
uchar code smg_we[]={0x7f,0xbf,0xdf,0xef};
//uchar code smg_we[]={0xfe,0xfd,0xfb,0xf7};
uchar dis_smg[8] = {0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8};
uchar smg_i = 3; //显示数码管的个位数
sbit SCL=P3^4; //SCL定义为P1口的第3位脚,连接ADC0832SCL脚//DO定义为P1口的第4 位脚,连接ADC0832DO脚sbit DO=P3^3;
脚ADC0832CS位脚,连接4口的第P1定义为sbit CS=P3^2; //CS
口的定义//18b20 IO= P3^5; sbit dq
sbit beep = P3^6; //蜂鸣器IO口定义
uint temperature,s_temp ; //温度的变量
uchar dengji,s_dengji; //烟物等级
uchar shoudong; //手动报警键
uint huoyan;
bit flag_300ms ;
uchar key_can;
//按键值的变量
uchar menu_1; //菜单设计的变量
bit flag_lj_en; //按键连加使能
//按键连3次连加后使能加的数就越大了bit flag_lj_3_en;
uchar key_time,flag_value; //用做连加的中间变量
bit key_500ms ;
uchar flag_clock;
uchar zd_break_en,zd_break_value; //自动退出设置界面
uchar a_a;
/***********************1ms延时函数*****************************/
void delay_1ms(uint q)
{
uint i,j;
for(i=0;i<q;i++)
for(j=0;j<120;j++);
}
/***********************小延时函数*****************************/
void delay_uint(uint q)
{
while(q..);
}
******************/ eepom中/******************把数据保存到单片机内部void write_eepom()
{
SectorErase(0x2000);
byte_write(0x2000, s_temp);
byte_write(0x2001, s_dengji);
byte_write(0x2060, a_a);
}
*****************/ /******************eepom中读出来把数据从单片机内部void read_eepom()
{
= byte_read(0x2000); s_temp
s_dengji = byte_read(0x2001);
= byte_read(0x2060); a_a
}
*****************/ /**************初始化开机自检eepomvoid init_eepom()
{
先读read_eepom(); //
EEPOM
if(a_a != 1) 新的单片机初始单片机内问//
{
s_temp = 50;
s_dengji = 5;
a_a = 1;
write_eepom();
}
}
/***********************18b20初始化函数*****************************/ void init_18b20()
{
bit q;
dq = 1; //把总线拿高
delay_uint(1); //15us
dq = 0; //给复位脉冲
//750us
delay_uint(80);
dq = 1; 等待//把总线拿高
delay_uint(10); //110us
q = dq; 读取//18b20初始化信号
delay_uint(20); //200us
dq = 1; 释放总线把总线拿高//
}
***************/ 写/*************18b20内的数据void write_18b20(uchar dat) {
uchar i;
for(i=0;i<8;i++)
{ 写数据是低位开始//dq = 0; // 把总线拿低写时间隙开始
总线写数据了18b20向dq = dat & 0x01; //
delay_uint(5); // 60us
dq = 1; //释放总线
dat >>= 1;
}
}
/*************读取18b20内的数据***************/
uchar read_18b20()
{
uchar i,value;
for(i=0;i<8;i++)
{
dq = 0; //把总线拿低读时间隙开始
value >>= 1; //读数据是低位开始
dq = 1; //释放总线
//开始读写数据if(dq == 1)
value |= 0x80;
delay_uint(5); //60us 读一个时间隙最少要保持60us 的时间
}
返回数据// return value;
}
***************/ 读出来的是小数读取温度的值/*************
uint read_temp()
{
uint value;
在读取温度的时候如果中断的太频繁了,就应该把中uchar low;
//断给关了,否则会影响到18b20的时序
init_18b20(); //初始化18b20
ROM
位64跳过// write_18b20(0xcc);
write_18b20(0x44); //启动一次温度转换命令
delay_uint(50);
//500us
init_18b20(); //初始化18b20
write_18b20(0xcc); // 跳过64位ROM
write_18b20(0xbe); //发出读取暂存器命令
EA = 0;
low = read_18b20();
//读温度低字节
value = read_18b20(); //读温度高字节
EA = 1;
value <<= 8; //把温度的高位左移8 位
//把读出的温度低位放到value |= low; value 的低八位中
value *= 0.0625; //转换到温度值
return value; //返回读出的温度
}
/***********读数模转换数据********************************************************/
//请先了解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)

return value;
return 0;
}
/***********************数码显示函数*****************************/ void display()
{
uchar i;
for(i=0;i<smg_i;i++)
{
// 位选P2 = smg_we[i];
P1 = dis_smg[i]; //段选
delay_1ms(1);
P1 = 0xff; //消隐
P2 = 0xff; //位选
}
}
***************/ 0定时器初始化程序/************* void time_init()
{
// EA = 1;
开总中断TMOD = 0X01; 1、定时器定时器// 0工作方式1
ET0 = 1; 中断0开定时器//
= 1;
TR0 定时0允许定时器//
}
***************/
按键处理数码管显示函数/****************
void key_with()
{
if(key_can == 4) //紧急报警键手动报警
{
if(menu_1 == 0)
shoudong = 1;
}
设置键// if(key_can == 1)
{
menu_1 ++;
if(menu_1 >= 3)
{
menu_1 = 0;
}
if(menu_1 == 0)
{
取温度的个位数显示// dis_smg[0] = smg_du[temperature % 10];
dis_smg[1] = smg_du[temperature / 10 % 10]; //取温度的十位显示
smg_i = 3;
}
if(menu_1 == 1)
{
dis_smg[0] = smg_du[s_temp % 10]; // 取个位显示取十位显示dis_smg[1] = smg_du[s_temp / 10 % 10] ; //
dis_smg[2] = 0xbf ;
A
dis_smg[3] = smg_du[10]; 显示//
smg_i = 4;
}
if(menu_1 == 2)
{
dis_smg[0] = smg_du[s_dengji % 10]; //取个位显示
dis_smg[1] = 0xbf ;
dis_smg[2] = 0xbf ;
dis_smg[3] = smg_du[11]; // 显示B
smg_i = 4;
}
}
if(menu_1 == 0)
{
if((key_can == 2) || (key_can == 3))
shoudong = 0; //取消手动报警
}
//设置高温报警if(menu_1 == 1)
{
if(key_can == 2)
{
if(flag_lj_3_en == 0)
按键按下未松开自动加三次s_temp ++ ; //
else
s_temp += 10; //按键按下未松开自动加三次之后每次自动加10 if(s_temp > 99)
s_temp = 99;
dis_smg[0] = smg_du[s_temp % 10]; //取个位显示dis_smg[1] = smg_du[s_temp / 10 % 10]; //取十位显示
dis_smg[2] = 0xbf ;
A
显示//dis_smg[3] = smg_du[10];
}
if(key_can == 3)
{
if(flag_lj_3_en == 0)
按键按下未松开自动加三次// s_temp .. ;
else
s_temp .= 10; // 按键按下未松开自动减三次之后每次自动减10
if(s_temp <= 10)
s_temp = 10 ;
dis_smg[0] = smg_du[s_temp % 10]; //取个位显示dis_smg[1] = smg_du[s_temp / 10 % 10]; //取十位显示
dis_smg[2] = 0xbf;
A
//显示dis_smg[3] = smg_du[10];
}
}
//设置低温报警if(menu_1 == 2)
{
if(key_can == 2)
{
if(flag_lj_3_en == 0)
s_dengji ++ ;
else
s_dengji ++ ;
if(s_dengji >= 9)
s_dengji = 9;
取个位显示// dis_smg[0] = smg_du[s_dengji % 10];
dis_smg[1] = 0xbf ;
dis_smg[2] = 0xbf ;
dis_smg[3] = smg_du[11];
//B
显示}
if(key_can == 3)
{
if(flag_lj_3_en == 0)
s_dengji .. ;
else
s_dengji ..;
if(s_dengji <= 1)
s_dengji = 1;
dis_smg[0] = smg_du[s_dengji % 10]; // 取个位显示dis_smg[1] = 0xbf ;
dis_smg[2] = 0xbf;
dis_smg[3] = smg_du[11];
//显示B
}
}
}
/****************独立按键处理函数************************/ void key()
{
static uchar key_new = 0,key_old = 0,key_value = 0;
if(key_new == 0)
//按键松开的时候做松手检测{
if((P2 & 0x0f) == 0x0f)
key_value ++;
else
key_value = 0;
if(key_value >= 5)
{
write_eepom();
key_value = 0;
key_time = 0;
key_new = 1;
flag_lj_en = 0; // 关闭连加使能
flag_lj_3_en = 0; // 关闭3秒后使能
flag_value = 0; // 清零
}
}
else
{
if((P2 & 0x0f) != 0x0f)
key_value ++; 按键按下的时候//else
key_value = 0;
if(key_value >= 5)
{
key_value = 0;
key_new = 0;
flag_lj_en = 1; //连加使能
zd_break_en = 1; //自动退出设置界使能
zd_break_value = 0; //自动退出设置界变量清零
}
}
key_can = 20;
if(key_500ms == 1)
{
key_500ms = 0;
zd_break_en = 1; //自动退出设置界使能
zd_break_value = 0; //自动退出设置界变量清零
key_new = 0;
key_old = 1;
}
if((key_new == 0) && (key_old == 1))
{
switch(P2 & 0x0f)
{
case 0x0e: key_can = 4; break; //得到k1键值
case 0x0d: key_can = 3; break; //得到k2键值
case 0x0b: key_can = 2; break; //得到k3键值
case 0x07: key_can = 1; break;
//得到k4键值
}
}
key_old = key_new;
}
/****************报警函数***************/
void clock_h_l()
{
static uchar value;
报警//if((dengji >= s_dengji) || (temperature >= s_temp) || (shoudong == 1)) {
value ++;
if(value >= 3)
{
value = 10;
beep = ~beep; // 蜂鸣器报警
}
}else
{
if((dengji < s_dengji) && (temperature < (s_temp . 1)) && (shoudong == 0)) //取消报警
{
value = 0;
beep = 1;
}
}
}
void main()
{
static uchar value;
读取温度值temperature = read_temp(); // 数据读init_eepom(); //eepom// 初始化定时器time_init();
delay_1ms(650);
while(1)
{
key();
//独立按键程序
if(key_can < 10)
{
按键按下要执行的程序// key_with();
}
temperature = read_temp(); //读取温度值if(flag_300ms == 1)
{
flag_300ms = 0;
clock_h_l();
dengji = ad0832read(1,0);
dengji = dengji * 10 / 250;
huoyan = ad0832read(1,1);
huoyan = 10 . huoyan * 10 / 250;
if(menu_1 == 0)
{
if(temperature >= 99)
temperature = 99;
if(value > 20)
value = 0;
if(huoyan < 2)
{
smg_i = 4;
显示烟物报警等级// dis_smg[3]=smg_du[dengji];
// . dis_smg[2]= 0xbf;
dis_smg[1]=smg_du[temperature/10_x0010_]; //十位
dis_smg[0]=smg_du[temperature_x0010_]; //个位ADC0832为
8位ADC,数值为0~255,我们将其分开放入l_tmpdate数组中显示
}else
{
shoudong = 1;
smg_i = 4;
dis_smg[3]=0x89; //H;
//H; dis_smg[2]=0x89;
//H; dis_smg[1]=0x89;
//H;
dis_smg[0]=0x89;
}
}
if(zd_break_en == 1) 自动退出设置界面程序//{
加一次300ms每// zd_break_value ++;
if(zd_break_value > 100) //30秒后自动退出设置界面{
menu_1 = 0; //
smg_i = 3;
zd_break_en = 0;
zd_break_value = 0;
}
}
}
display(); //数码管显示函数
}
}
/*************定时器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;
}
按下按键使能if(flag_lj_en == 1) //{
key_time ++;
if(key_time >= 10) //500ms
{
key_time = 0;
key_500ms = 1; //500ms
flag_value ++;
if(flag_value > 3)
{
flag_value = 10;
flag_lj_3_en = 1; //3次后1.5 秒连加大些
}
}
}
}。

相关文档
最新文档