io_module.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. from modbus import Modbus, MBError, NoResponseError
  2. import colorama
  3. from colorama import Fore
  4. import time
  5. import os
  6. reg_table = {'in_bits': 0x0100, 'in_cnt': 0x0102, 'in_mode': 0x0120, 'in_norm': 0x0122, 'in_deb_start': 0x124,
  7. 'out_cur': 0x0200, 'out_mode': 0x0202, 'out_mode_save': 0x0203, 'pwm_duty': 0x0210,
  8. 'pwm_duty_save': 0x0220, 'pwm_per': 0x0230, 'pwm_per_save': 0x0240,
  9. 'rtc_unix': 0x0802, 'rtc_sinhro': 0x0804, 'uptime': 0x0800,}
  10. class IO_Module(Modbus):
  11. def __init__(self, tty: str, brate: int, address: int):
  12. super().__init__(tty, brate, address)
  13. self.update_segment_number = 0
  14. def iap_start(self):
  15. """Reboot device in IAP mode"""
  16. request = bytes((self.address, 0x41, 0x01))
  17. response = self.raw_communicate(request + self._crc(request))
  18. def write_fw_start(self, size: int):
  19. """Ask device to start update"""
  20. self.update_segment_number = 0
  21. request = bytes((self.address, 0x41, 0x01, 0xEF, 0xBE, 0xAD, 0xDE)) + size.to_bytes(4, 'big')
  22. response = self.raw_communicate(request + self._crc(request), 5)
  23. if len(response) == 0:
  24. raise NoResponseError('No response on WRITE_START command')
  25. if len(response) != 5:
  26. raise MBError('Incorrect response length')
  27. if response[:3] != bytes((self.address, 0x41, 0x01)):
  28. raise MBError('Incorrect response')
  29. def write_fw_part(self, data: bytes):
  30. """Write piece of FW data in IAP mode"""
  31. header = bytes((self.address, 0x41, 0x02))
  32. request = b''.join((header, self.update_segment_number.to_bytes(2, 'big'), data))
  33. response = self.raw_communicate(request + self._crc(request), 5)
  34. # self.print_hex(response)
  35. if len(response) != 5:
  36. raise MBError('Incorrect response length')
  37. if (response[:3]) != header:
  38. raise MBError('Incorrect response')
  39. self.update_segment_number += 1
  40. def iap_finish(self):
  41. """Complete FW transmission and check response"""
  42. header = request = bytes((self.address, 0x41, 0x03))
  43. response = self.raw_communicate(request + self._crc(request), 5)
  44. if len(response) != 5:
  45. raise MBError('Incorrect response length')
  46. if response[:3] != header:
  47. raise MBError('Incorrect response')
  48. def update(self, path):
  49. self.MB_TIMEOUT = 3
  50. size = os.path.getsize('fw.bin')
  51. print('Switch to IAP mode')
  52. self.iap_start()
  53. time.sleep(4)
  54. print(f'Start writing {size} bytes of FW')
  55. self.write_fw_start(size)
  56. time.sleep(2)
  57. print(f'Open FW file "{path}"...')
  58. with open(path, 'rb') as f:
  59. done = progress_cur = progress_pre = 0
  60. while True:
  61. buf = f.read(128)
  62. if len(buf):
  63. self.write_fw_part(buf)
  64. progress_cur = done / size
  65. else:
  66. break
  67. print('End of transmission')
  68. self.iap_finish()
  69. # 0x0100 - текущее состояние входов
  70. def get_inputs_bit(self) -> str:
  71. data = self.read_holding_registers(reg_table['in_bits'], 1)
  72. return format(data[0], '08b')
  73. # 0x0101 - 0x0110 Счетчики импульсов
  74. def get_inputs_counters(self):
  75. data = []
  76. for i in range(reg_table['in_cnt'], reg_table['in_cnt'] + 16, 2):
  77. data.append(self.read_uint32_holding(i))
  78. return data
  79. # 0x0120 - режим работы входов
  80. def get_inputs_mode(self):
  81. data = self.read_holding_registers(reg_table['in_mode'], 1)
  82. return format(data[0], '08b')
  83. def set_inputs_mode(self, val):
  84. self.write_holding_register(reg_table['in_mode'], val)
  85. #
  86. def set_input_mode(self, input, val):
  87. ret = self.read_holding_registers(reg_table['in_mode'], 1)
  88. if val == 1:
  89. data = ret[0] | (0b1 << (input - 1))
  90. else:
  91. data = ret[0] & ~(0b1 << (input - 1))
  92. self.set_inputs_mode(data)
  93. # 0x0122 - нормальное состояние входов
  94. def get_inputs_norm_state(self):
  95. data = self.read_holding_registers(reg_table['in_norm'], 1)
  96. return format(data[0], '08b')
  97. def set_inputs_norm_state(self, val):
  98. self.write_holding_register(reg_table['in_norm'], val)
  99. # 0x0124 - время антидребезга (ms)
  100. def get_debounce_channel(self, input):
  101. data = self.read_holding_registers(reg_table['in_deb_start'] + input - 1, 1)
  102. return data[0]
  103. def get_debounce_channels(self):
  104. return self.read_holding_registers(reg_table['in_deb_start'], 8)
  105. def set_debounce_channel(self, input, val):
  106. self.write_holding_register(reg_table['in_deb_start'] + input - 1, val)
  107. # 0x0200 - текущее состояние выходов в обычно режиме
  108. def get_outputs(self):
  109. data = self.read_holding_registers(reg_table['out_cur'], 1)
  110. return format(data[0], '08b')
  111. # 0x0200 - текущее состояние выходов в обычно режиме
  112. def set_outputs(self, val):
  113. self.write_holding_register(reg_table['out_cur'], val)
  114. def set_output(self, output, val):
  115. ret = self.read_holding_registers(reg_table['out_cur'], 1)
  116. if val == 1:
  117. data = ret[0] | (0b1 << (output - 1))
  118. else:
  119. data = ret[0] & ~(0b1 << (output - 1))
  120. self.set_outputs(data)
  121. # 0x0202 - режим работы выходов; 0 - обычный, 1 - ШИМ
  122. def get_outputs_mode(self):
  123. data = self.read_holding_registers(reg_table['out_mode'], 1)
  124. return format(data[0], '08b')
  125. def set_outputs_mode(self, val):
  126. self.write_holding_register(reg_table['out_mode'], val)
  127. def set_output_mode(self, output, val):
  128. ret = self.read_holding_registers(reg_table['out_mode'], 1)
  129. if val == 1:
  130. data = ret[0] | (0b1 << (output - 1))
  131. else:
  132. data = ret[0] & ~(0b1 << (output - 1))
  133. self.set_outputs_mode(data)
  134. # 0x0203 - состояние выходов (режим обычного выхода) в безопасном режиме работы
  135. def get_outputs_mode_save(self):
  136. data = self.read_holding_registers(reg_table['out_mode_save'], 1)
  137. return format(data[0], '08b')
  138. def set_outputs_mode_save(self, val):
  139. self.write_holding_register(reg_table['out_mode_save'], val)
  140. def set_output_mode_save(self, output, val):
  141. ret = self.read_holding_registers(reg_table['out_mode_save'], 1)
  142. if val == 1:
  143. data = ret[0] | (0b1 << (output - 1))
  144. else:
  145. data = ret[0] & ~(0b1 << (output - 1))
  146. self.set_outputs_mode_save(data)
  147. # 0x0210 - заполнение PWM (%)
  148. def get_pwm_duty(self, output):
  149. data = self.read_holding_registers(reg_table['pwm_duty'] + output - 1, 1)
  150. return data[0]
  151. def get_pwm_duty_all(self):
  152. return self.read_holding_registers(reg_table['pwm_duty'], 8)
  153. def set_pwm_duty(self, output, val):
  154. self.write_holding_register(reg_table['pwm_duty'] + output - 1, val)
  155. # 0x0220 - заполнение PWM (%)
  156. def get_pwm_duty_all_save(self):
  157. return self.read_holding_registers(reg_table['pwm_duty_save'], 8)
  158. # 0x0230 - период PWM (0.1 сек)
  159. def get_pwm_period_all(self):
  160. return self.read_holding_registers(reg_table['pwm_per'], 8)
  161. # 0x0240 - период PWM в безопасном режиме (0.1 сек)
  162. def get_pwm_period_all_save(self):
  163. return self.read_holding_registers(reg_table['pwm_per_save'], 8)
  164. def get_uptime(self):
  165. return self.read_uint32_holding(reg_table['uptime'])
  166. def get_rtc(self):
  167. return self.read_uint32_holding(reg_table['rtc_unix'])
  168. def set_rtc(self, utc):
  169. self.write_uint32(reg_table['rtc_sinhro'], utc)
  170. def main():
  171. colorama.init(autoreset=True)
  172. dev = IO_Module('COM24', 115200, 14)
  173. dev.MB_DEBUG = False
  174. trigger = 0
  175. # dev.update('fw.bin')
  176. # Запрос системных параметров, установка времени
  177. # print('Device uptime:', dev.get_uptime())
  178. # unix_time = dev.get_rtc()
  179. # print(f'RTC: {time.ctime(unix_time)}. Unix time stamp: {unix_time}')
  180. # print('Set time:', int(time.time()))
  181. # dev.set_rtc(int(time.time()))
  182. # time.sleep(1)
  183. # unix_time = dev.get_rtc()
  184. # print(f'RTC: {time.ctime(unix_time)}. Unix time stamp: {unix_time}')
  185. for i in range(1, 9):
  186. dev.set_input_mode(i, 1)
  187. # print('Inputs mode [bit field] :', Fore.GREEN + dev.get_inputs_mode())
  188. for i in range(1, 9):
  189. dev.set_debounce_channel(i, 50 + i)
  190. dev.set_output(i, 0)
  191. # 0x0203 - состояние выходов (режим обычного выхода) в безопасном режиме работы
  192. for i in range(1, 9):
  193. dev.set_output_mode_save(i, 0)
  194. # 0x0210 - заполнение PWM (ms)
  195. for i in range(1, 9):
  196. dev.set_pwm_duty(i, 10 + i)
  197. # Установить 1-ый выход в режим PWM
  198. # dev.set_output_mode(1, 0)
  199. # Установить нормальное состояние входов
  200. # dev.set_inputs_norm_state(0b00110101)
  201. # print(dev.get_pwm_duty_all())
  202. while True:
  203. print('~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~')
  204. # Значения входов (битовое поле)
  205. print('Inputs values [bit field] :', Fore.GREEN + dev.get_inputs_bit())
  206. # Режим работы входов (битовое поле)
  207. print('Inputs mode [bit field] :', Fore.GREEN + dev.get_inputs_mode())
  208. # Нормальное состояние входов (битовое поле)
  209. print('Inputs norm [bit field] :', Fore.GREEN + dev.get_inputs_norm_state())
  210. # Период антидребезга (ms)
  211. print('Debounce input (ms) :', Fore.GREEN + ' | '.join(str(el) for el in dev.get_debounce_channels()))
  212. # Значение счетчиков
  213. data = dev.get_inputs_counters()
  214. print('Inputs counters :', Fore.GREEN + ' | '.join(str(el) for el in data))
  215. # Текущее состояние выходов в обычном режиме
  216. print('Outputs norm [bit field] :', Fore.GREEN + dev.get_outputs())
  217. # Состояние выходов в безопасном режиме
  218. print('Outputs save [bit field] :', Fore.GREEN + dev.get_outputs_mode_save())
  219. # Режим работы выходов
  220. print('Outputs mode [bit field] :', Fore.GREEN + dev.get_outputs_mode())
  221. # 0x0210 - заполнение PWM (ms)
  222. print('PWM duty cycle [%] :', Fore.GREEN + ' | '.join(str(el) for el in dev.get_pwm_duty_all()))
  223. # 0x0220 - заполнение PWM в безопасном режиме (ms)
  224. print('PWM duty cycle (save) [%] :', Fore.GREEN + ' | '.join(str(el) for el in dev.get_pwm_duty_all_save()))
  225. # 0x0230 - период PWM (0.1 сек)
  226. print('PWM period [0.1 sec] :', Fore.GREEN + ' | '.join(str(el) for el in dev.get_pwm_period_all()))
  227. # 0x0240 - период PWM в безопасном режиме (0.1 сек)
  228. print('PWM period save [0.1 sec] :', Fore.GREEN + ' | '.join(str(el) for el in dev.get_pwm_period_all_save()))
  229. # Дергаем одним выходом
  230. # dev.set_output(2, trigger)
  231. # trigger = not trigger
  232. # # Для проверки выходов в обычном режиме
  233. # for i in range(1, 9):
  234. # dev.set_output(i, 1)
  235. # print('Outputs norm [bit field] :', Fore.GREEN + dev.get_output())
  236. # time.sleep(0.1)
  237. # for i in range(1, 9):
  238. # dev.set_output(i, 0)
  239. # print('Outputs norm [bit field] :', Fore.GREEN + dev.get_output())
  240. # time.sleep(0.1)
  241. # # Режим работы выходов
  242. # for i in range(1, 9):
  243. # dev.set_output_mode(i, 1)
  244. # print('Outputs mode [bit field] :', Fore.GREEN + dev.get_outputs_mode())
  245. # time.sleep(0.1)
  246. # for i in range(1, 9):
  247. # dev.set_output_mode(i, 0)
  248. # print('Outputs mode [bit field] :', Fore.GREEN + dev.get_outputs_mode())
  249. # time.sleep(0.1)
  250. # break
  251. time.sleep(1)
  252. # for i in range(1, 9):
  253. # dev.set_input_mode(i, 1)
  254. # print('Inputs mode [bit field] :', Fore.GREEN + dev.get_inputs_mode())
  255. # for i in range(8, 0, -1):
  256. # dev.set_input_mode(i, 0)
  257. # print('Inputs mode [bit field] :', Fore.GREEN + dev.get_inputs_mode())
  258. if __name__ == '__main__':
  259. main()