|
我是单片机小白,自己摸索了毕业设计,感觉好困难,论文出了一些问题,劳烦大神指教一二。 我们的毕业设计要求使用protues进行电路仿真,相关系统原理图中的各个器件按照参考资料都已画好,程序也有(存在问题,编译下来有很多的错误),鄙人目前使用的软件版本是protues7.8和keil4,来进行联调仿真。
老师制定的任务书如下
论文题目:基于单片机的智能家居系统设计
设计目的及任务
以单片机为控制核心,设计实现智能家居控制,该系统具有室内防盗报警功能和室内环境参数实时检测功能。主要设计任务:
1. 实现红外人体感应报警;
2. 实现智能检测室内温度湿度;
3. 实现智能时钟显示。
设计思路
参考所查阅的资料,基本设计思路是:该系统AT89C52(原计划用STC89C52,可是protues元件库中没有,只好用它来代替)为控制中心,加以外围电路包括采集电路、传输电路、LCD1602( LM016)显示电路、数据处理电路、报警装置电路、时钟电路及电路元器件,报警系统装置采用发光二极管和蜂鸣器,数据采集装置使用温湿度传感器(SHT10),时钟芯片选用DS1302。单片机开发软件采用C语言编程,实现室内防盗报警功能和室内环境参数实时检测功能。
技术要求、预期成果
完成单片机的智能家居系统设计,在设计过程中能熟练的使用C语言和proteus仿真软件,掌握单片机及传感器的技术原理和应用,熟悉系统各部分电路的功能,使整个系统能够完成对室内环境的实时监测和及时应对。在答辩中能够尽可能的演示设计中的环境数据采集和实时监测的过程。
存在的主要问题:
1.我按照参考资料画好了protue仿真的大部分器件,连接好了温湿度传感器(SHT10)的相关电路,结果是用掉了p0、p2、p3的大部分接口,只剩下p1口了,时钟芯DS1302和热释电红外传感如何设计才能接到AT89C52的单片机上,还能实现红外人体感应报警和智能时钟显示吗?
2.室内防盗报警功能是如果监测到有人闯入室内,会发出声光报警。红外人体感应通过热释电红外传感器实现,报警系统装置(声光报警电路)采用发光二极管和蜂鸣器实现。参考资料里采用三极管驱动蜂鸣器报警来提醒温湿度超过限定值,我只需要实现室内环境参数实时检测功能,温湿度的报警电路可以去掉吗,用室内防盗报警电路替换,也就是用发光二极管和蜂鸣器,配合热释电红外传感器实现防盗报警,这个怎么更改呢?
3.智能时钟显示这一块,显示器已经用来显示温湿度了,怎么再切换显示时间呢,而且时钟芯片仅起显示时间的功能,不对电路做其他控制,具体如何设置呢?
4.原资料的程序编译下来有很多错误,无法生成.hex文件,求大神结合我要实现的功能更改一下我的源程序。
原参考资料仿真图
我参考画的
热释电红外传感器内部电路图
声光报警模块电路
参考资料源程序(本身好像有错还是不完整,编译之后无法生成.hex文件)
- //*********主函数*****************
- void main(void)
- {
- value humi_val,temp_val;
- unsigned char error,checksum;
- unsigned int wendu,shidu;
- LCD_init();
- s_connectionreset();
- LCD_disp_str(0,1,"TE ");
- LCD_disp_str(0,2,"RH ");
- //*********初始化温度显示区*********
- LCD_disp_str(2,1,"TTT.T C");
- //*********初始化湿度显示区*********
- LCD_disp_str(2,2,"RRR.R%");
- delay_n10us(20000); //延时0.2s
- while(1)
- { error=0;
- error+=s_measure((unsigned char*)&humi_val.i,&checksum,HUMI);
- error+=s_measure((unsigned char*)&temp_val.i,&checksum,TEMP);
- if(error!=0) s_connectionreset();
- else
- { humi_val.f=(float)humi_val.i;
- temp_val.f=(float)temp_val.i;
- calc_SHT10(&humi_val.f,&temp_val.f);
- wendu=10*temp_val.f;
- LCD_disp_char(2,1,wendu/1000+'0'); //显示温度百位
- LCD_disp_char(3,1,(wendu%1000)/100+'0'); //显示温度十位
- LCD_disp_char(4,1,(wendu%100)/10+'0'); //显示温度个位
- LCD_disp_char(6,1,(wendu%10)+'0'); //显示温度小数点后第一位
- shidu=10*humi_val.f;
- LCD_disp_char(2,2,shidu/1000+'0'); //显示湿度百位
- LCD_disp_char(3,2,(shidu%1000)/100+'0'); //显示湿度十位
- LCD_disp_char(4,2,(shidu%100)/10+'0'); //显示湿度个位
- LCD_disp_char(6,2,(shidu%10)+'0'); //显示湿度小数点后第一位
- }
-
- delay_n10us(80000); //延时约0.8s
- }
- }
- 系统部分程序:
- /*************定义接口********************
- P0------DB0~DB7 (LCD1602)
- P2.4------RS (LCD1602)
- P2.5------RW (LCD1602)
- P2.6------E (LCD1602)
- P3.5------SOUNDER (蜂鸣器)
- P3.6------SCK (SHT10) //时钟
- P3.7------DATA (SHT10) //数据
- *****************************************/
- #include <AT89x51.h>
- #include <intrins.h>
- #include <math.h> //Keil library
- #include <stdio.h> //Keil library
- #define TEMPUP 24 //温度上限, 达到该温度蜂鸣器报警
- typedef unsigned char BYTE;
- typedef unsigned int WORD;
- typedef bit BOOL ;
- /*******************************************/
- sbit buzzer = P3^5; //蜂鸣器
- BOOL start;
- /******************************************/
- //************第一部分LCD1602设置******
- START*************************************/
- #define LCD_DB P0
- sbit LCD_RS=P2^5; //P2^5是p2.5
- sbit LCD_RW=P2^6; //P2^6是p2.6
- sbit LCD_E=P2^7; //P2^7是p2.7
- /******定义函数****************/
- #define uchar unsigned char
- #define uint unsigned int
- void LCD_init(void); //初始化函数
- void LCD_write_command(ucharcommand); //写指令函数
- void LCD_write_data(uchardat); //写数据函数
- void LCD_disp_char(ucharx,uchar y,uchar dat);//在某个屏幕位置上显示一个字符,X(0-15),y(1-2)
- void LCD_disp_str(ucharx,uchar y,uchar *str); //LCD1602显示字符串函数
- void delay_n10us(uintn); //延时函数
- /*--------------------------------------
- ;初始化LCD1602
- ;-------------------------------------*/
- void LCD_init(void)
- {
- delay_n10us(10);
- LCD_write_command(0x38);//设置8位格式,2行,5x7
- delay_n10us(10);
- LCD_write_command(0x0c);//整体显示,关光标,不闪烁
- delay_n10us(10);
- LCD_write_command(0x06);//设定输入方式,增量不移位
- delay_n10us(10);
- LCD_write_command(0x01);//清除屏幕显示
- delay_n10us(100); //延时清屏,延时函数,延时约n个10us
- }
- /*--------------------------------------
- ;LCD1602写指令函数
- ;-------------------------------------*/
- void LCD_write_command(uchardat)
- {
- delay_n10us(10);
- LCD_RS=0; //指令
- LCD_RW=0; //写入
- LCD_E=1; //允许
- LCD_DB=dat;
- delay_n10us(10);
- LCD_E=0;
- delay_n10us(10);
- }
- /*--------------------------------------
- ;LCD1602写数据函数
- ;-------------------------------------*/
- void LCD_write_data(uchardat)
- {
- delay_n10us(10);
- LCD_RS=1; //数据
- LCD_RW=0; //写入
- LCD_E=1; //允许
- LCD_DB=dat;
- delay_n10us(10);
- LCD_E=0;
- delay_n10us(10);
- }
- /*--------------------------------------
- ;LCD1602显示一个字符函数,在某个屏幕位置上显示一个字符,X(0-15),y(1-2)。
- ;-------------------------------------*/
- void LCD_disp_char(ucharx,uchar y,uchar dat)
- {
- uchar address;
- if(y==1)
- address=0x80+x;
- else
- address=0xc0+x;
- LCD_write_command(address);
- LCD_write_data(dat);
- }
- /*--------------------------------------
- ;LCD1602显示字符串函数,在某个屏幕起始位置{X(0-15),y(1-2)}上显示一个字符串。
- ;-------------------------------------*/
- void LCD_disp_str(ucharx,uchar y,uchar *str)
- {
- uchar address;
- if(y==1)
- address=0x80+x;
- else
- address=0xc0+x;
- LCD_write_command(address);
- while(*str!='\0')
- {
- LCD_write_data(*str);
- str++;
- }
- }
- /*--------------------------------------
- ;延时函数,延时约n个10us
- ;-------------------------------------*/
- void delay_n10us(uintn) //延时n个10us@12M晶振
- {
- uint i;
- for(i=n;i>0;i--)
- {
- _nop_();_nop_();_nop_();_nop_();_nop_();_nop_();
- }
- }
- //***************第一部分LCD1602设置*******
- END****************************************
- //*************第二部分SHT10设置*************
- START****************************************
- sbit SCK = P3^6; //定义通讯时钟端口
- sbit DATA = P3^7; //定义通讯数据端口
- typedef union
- { unsigned int i; //定义了两个共用体
- float f;
- } value;
- enum {TEMP,HUMI}; //TEMP=0,HUMI=1
- #define noACK 0 //用于判断是否结束通讯
- #define ACK 1 //结束数据传输
- //adr command r/w
- #define STATUS_REG_W0x06 //000 0011 0
- #define STATUS_REG_R0x07 //000 0011 1
- #define MEASURE_TEMP0x03 //000 0001 1
- #define MEASURE_HUMI0x05 //000 0010 1
- #define RESET 0x1e //000 1111 0
- /****************定义函数****************/
- voids_transstart(void); //启动传输函数
- voids_connectionreset(void); //连接复位函数
- char s_write_byte(unsignedchar value);//SHT10写函数
- char s_read_byte(unsignedchar ack); //SHT10读函数
- char s_measure(unsigned char*p_value, unsigned char *p_checksum, unsigned char mode);//测量温湿度函数
- void calc_SHT10(float*p_humidity ,float *p_temperature);//温湿度补偿
- /*--------------------------------------
- ;启动传输函数
- ;-------------------------------------*/
- void s_transstart(void)
- // generates a transmissionstart
- // _____ ________
- // DATA: |_______|
- // ___ ___
- // SCK : ___| |___| |______
- {
- DATA=1; SCK=0; //Initial state
- _nop_();
- SCK=1;
- _nop_();
- DATA=0;
- _nop_();
- SCK=0;
- _nop_();_nop_();_nop_();
- SCK=1;
- _nop_();
- DATA=1;
- _nop_();
- SCK=0;
- }
- /*--------------------------------------
- ;连接复位函数
- ;-------------------------------------*/
- void s_connectionreset(void)
- // communication reset:DATA-line=1 and at least 9 SCK cycles followed by transstart
- // _____________________________________________________ ________
- // DATA: |_______|
- // _ _ _ _ _ _ _ _ _ ___ ___
- // SCK : __| |__| |__| |__| |__||__| |__| |__| |__| |______| |___| |______
- {
- unsigned char i;
- DATA=1; SCK=0; //Initial state
- for(i=0;i<9;i++) //9 SCK cycles
- {
- SCK=1;
- SCK=0;
- }
- s_transstart(); //transmission start
- }
- /*--------------------------------------
- ;SHT10写函数
- ;-------------------------------------*/
- char s_write_byte(unsignedchar value)
- //----------------------------------------------------------------------------------
- // writes a byte on theSensibus and checks the acknowledge
- {
- unsigned char i,error=0;
- for (i=0x80;i>0;i/=2) //shift bit for masking
- {
- if (i & value) DATA=1; //masking value with i , write toSENSI-BUS
- elseDATA=0;
- SCK=1; //clk for SENSI-BUS
- _nop_();_nop_();_nop_(); //pulswith approx. 3 us
- SCK=0;
- }
- DATA=1; //release DATA-line
- SCK=1; //clk #9for ack
- error=DATA; //check ack (DATA willbe pulled down by SHT10),DATA在第9个上升沿将被SHT10自动下拉为低电
- _nop_();_nop_();_nop_();
- SCK=0;
- DATA=1; //release DATA-line
- return error; //error=1 in case of noacknowledge //返回:0成功,1失败
- }
- /*--------------------------------------
- ;SHT10读函数
- ;-------------------------------------*/
- char s_read_byte(unsignedchar ack)
- // reads a byte form theSensibus and gives an acknowledge in case of "ack=1"
- {
- unsigned char i,val=0;
- DATA=1; //release DATA-line
- for (i=0x80;i>0;i/=2) //shift bit for masking
- { SCK=1; //clk for SENSI-BUS
- if (DATA) val=(val | i); //read bit
- _nop_();_nop_();_nop_(); //pulswith approx. 3 us
- SCK=0;
- }
- if(ack==1)DATA=0; //in case of"ack==1" pull down DATA-Line
- else DATA=1; //如果是校验(ack==0),读取完后结束通讯
- _nop_();_nop_();_nop_(); //pulswith approx. 3 us
- SCK=1; //clk #9 for ack
- _nop_();_nop_();_nop_(); //pulswith approx. 3 us
- SCK=0;
- _nop_();_nop_();_nop_(); //pulswith approx. 3 us
- DATA=1; //release DATA-line
- return val;
- }
- /*--------------------------------------
- ;测量温湿度函数
- ;-------------------------------------*/
- char s_measure(unsigned char*p_value, unsigned char *p_checksum, unsigned char mode)
- // makes a measurement(humidity/temperature) with checksum
- {
- unsigned error=0;
- unsigned int i;
- s_transstart(); //transmission start
- switch(mode){ //send command to sensor
- caseTEMP :error+=s_write_byte(MEASURE_TEMP); break;
- case HUMI : error+=s_write_byte(MEASURE_HUMI); break;
- default : break;
- }
- for (i=0;i<65535;i++) if(DATA==0) break;//wait until sensor has finished the measurement
- if(DATA) error+=1; // or timeout (~2 sec.) isreached
- *(p_value) =s_read_byte(ACK); //read thefirst byte (MSB)
- *(p_value+1)=s_read_byte(ACK); //read the second byte (LSB)
- *p_checksum =s_read_byte(noACK); //read checksum
- return error;
- }
- /*--------------------------------------
- ;温湿度补偿函数
- ;-------------------------------------*/
- void calc_SHT10(float*p_humidity ,float *p_temperature)
- // calculates temperature[C] and humidity [%RH]
- // input : humi [Ticks] (12 bit)
- // temp [Ticks] (14 bit)
- // output: humi [%RH]
- // temp [C]
- { const float C1=-4.0; // for 12 Bit
- const float C2=+0.0405; // for 12 Bit
- const float C3=-0.0000028; // for 12 Bit
- const float T1=+0.01; // for 14 Bit @ 5V
- const float T2=+0.00008; // for 14 Bit @ 5V
- float rh=*p_humidity; // rh: Humidity [Ticks] 12 Bit
- float t=*p_temperature; // t: Temperature [Ticks] 14 Bit
- float rh_lin; // rh_lin: Humidity linear
- float rh_true; // rh_true: Temperaturecompensated humidity
- float t_C; // t_C : Temperature [C]
- t_C=t*0.01 - 40; //calc. temperature fromticks to [C]
- rh_lin=C3*rh*rh + C2*rh + C1; //calc. humidity from ticks to [%RH]
- rh_true=(t_C-25)*(T1+T2*rh)+rh_lin; //calc. temperature compensated humidity[%RH]
- if(rh_true>100)rh_true=100; //cut if the value is outside of
- if(rh_true<0.1)rh_true=0.1; //the physical possible range
- *p_temperature=t_C; //return temperature [C]
- *p_humidity=rh_true; //return humidity[%RH]
- }
- //**********第二部分SHT10设置***************
- //END****************************************
复制代码
|
|