局域网控制系统-下位机-单片机
扫描二维码
随时随地手机看文章
1 /*-----------------------------------
2 多功能下位机
3 STC89C52RC 11.0592MHz
4
5 -----------------------------------*/
6 #include
7 #include
8
9 char code huanhang[3]={0x0d,0x0a,0}; // "rn"
10 //-----------------普通输出端口---------------//
11 sbit LED0=P1^0;
12 sbit LED1=P1^1;
13 sbit LED2=P1^2;
14 sbit jdq_00=P1^3;
15 sbit fmq_00=P2^5;
16
17 /**********DS18B20***********/
18 bit Temp_Symbol=0;
19 //温度传感_0---------
20 sbit DQ=P2^6;
21 //------------------串口通信协议-----------------//
22 /*
23 客户端数据包格式解释(长度恒为15):
24 例如:A01_fmq_01Off___#
25 A--------数据包的开始标记(可以为A到Z)
26 01-----设备代号
27 fmq_01Off___--------指令(长度恒为10),指令的前4个人字符是指令头部,指令的后6个字符是指令尾部
28 #---------数据包的结束标记
29
30 服务器端数据包格式解释(长度恒为15):
31 例如:A02_SenT010250#
32 A--------数据包的开始标记(可以为A到Z)
33 02-----设备代号
34 SenT010250--------指令(长度恒为10),指令的前4个人字符是指令头部,指令的后6个字符是指令尾部
35 #---------数据包的结束标记
36 */
37 char buf_string[16]; //定义数据包长度为15个字符
38 #define deviceID_1Bit '0' //用于串口通信时,定义本地设备ID的第1位
39 #define deviceID_2Bit '2' //用于串口通信时,定义本地设备ID的第2位
40 #define datapackage_headflag 'A' //用于串口通信时,定义数据包头部的验证标记
41
42 char DataPackage_DS18B20[16]={datapackage_headflag,deviceID_1Bit,deviceID_2Bit,'_','S','e','n','T','_','_','_','_','_','_','#'};
43 char HeartBeat[16]={datapackage_headflag,deviceID_1Bit,deviceID_2Bit,'_','B','e','a','t','_','_','_','_','_','_','#'};
44 //----------------------------------------------//
45
46 /**************************
47 辅助函数
48 ***************************/
49 //改变要发送的DS18B20数据包
50 void Change_DataPackage_DS18B20(int DS18B20_Value)
51 {
52 if(Temp_Symbol)
53 {
54 DataPackage_DS18B20[10] = '-';
55 }
56 else
57 {
58 DataPackage_DS18B20[10] = '0';
59 }
60 DataPackage_DS18B20[11] = 0x30+DS18B20_Value % 10000 / 1000;
61 DataPackage_DS18B20[12] = 0x30+DS18B20_Value % 1000 / 100;
62 DataPackage_DS18B20[13] = 0x30+DS18B20_Value % 100 / 10;
63 }
64 /********************************
65 DS18B20
66 若没有效果,意味着延时可能有问题
67 测温范围 -55℃~+125℃
68
69 单线通信接口
70 1)边沿=触发
71 2)电平持续时间=信息
72 3)一字节的电平组合=指令
73 ********************************/
74 //延时------
75 void delay_DS18B20(unsigned int t)
76 {
77 for (;t>0;t--);
78 }
79 //复位,使得从设备可以接收指令-----------
80 void Reset_DS18B20()
81 {
82 char presence=1;
83 while(presence)
84 {
85 while(presence)
86 {
87 DQ=1;_nop_();_nop_();//从高拉倒低
88 DQ=0;
89 delay_DS18B20(50); //550 us
90 DQ=1;
91 delay_DS18B20(6); //66 us
92 presence=DQ; //presence=0 复位成功,继续下一步
93 }
94 delay_DS18B20(45); //延时500 us
95 presence=~DQ;
96 }
97 DQ=1; //拉高电平
98 }
99 //写DS一个字节数据----------
100 void WriteByte_DS18B20(unsigned char val)
101 {
102 unsigned char i;
103 for(i=8;i>0;i--)
104 {
105 DQ=1;_nop_();_nop_(); //从高拉倒低
106 DQ=0;_nop_();_nop_();_nop_();_nop_(); //5 us
107 DQ=val&0x01; //最低位移出
108 delay_DS18B20(6); //66 us
109 val=val/2; //右移1位
110 }
111 DQ=1;
112 delay_DS18B20(1);
113 }
114 //读DS一个字节数据---------
115 unsigned char ReadByte_DS18B20()
116 {
117 unsigned char i;
118 unsigned char byte=0;
119 for(i=8;i>0;i--)
120 {
121 DQ=1;_nop_();_nop_();
122 byte>>=1;
123 DQ=0;_nop_();_nop_();_nop_();_nop_(); //4 us
124 DQ=1;_nop_();_nop_();_nop_();_nop_(); //4 us
125 if(DQ)byte|=0x80;
126 delay_DS18B20(6); //66 us
127 }
128 DQ=1;
129 return(byte);
130 }
131 //让DS18B20测量一次温度,并将测量结果存放在其内部RAM----------
132 void MeasureTemp_DS18B20()
133 {
134 Reset_DS18B20();
135 delay_DS18B20(200);
136 WriteByte_DS18B20(0xcc); //发送无条件选中命令,选中总线上仅有的DS18B20从设备
137 WriteByte_DS18B20(0x44); //温度转换命令
138
139 }
140 //向DS18B20请求读取温度值--------------
141 void ReadTemperature_DS18B20()
142 {
143 Reset_DS18B20();
144 delay_DS18B20(1);
145 WriteByte_DS18B20(0xcc); //发送无条件选中命令,选中总线上仅有的DS18B20从设备
146 WriteByte_DS18B20(0xbe); //发送读取温度命令
147 }
148 //获取并返回DS18B20内部温度测量值--------
149 int GetTemperature_DS18B20()
150 {
151 int temp=0;
152 unsigned char temperature_H,temperature_L; //需要连续读取2个字节数据并进行处理,才能得出一次温度值
153 MeasureTemp_DS18B20(); //先写入转换命令
154 ReadTemperature_DS18B20(); //然后等待转换完后发送读取温度命令
155 temperature_L=ReadByte_DS18B20(); //读取温度值共16位,先读低字节
156 temperature_H=ReadByte_DS18B20(); //再读高字节
157 temp=temperature_H;
158 temp<<=8;
159 temp|=temperature_L;
160 if(temp<0) //当温度值为负数(高5位为符号位)
161 {
162 temp=~temp;
163 temp=temp+1;
164 temp=0.0625*temp*100+0.5; //temp*100 意味着取2位小数, +0.5 意味着四舍五入
165 Temp_Symbol=1;
166 }
167 else //当温度值为正数
168 {
169 temp=0.0625*temp*100