找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3049|回复: 0
收起左侧

18B20达到 0.01精度的汇编程序

[复制链接]
ID:80436 发表于 2015-5-21 23:57 | 显示全部楼层 |阅读模式
本帖最后由 liuqq 于 2015-5-22 00:02 编辑


  1. ;温度传感器18B20汇编程序,采用器件默认的12位转化,最大转化时间750微秒
  2. ;将温度数据通过串口发送出去,波特率2400
  3. ;本程序专为AT89C51实验开发板编写.适合12晶振
  4. ;本程序经过验证,可以显示温度+/-和两位整数温度和两位小数温度数据

  5. DOT EQU 30H
  6. ZHENGSHU EQU 31H
  7. FLAG1 EQU 38H ;是否检测到DS18B20的标志位
  8. ;定义温度数据
  9. DIS_1 EQU 32H ;符号
  10. DIS_2 EQU 33H ;十位
  11. DIS_3 EQU 34H ;个位
  12. DIS_4 EQU 35H ;小数点后第一位
  13. DIS_5 EQU 36H ;小数点后第二位
  14. WDDATA BIT P2.2 ;定义DS18B20的数据脚为P2.2端口

  15. ORG 0000H

  16. ;以下为主程序进行CPU中断方式设置
  17. CLR EA ;关闭总中断
  18. MOV SCON,#50H ;设置成串口1方式
  19. MOV TMOD,#20H ;波特率发生器T1工作在模式2上
  20. MOV TH1,#0F3H ;预置初值(按照波特率2400BPS预置初值)
  21. MOV TL1,#0F3H ;预置初值(按照波特率2400BPS预置初值)
  22. SETB TR1 ;启动定时器T1
  23. ;以上完成串口2400通讯初始化设置

  24. ;-------------------------
  25. ; 主程序
  26. ;-------------------------
  27. MAIN:
  28. LCALL INIT_1820 ;调用复位DS18B20子程序
  29. MAIN1:
  30. LCALL GET_TEMPER;调用读温度子程序
  31. LCALL FORMULA ;通过公式计算,小数点后显示两位
  32. LCALL BCD
  33. LCALL DISPLAY ;调用串口显示子程序
  34. LCALL DELAY500 ;延时0.5秒
  35. LCALL DELAY500 ;延时0.5秒
  36. LCALL DELAY500 ;延时0.5秒
  37. AJMP MAIN1

  38. ;-------------------------
  39. ; DS18B20复位初始化程序
  40. ;-------------------------
  41. INIT_1820:
  42. SETB WDDATA
  43. NOP
  44. CLR WDDATA
  45. ;主机发出延时540微秒的复位低脉冲
  46. MOV R0,#36
  47. LCALL DELAY
  48. SETB WDDATA;然后拉高数据线
  49. NOP
  50. NOP
  51. MOV R0,#36
  52. TSR2:
  53. JNB WDDATA,TSR3;等待DS18B20回应
  54. DJNZ R0,TSR2
  55. LJMP TSR4 ; 延时
  56. TSR3:
  57. SETB FLAG1 ; 置标志位,表示DS1820存在
  58. LJMP TSR5
  59. TSR4:
  60. CLR FLAG1 ; 清标志位,表示DS1820不存在
  61. LJMP TSR7
  62. TSR5:
  63. MOV R0,#06BH
  64. TSR6:
  65. DJNZ R0,TSR6 ;复位成功!时序要求延时一段时间
  66. TSR7:
  67. SETB WDDATA
  68. RET
  69. ;-------------------
  70. ; 读出转换后的温度值
  71. ;-------------------
  72. GET_TEMPER:
  73. SETB WDDATA ; 定时入口
  74. LCALL INIT_1820 ;先复位DS18B20
  75. JB FLAG1,TSS2
  76. RET ; 判断DS1820是否存在?若DS18B20不存在则返回
  77. TSS2:
  78. MOV A,#0CCH ; 跳过ROM匹配
  79. LCALL WRITE_1820
  80. MOV A,#44H ; 发出温度转换命令
  81. LCALL WRITE_1820

  82. MOV R0,#50 ;等待AD转换结束,12位的话750微秒.
  83. LCALL DELAY
  84. LCALL INIT_1820 ;准备读温度前先复位
  85. MOV A,#0CCH ; 跳过ROM匹配
  86. LCALL WRITE_1820
  87. MOV A,#0BEH ; 发出读温度命令
  88. LCALL WRITE_1820
  89. LCALL READ_18200; 将读出的九个字节数据保存到60H-68H
  90. RET

  91. ;----------------------------------
  92. ;写DS18B20的子程序(有具体的时序要求)
  93. ;----------------------------------
  94. WRITE_1820:
  95. MOV R2,#8 ;一共8位数据
  96. CLR C
  97. WR1:
  98. CLR WDDATA
  99. MOV R3,#6
  100. DJNZ R3,$
  101. RRC A
  102. MOV WDDATA,C
  103. MOV R3,#24
  104. DJNZ R3,$
  105. SETB WDDATA
  106. NOP
  107. DJNZ R2,WR1
  108. SETB WDDATA
  109. RET
  110. ;--------------------------------------------------
  111. ; 读DS18B20的程序,从DS18B20中读出九个字节的数据
  112. ;--------------------------------------------------
  113. READ_18200:
  114. MOV R4,#9
  115. MOV R1,#60H ; 存入60H开始的九个单元
  116. RE00:
  117. MOV R2,#8
  118. RE01:
  119. CLR C
  120. SETB WDDATA
  121. NOP
  122. NOP
  123. CLR WDDATA
  124. NOP
  125. NOP
  126. NOP
  127. SETB WDDATA
  128. MOV R3,#09
  129. RE10:
  130. DJNZ R3,RE10
  131. MOV C,WDDATA
  132. MOV R3,#23
  133. RE20:
  134. DJNZ R3,RE20
  135. RRC A
  136. DJNZ R2,RE01
  137. MOV @R1,A
  138. INC R1
  139. DJNZ R4,RE00
  140. RET

  141. ;------------------------
  142. ;温度计算子程序
  143. ;------------------------
  144. FORMULA: ; 按公式:T实际=(T整数-0.25)+( M每度-M剩余)/ M每度
  145. ;计算出实际温度,整数部分和小数部分分别存于ZHENGSHU单元和DOT单元

  146. ;将61H中的低4位移入60H中的高4位,得到温度的整数部分,并存于ZHENGSHU单元
  147. MOV 29H,61H
  148. MOV A,60H
  149. MOV C,48H
  150. RRC A
  151. MOV C,49H
  152. RRC A
  153. MOV C,4AH
  154. RRC A
  155. MOV C,4BH
  156. RRC A
  157. MOV ZHENGSHU,A

  158. ; ( M每度-M剩余)/ M每度,小数值存于A中
  159. MOV A,67h
  160. SUBB A,66h
  161. MOV B,#64H
  162. MUL AB
  163. MOV R4,B
  164. MOV R5,A
  165. MOV R7,67H
  166. LCALL DIV457
  167. MOV A,R3


  168. ;再减去0.25,实际应用中减去25
  169. SUBB A,#19H
  170. MOV DOT,A ;小数部分存于DOT中
  171. MOV A,ZHENGSHU
  172. SUBB A,#00H ;整数部分减去来自小数部分的借位
  173. MOV ZHENGSHU,A
  174. MOV C,4BH
  175. JNC ZHENG ;是否为负数
  176. CPL A
  177. INC A
  178. MOV DIS_1,#2DH ; 零度以下时,第一位显示"-"号
  179. MOV ZHENGSHU,A
  180. ZHENG:
  181. MOV DIS_1,#2BH ; 零度以上时,第一位显示"+"号
  182. RET
  183. ;------------------------
  184. ;双字节除以单字节子程序
  185. ;------------------------
  186. DIV457: CLR C
  187. MOV A,R4
  188. SUBB A,R7
  189. JC DV50
  190. SETB OV ;商溢出
  191. RET
  192. DV50: MOV R6,#8 ;求平均值(R4R5/R7-→R3)
  193. DV51: MOV A,R5
  194. RLC A
  195. MOV R5,A
  196. MOV A,R4
  197. RLC A
  198. MOV R4,A
  199. MOV F0,C
  200. CLR C
  201. SUBB A,R7
  202. ANL C,/F0
  203. JC DV52
  204. MOV R4,A
  205. DV52: CPL C
  206. MOV A,R3
  207. RLC A
  208. MOV R3,A
  209. DJNZ R6,DV51
  210. MOV A,R4 ;四舍五入
  211. ADD A,R4
  212. JC DV53
  213. SUBB A,R7
  214. JC DV54
  215. DV53: INC R3
  216. DV54: CLR OV
  217. RET
  218. ;---------------------
  219. ;转换成非压缩的BCD码
  220. ;---------------------
  221. BCD: MOV A,ZHENGSHU
  222. MOV B,#0AH
  223. DIV AB
  224. ORL A,#00110000B ;转换成ASCII码
  225. MOV DIS_2,A
  226. MOV DIS_3,B
  227. MOV A,DIS_3
  228. ORL A,#00110000B ;转换成ASCII码
  229. MOV DIS_3,A

  230. MOV A,DOT
  231. MOV B,#0AH
  232. DIV AB
  233. ORL A,#00110000B ;转换成ASCII码
  234. MOV DIS_4,A
  235. MOV DIS_5,B
  236. MOV A,DIS_5
  237. ORL A,#00110000B ;转换成ASCII码
  238. MOV DIS_5,A

  239. RET
  240. ;----------------------
  241. ;串口显示数据子程序
  242. ;----------------------
  243. DISPLAY:

  244. CLR TI
  245. MOV A,DIS_1
  246. MOV SBUF,A
  247. JNB TI,$ ;发送给PC,通过串口调试助手显示+/-

  248. CLR TI
  249. MOV A,DIS_2
  250. MOV SBUF,A
  251. JNB TI,$ ;发送给PC,通过串口调试助手显示整数第一位

  252. CLR TI
  253. MOV A,DIS_3
  254. MOV SBUF,A
  255. JNB TI,$ ;发送给PC,通过串口调试助手显示整数第二位

  256. CLR TI
  257. MOV A,#2EH
  258. MOV SBUF,A
  259. JNB TI,$ ;发送给PC,通过串口调试助手显示小数点

  260. CLR TI
  261. MOV A,DIS_4
  262. MOV SBUF,A
  263. JNB TI,$ ;发送给PC,通过串口调试助手显示小数第一位

  264. CLR TI
  265. MOV A,DIS_5
  266. MOV SBUF,A
  267. JNB TI,$ ;发送给PC,通过串口调试助手显示小数第一位


  268. CLR TI
  269. MOV A,#0DH;换行
  270. MOV SBUF,A
  271. JNB TI,$ ;发送给PC,通过串口调试助手显示

  272. CLR TI
  273. MOV A,#0AH;换行
  274. MOV SBUF,A
  275. JNB TI,$ ;发送给PC,通过串口调试助手显示

  276. RET

  277. ;----------------------
  278. ;延时子程序
  279. ;----------------------
  280. ;为保证DS18B20的严格I/O时序,需要做较精确的延时
  281. ;在DS18B20操作中,用到的延时有15 μs,90 μs,270 μs,540 μs
  282. ;因这些延时均为15 μs的整数倍,因此可编写一个DELAY15(n)函数
  283. DELAY: ;11.05962M晶振
  284. LOOP: MOV R1,#06H
  285. LOOP1: DJNZ R1,LOOP1
  286. DJNZ R0,LOOP
  287. RET

  288. ;500毫秒延时子程序,占用R4、R5
  289. DELAY500:MOV R4,#248
  290. DA222:MOV R5,#248
  291. DJNZ R5,$
  292. DJNZ R4,DA222
  293. RET

  294. END


复制代码

回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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