AirVOC_1000.lua 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. --[[
  2. @module AirVOC_1000
  3. @summary AirVOC_1000应用功能模块
  4. @version 1.0
  5. @date 2025.10.21
  6. @author 沈园园
  7. @usage
  8. 本文件为AirVOC_1000驱动配置文件,核心业务逻辑为:
  9. 1、打开AirVOC_1000;
  10. 2、读取TVOC的空气质量等级;
  11. 本文件没有对外接口,直接require "AirVOC_1000"就可以加载运行;
  12. ]]
  13. --本文件中的主机是指I2C主机,具体指Air8000
  14. --本文件中的从机是指I2C从机,具体指AirVOC_1000配件板上的ags02ma VOC(挥发性有机化合物)气体传感器芯片
  15. local AirVOC_1000 =
  16. {
  17. -- i2c_id:主机的i2c id;
  18. }
  19. -- 从机地址为0x1A
  20. local slave_address = 0x1A
  21. -- TVOC数据的寄存器地址
  22. local DATA_REG_ADDR = 0x00
  23. -- TVOC数据的长度
  24. local DATA_REG_LEN = 0x05
  25. -- 计算数据表data中所有数据元素的crc8校验值
  26. local function crc8(data)
  27. local crc = 0xFF
  28. for i = 1, #data do
  29. crc = bit.bxor(crc, data[i])
  30. for j = 1, 8 do
  31. crc = crc * 2
  32. if crc >= 0x100 then
  33. crc = bit.band(bit.bxor(crc, 0x31), 0xff)
  34. end
  35. end
  36. end
  37. return crc
  38. end
  39. -- 读取AirVOC_1000的寄存器中指定长度的数据
  40. --reg:number类型;
  41. -- 表示AirVOC_1000上的寄存器地址;
  42. -- 必须传入,不允许为空;
  43. --返回值:失败返回false;成功返回读取到的指定长度的数据(string类型)
  44. local function read_register(reg, len)
  45. -- 发送寄存器地址
  46. i2c.send(AirVOC_1000.i2c_id, slave_address, reg)
  47. -- sys.wait(20)
  48. -- 读取应答的数据
  49. local data = i2c.recv(AirVOC_1000.i2c_id, slave_address, len)
  50. -- log.info("read_register", data:toHex())
  51. -- 读取到的数据为指定的长度,则表示读取成功
  52. -- 否则读取失败
  53. if type(data)=="string" and data:len()==len then
  54. return data
  55. else
  56. log.error("AirVOC_1000 read_register error", type(data), type(data)=="string" and data:len() or "invalid type", len)
  57. return false
  58. end
  59. end
  60. --打开AirVOC_1000;
  61. --i2c_id:number类型;
  62. -- 主机使用的I2C ID,用来控制AirVOC_1000;
  63. -- 取值范围:仅支持0和1;
  64. -- 如果没有传入此参数,则默认为0;
  65. --返回值:成功返回true,失败返回false
  66. function AirVOC_1000.open(i2c_id)
  67. --如果i2c_id为nil,则赋值为默认值0
  68. if i2c_id==nil then i2c_id=0 end
  69. --检查参数的合法性
  70. if not (i2c_id == 0 or i2c_id == 1) then
  71. log.error("AirVOC_1000.open", "invalid i2c_id", i2c_id)
  72. return false
  73. end
  74. AirVOC_1000.i2c_id = i2c_id
  75. --初始化I2C
  76. if i2c.setup(i2c_id, i2c.FAST) ~= 1 then
  77. log.error("AirVOC_1000.open", "i2c.setup error", i2c_id)
  78. return false
  79. end
  80. return true
  81. end
  82. --读取TVOC的ppb值;
  83. --ppb: 代表 parts per billion,即 十亿分之一。 1 ppb TVOC 表示在每 10 亿个体积单位的空气中,含有 1 个体积单位的 TVOC
  84. --返回值:失败返回false;
  85. -- 成功返回ppb值(number类型)
  86. function AirVOC_1000.get_ppb()
  87. --从寄存器DATA_REG_ADDR中读取DATA_REG_LEN长度的数据
  88. local raw = read_register(DATA_REG_ADDR, DATA_REG_LEN)
  89. --读取数据出错
  90. if not raw then
  91. log.error("AirVOC_1000.get_ppb", "read_register error")
  92. return false
  93. end
  94. --检查校验值
  95. if crc8({raw:byte(1), raw:byte(2), raw:byte(3), raw:byte(4)}) ~= raw:byte(5) then
  96. log.error("AirVOC_1000.get_ppb", "crc error")
  97. return false
  98. end
  99. --解析数据: 大端格式
  100. local tvoc = (raw:byte(2) << 16) |
  101. (raw:byte(3) << 8) |
  102. raw:byte(4)
  103. return tvoc
  104. end
  105. --读取TVOC的ppm值;
  106. --ppm: 代表 parts per million,即 百万分之一。 1 ppm TVOC 表示在每 100 万个体积单位的空气中,含有 1 个体积单位的 TVOC
  107. --返回值:失败返回false;
  108. -- 成功返回ppb值(number类型)
  109. function AirVOC_1000.get_ppm()
  110. --读取ppb值
  111. local ppb = AirVOC_1000.get_ppb()
  112. --如果ppb读取失败
  113. if not ppb then
  114. log.error("AirVOC_1000.get_ppm", "get_ppb error")
  115. return false
  116. end
  117. --ppb = ppm*1000
  118. return ppb/1000
  119. end
  120. --读取TVOC的空气质量等级;
  121. -- 根据 TVOC 浓度(通常以 ppb 或 ppm 表示)划分的等级,用于评估室内空气质量的优劣和对人体健康的潜在风险;
  122. -- 不同国家、地区或机构的标准可能略有差异,但核心划分逻辑相似:浓度越低,等级越好,风险越低。
  123. --返回值:失败返回false;
  124. -- 成功返回空气质量等级值(number类型,数值越小,表示空气质量越好)和空气质量描述(string类型,例如优、良好、轻度污染、中度污染、重度污染)
  125. -- 1:表示优
  126. -- 2:表示良好
  127. -- 3:表示轻度污染
  128. -- 4:表示中度污染
  129. -- 5:表示重度污染
  130. function AirVOC_1000.get_quality_level()
  131. --读取ppb值
  132. local ppb = AirVOC_1000.get_ppb()
  133. --如果ppb读取失败
  134. if not ppb then
  135. log.error("AirVOC_1000.get_qulity_level", "get_ppb error")
  136. return false
  137. end
  138. --根据ppb值计算空气质量等级
  139. if ppb < 200 then
  140. return 1,"优"
  141. elseif ppb < 1000 then
  142. return 2,"良好"
  143. elseif ppb < 3000 then
  144. return 3,"轻度污染"
  145. elseif ppb < 5000 then
  146. return 4,"中度污染"
  147. else
  148. return 5,"重度污染"
  149. end
  150. end
  151. --关闭AirVOC_1000
  152. --返回值:成功返回true,失败返回false
  153. function AirVOC_1000.close()
  154. --close接口没有返回值,理论上不会关闭失败
  155. i2c.close(AirVOC_1000.i2c_id)
  156. return true
  157. end
  158. return AirVOC_1000