/********************************************** 设计者:胥学辰 时间:2011-11-1——2011-11-6 功能:18B20采集温度,dht11采集湿度,光敏电阻采集光照等, 通过GSM接受指令传输参数,控制外部器件改变环境参数。 完整代码下载:http://www.51hei.com/f/gpsdd.rar ***********************************************/ #include<reg52.h> #include<intrins.h> #include<string.h> #include<stdio.h> #define uchar unsigned char #define uint unsigned int sbit DQ_wen=P2^0; sbit DQ_shi=P2^1; uchar i=0,j=0,flag,tm,temp,wet,dat; char num[]="15282983798"; //定义要拨打的电话号码 char code Huifu_S[]="Fa Song Success!"; //定义成功发送要回复的内容 char code Huifu_F[]="Fa Song Failure!"; //定义失败发送要回复的内容 char Huifu_wsg[]="temp:00.00C wet:23%R"; //定义参数回复数组 char xdata Huancun[200]; //定义返回值存储数组 char code Ok[]="OK"; //返回OK值标志 char code CMTI[]="+CMTI"; //接收短信标志 char code open_1[]="open1"; //短信控制1标志 char code open_2[]="open2"; //短信控制2标志 char code open_3[]="open3"; //短信控制3标志 char code open_4[]="open4"; //短信控制4标志 void delay_key(uchar z); //按键延时函数 void delay_GSM(uint y); //GSM模块延时函数 void delay_18b20(uchar i); //18b20延时函数 void delay_dht11_C(uint j); //dht11长延时函数 void delay_dht11_D(uchar i); //dht11短延时函数 void chuankou_init(); //串口初始化函数 void clear_huancun(); //清除缓存函数 void send_ASCII(uchar *b); //发送ASCII代码函数 void send_16_Hex(uchar c); //发送16进制代码函数 bit panduan(uchar *a,uchar *b); //判断是否返回OK与短信成功代码 void GSM_Text_Message(uchar*num,uchar*Huifu); //Text模式短信发送函数 void GSM_Call(uchar *num); //拨打电话函数 void Tiqu_num(); //提取来电号码函数 void init_18b20(); //18b20初始复位函数 void write_18b20_dat(uchar dat); //18b20写入字节函数 uchar read_18b20_dat(); //18b20读出字节函数 uchar read_18b20_temp_zs(); //18b20读取温度整数函数 uchar read_18b20_temp_xs(); //18b20读取温度小数函数 void dht11_init(void); //dht11初始化函数 uchar read_dht11(void); //dht11读取湿度函数 void chuli_wen_wet_light(); //18b20_dht11_light处理函数 void main() //主函数 { chuankou_init(); //串口初始化 clear_huancun(); //清除缓存 delay_GSM(1000); //延时 while(!panduan(Huancun,Ok)) //判断返回值 { send_ASCII("AT"); //发送联机命令 send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 P0=0x01; //步骤指示灯1 delay_GSM(600); //延时 } clear_huancun(); //清除缓存 P0=0x02; //步骤指示灯2 delay_GSM(600); //延时 send_ASCII("AT+CNMI=2,1"); //设置当有新短信到来时提示 send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 delay_GSM(600); //延时 while(!panduan(Huancun,Ok)); //等待设置成功 clear_huancun(); //清除缓存内容 P0=0x03; //步骤指示灯3 delay_GSM(600); //延时 send_ASCII("AT+CMGF=1"); //文本(TEXT)模式 send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 delay_GSM(600); //延时 while(!panduan(Huancun,Ok)); //等待设置成功 clear_huancun(); //清除缓存内容 P0=0x04; //步骤指示灯4 delay_GSM(600); //延时 send_ASCII("AT+CMGD=1"); //删除第一条信息 send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 delay_GSM(600); //延时 while(!panduan(Huancun,Ok)); //等待设置成功 clear_huancun(); //清除缓存内容 while(1) //与GSM模块联机成功后的大循环 { P0=0x05; //步骤指示灯5 chuli_wen_wet_light(); //实时采集处理温度 if(panduan(Huancun,CMTI)) //若缓存字符串中含有"+CMTI"就表示有新的短信 { P0=0x06; //步骤指示灯6 clear_huancun(); //清除缓存内容 delay_GSM(600); //延时 send_ASCII("AT+CMGR=1"); //发送读取信息指令 send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 delay_GSM(600); //延时 Tiqu_num(); //提取来电号码 if(panduan(Huancun,open_1)) //如果信息内容是"open1" { P0=0x10; //指示灯 GSM_Text_Message(num,Huifu_wsg); //发送环境参数信息 GSM_Text_Message(num,Huifu_S); //发送成功短信回复 } else if(panduan(Huancun,open_2)) //如果信息内容是"open2" { P0=0x20; //指示灯 GSM_Text_Message(num,Huifu_S); //发送成功短信回复 } else if(panduan(Huancun,open_3)) //如果信息内容是"open3" { P0=0x40; //指示灯 GSM_Text_Message(num,Huifu_S); //发送成功短信回复 } else if(panduan(Huancun,open_4)) //如果信息内容是"open4" { P0=0x80; //指示灯 GSM_Text_Message(num,Huifu_S); //发送成功短信回复 } else GSM_Text_Message(num,Huifu_F); //发送失败短信回复 clear_huancun(); //清除缓存内容 delay_GSM(600); //延时 send_ASCII("AT+CMGD=1"); //删除读取后的信息 send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 delay_GSM(600); //延时 while(!panduan(Huancun,Ok)); //等待设置成功 clear_huancun(); //清除缓存内容 } } } /**************************** 各环节延时函数 *****************************/ void delay_key(uchar x) //按键控制延时函数 { uchar y; for(;x>0;x--) for(y=110;y>0;y--); } void delay_GSM(uint y) //GSM模块延时函数 { uint x; for(;y>0;y--) { x=1500; while(x--); } } void delay_18b20(uchar i) //18b20延时函数 { while(i--); } void delay_dht11_C(uint j) //dht11长延时函数 { uchar i; for(;j>0;j--) for(i=0;i<27;i++); } void delay_dht11_D(uchar i) //dht11短延时函数 { for(;i>0;i--) _nop_(); } void chuankou_init() //串口初始化函数 { SCON=0x50; //串口:方式1,允许发送和接收 TMOD=0x20; //定时器1:模式2,8位自动重装模式,用于产生波特率 TH1=0xFD; //11.0592MHZ晶振,波特率为9600 TL1=0xFD; TR1=1; //开启定时器1 ES=1; //开启串口中断 EA=1; //开启全局中断 } /**************************** GSM部分函数 *****************************/ void clear_huancun() //清除缓存内容函数 { uchar k; for(k=0;k<200;k++) { Huancun[k]=0; //将缓存内容清零 } i=0; //接收字符串的起始存储位置 } void send_ASCII(uchar *b) //发送字符串函数 { ES=0; //关串口中断 for(b;*b!='\0';b++) { SBUF=*b; while(TI!=1); //等待发送完成 TI=0; //清除发送中断标志位 } ES=1; //开串口中断 } void send_16_Hex(uchar c) //发送16进制函数 { ES=0; //关串口中断 SBUF=c; while(TI!=1); //等待发送完成 TI=0; //清除发送中断标志位 ES=1; //开串口中断 } bit panduan(uchar *a,uchar *b) //判断是否成功返回值函数 { if(strstr(a,b)!=NULL) //如果a包含了b,则1;否则0 return 1; else return 0; } void Tiqu_num() //提取来电号码函数函数 { uchar i; for(i=0;i<11;i++) { num[i]=Huancun[i+36]; } } void GSM_Text_Message(uchar *num,uchar *Huifu) //Text模式短信发送 { send_ASCII("AT"); //发送联机命令 send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 delay_GSM(200); send_ASCII("AT+CMGF=1"); //发送TXET命令 send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 delay_GSM(200); send_ASCII("AT+CSCS=GSM"); send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 delay_GSM(200); send_ASCII("AT+CSMP=17,167,0,0"); send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 delay_GSM(200); send_ASCII("AT+CMGS="); //接收短信手机号 send_ASCII(num); send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); //发送换行符 delay_GSM(200); send_ASCII(Huifu); //发送发送成功命令 send_ASCII("\x1a"); //16进制结束标志 delay_GSM(200); } void GSM_Call(uchar *num) //拨打电话函数 { send_ASCII("ATD"); send_ASCII(num); send_ASCII("F;"); send_16_Hex(0x0d); //发送回车符 send_16_Hex(0x0a); delay_GSM(200); } /**************************** 18b20函数 *****************************/ void init_18b20() { uchar a; DQ_wen=1; //DQ复位 delay_18b20(8); //稍做延时 DQ_wen=0; //单片机将DQ拉低 delay_18b20(80); //精确延时大于480us DQ_wen=1; //拉高总线 delay_18b20(14); //延时 a=DQ_wen; //稍做延时后 如果a=0则初始化成功 a=1则初始化失败 delay_18b20(100); } void write_18b20_dat(uchar dat) //18b20写入字符函数 { uchar i; for(i=0;i<8;i++) { DQ_wen=0; DQ_wen=dat&0x01; delay_18b20(5); DQ_wen=1; dat>>=1; } } uchar read_18b20_dat() //18b20读出字符函数 { uchar i,dat=0; DQ_wen=1; for(i=0;i<8;i++) { DQ_wen=0; _nop_(); _nop_(); dat>>=1; DQ_wen=1; _nop_(); _nop_(); if(DQ_wen) dat|=0x80; delay_18b20(30); DQ_wen=1; } return dat; } uchar read_18b20_temp_zs() //18b20温度整数读取函数 { uchar a,b,c; init_18b20(); write_18b20_dat(0xcc); write_18b20_dat(0x44); init_18b20(); write_18b20_dat(0xcc); write_18b20_dat(0xbe); a=read_18b20_dat(); b=read_18b20_dat(); c=a&0x0f; b<<=4; b+=(a&0xf0)>>4; return b; } uchar read_18b20_temp_xs() //18b20温度小数读取函数 { uchar a,b,c; init_18b20(); write_18b20_dat(0xcc); write_18b20_dat(0x44); init_18b20(); write_18b20_dat(0xcc); write_18b20_dat(0xbe); a=read_18b20_dat(); b=read_18b20_dat(); c=a&0x0f; b<<=4; b+=(a&0xf0)>>4; return c; } /**************************** dht11函数 *****************************/ void dht11_init(void) //dht11初始化复位函数 { uchar i; for(i=0;i<8;i++) { flag=2; while((!DQ_shi)&&flag++); delay_dht11_D(2); tm=0; if(DQ_shi) tm=1; flag=2; while((DQ_shi)&&flag++); if(flag==1) break; dat<<=1; dat|=tm; } } uchar read_dht11(void) //dht11读取湿度函数 { uchar wet_L,temp_L,temp_H,wet_H,JY; DQ_shi=0; delay_dht11_C(180); DQ_shi=1; delay_dht11_D(4); DQ_shi=1; if(!DQ_shi) { flag=2; while((!DQ_shi)&&flag++); flag=2; while((DQ_shi)&&flag++); dht11_init(); wet_H=dat; dht11_init(); wet_L=dat; dht11_init(); temp_H=dat; dht11_init(); temp_L=dat; dht11_init(); JY=dat; DQ_shi=1; tm=(temp_H+temp_L+wet_H+wet_L); if(tm==JY) { temp=temp_H; wet=wet_H; } } return wet; } void chuli_wen_wet_light() //采集数据处理函数 { uchar temp_zs,temp_xs,wet_zs; temp_zs=read_18b20_temp_zs(); //读取18b20温度整数 temp_xs=read_18b20_temp_xs(); //读取18b20温度小数 wet_zs=read_dht11(); //dht11湿度整数 Huifu_wsg[6]=temp_zs%10+'0'; //数据转化为字符存入字符数组 Huifu_wsg[5]=temp_zs/10+'0'; Huifu_wsg[9]=temp_xs%10+'0'; Huifu_wsg[8]=temp_xs/10+'0'; Huifu_wsg[17]=wet_zs%10+'0'; Huifu_wsg[16]=wet_zs/10+'0'; } void timer4() interrupt 4 using 1 { ES=0; //关串口中断,防止中断嵌套 if(TI) //如果是发送中断,则不做任何处理 TI=0; //清除发送中断标志位 if(RI) //如果是接送中断,则进行处理 { RI=0; //清除接收中断标志位 Huancun[i]=SBUF; //将接收到的字符串存到缓存中 i++; //缓存指针向后移动 if(i>200) //如果缓存满,将缓存指针指向缓存的首地址 i=0; } ES = 1; //开启串口中断 }