digital_io.py 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. from io_module import IO_Module
  2. from modbus import Modbus, MBError, NoResponseError
  3. from log_reader import DigitalLogReader
  4. import colorama
  5. from colorama import Fore
  6. from time import sleep
  7. import time
  8. from datetime import datetime, timedelta, timezone
  9. from mb_registers import DIO_REGS, DioReg
  10. from serial import Serial
  11. class IO_Digital(IO_Module):
  12. def __init__(self, modbus: Modbus):
  13. self.modbus = modbus
  14. super().__init__(self.modbus)
  15. self.log = DigitalLogReader(self.modbus)
  16. '''Запрос параметров'''
  17. # 0x0100 - Состояние входов (вкл./выкл.)
  18. def get_inputs_state(self) -> str:
  19. data = self.modbus.read_holding_registers(DioReg.IN_STATE.value , 1)
  20. return format(data[0], '08b')
  21. # 0x0101 - текущее состояние входов
  22. def get_inputs_bit(self) -> str:
  23. data = self.modbus.read_holding_registers(DioReg.IN_BITS.value , 1)
  24. return format(data[0], '08b')
  25. # 0x0101 - 0x0110 Счетчики импульсов
  26. def get_inputs_counters(self):
  27. data = []
  28. for i in range(DIO_REGS['in_cnt'], DIO_REGS['in_cnt'] + 16, 2):
  29. data.append(self.modbus.read_uint32_holding(i))
  30. return data
  31. # 0x0101 - 0x0110 Счетчики импульсов
  32. def get_inputs_counters(self):
  33. data = []
  34. for i in range(DIO_REGS['in_cnt'], DIO_REGS['in_cnt'] + 16, 2):
  35. data.append(self.modbus.read_uint32_holding(i))
  36. return data
  37. # 0x0120 - режим работы входов
  38. def get_inputs_mode(self):
  39. data = self.modbus.read_holding_registers(DIO_REGS['in_mode'], 1)
  40. return format(data[0], '08b')
  41. # 0x0122 - нормальное состояние входов
  42. def get_inputs_norm_state(self):
  43. data = self.modbus.read_holding_registers(DIO_REGS['in_norm'], 1)
  44. return format(data[0], '08b')
  45. # 0x0124 - время антидребезга (ms)
  46. def get_debounce_channel(self, input):
  47. data = self.modbus.read_holding_registers(DIO_REGS['in_deb_start'] + input - 1, 1)
  48. return data[0]
  49. # 0x0130
  50. def get_loads_bit(self):
  51. data = self.modbus.read_holding_registers(DIO_REGS['load_bits'], 1)
  52. return format(data[0], '08b')
  53. # 0x0131
  54. def get_cred_bit(self):
  55. data = self.modbus.read_holding_registers(DIO_REGS['cred_bits'], 1)
  56. return format(data[0], '08b')
  57. def get_debounce_channels(self):
  58. return self.modbus.read_holding_registers(DIO_REGS['in_deb_start'], 8)
  59. '''Установка параметров'''
  60. # 0x0100 - Состояние входов (вкл./выкл.)
  61. def set_inputs_state(self, val):
  62. self.modbus.write_holding_register(DioReg.IN_STATE.value, val)
  63. #
  64. def set_inputs_mode(self, val):
  65. self.modbus.write_holding_register(DIO_REGS['in_mode'], val)
  66. #
  67. def set_input_mode(self, input, val):
  68. ret = self.modbus.read_holding_registers(DIO_REGS['in_mode'], 1)
  69. if val == 1:
  70. data = ret[0] | (0b1 << (input - 1))
  71. else:
  72. data = ret[0] & ~(0b1 << (input - 1))
  73. self.set_inputs_mode(data)
  74. # 0x0200 - текущее состояние выходов в обычно режиме
  75. def set_outputs(self, val):
  76. self.modbus.write_holding_register(DIO_REGS['out_cur'], val)
  77. # Установить значение на конкретном выходе [1..8]
  78. def set_output(self, output, val):
  79. ret = self.modbus.read_holding_registers(DIO_REGS['out_cur'], 1)
  80. if val == 1:
  81. data = ret[0] | (0b1 << (output - 1))
  82. else:
  83. data = ret[0] & ~(0b1 << (output - 1))
  84. self.set_outputs(data)
  85. def print_inputs(self):
  86. # Состояние входов (вкл./выкл.)
  87. print('Inputs state [bit field] :', Fore.GREEN + self.get_inputs_mode())
  88. # Значения входов (битовое поле)
  89. print('Inputs values [bit field] :', Fore.GREEN + self.get_inputs_bit())
  90. # Значение счетчиков
  91. data = self.get_inputs_counters()
  92. print('Inputs counters :', Fore.GREEN + ' | '.join(str(el) for el in data))
  93. # Режим работы входов (битовое поле)
  94. print('Inputs mode [bit field] :', Fore.GREEN + self.get_inputs_mode())
  95. # Нормальное состояние входов (битовое поле)
  96. print('Inputs norm [bit field] :', Fore.GREEN + self.get_inputs_norm_state())
  97. # Период антидребезга (ms)
  98. print('Debounce input (ms) :', Fore.GREEN + ' | '.join(str(el) for el in self.get_debounce_channels()))
  99. def print_loads(self):
  100. # Значения датчиков нагрузки (битовое поле)
  101. print('Loads values [bit field] :', Fore.GREEN + self.get_loads_bit())
  102. # Слово достоверности датчиков нагрузки (битовое поле)
  103. print('Credibility loads [bit field] :', Fore.GREEN + self.get_cred_bit())
  104. class IO_DigitalTester:
  105. def __init__(self, dev_tester: IO_Digital, dev_dut: IO_Digital):
  106. self.tester = dev_tester
  107. self.dut = dev_dut
  108. '''Управление DUT'''
  109. # Подать/снять питание на DUT
  110. def dut_switch(self, state: bool):
  111. self.tester.set_output(1, state)
  112. # Установить 1 на нечетных входах DUT
  113. def dut_set_odd_inputs(self):
  114. self.tester.set_output(2, False)
  115. # Установить 1 на четный входах DUT
  116. def dut_set_even_imputs(self):
  117. self.tester.set_output(2, True)
  118. '''Тест входов'''
  119. def test_inputs(self):
  120. self.dut_set_even_imputs()
  121. time.sleep(0.1)
  122. self.dut.print_inputs()
  123. time.sleep(1)
  124. self.dut_set_odd_inputs()
  125. time.sleep(0.1)
  126. self.dut.print_inputs()
  127. time.sleep(1)
  128. '''Тест выходов и датчика обрыва нагрузки'''
  129. def test_load(self):
  130. # Все выходы DUT разомкнуты
  131. print("Все выходы DUT разомкнуты...")
  132. self.dut.set_outputs(0)
  133. self.dut.print_loads()
  134. time.sleep(0.1)
  135. # Замкнуть все выходы DUT (лампочка должна гореть)
  136. print("Все выходы DUT замкнуты...")
  137. self.dut.set_outputs(0b11111111)
  138. time.sleep(0.1)
  139. for i in range(1, 9):
  140. # Разомкнуть i-ый выход DUT
  141. print(f"Разомкнуть выход {i}")
  142. self.dut.set_output(i, 0)
  143. time.sleep(1)
  144. self.dut.print_loads()
  145. self.dut.set_output(i, 1)
  146. time.sleep(1)
  147. def get_load(self):
  148. self.dut.set_outputs(0b11110111)
  149. # self.dut.set_outputs(0)
  150. time.sleep(1)
  151. self.dut.print_loads()
  152. def main():
  153. colorama.init(autoreset=True)
  154. serial_port = Serial('COM22', 115200, timeout=0.05, parity='N', xonxoff=False)
  155. modbus_tester = Modbus(serial_port, 1)
  156. # dev_tester = IO_Digital(modbus_tester)
  157. dio = IO_Digital(modbus_tester)
  158. '''Тесты отдельного модуля DIO'''
  159. # dio.sys.get_system_vars()
  160. print(dio.sys.get_save_mode())
  161. print(dio.sys.get_save_delay())
  162. '''Сохранить информацию о модуле'''
  163. # dio.sys.set_info('This is super DIO!')
  164. # dio.print_inputs()
  165. # dio.get_inputs_counters()
  166. # modbus_dut = Modbus(serial_port, 2)
  167. # dev_dut = IO_Digital(modbus_dut)
  168. # tester = IO_DigitalTester(dev_tester, dev_dut)
  169. '''Включить DUT'''
  170. # tester.dut_switch(True)
  171. '''Запросить системные настройки DUT'''
  172. # dev_dut.sys.get_system_vars()
  173. '''Тест входов. Переключение значений на входах DUT'''
  174. # for i in range(10):
  175. # tester.test_inputs()
  176. '''Тестирование выходов и датчиков обрыва нагрузки'''
  177. # tester.test_load()
  178. # for i in range(100):
  179. # tester.get_load()
  180. '''Выходы'''
  181. # for i in range(100):
  182. # dio.set_outputs(0b00000001)
  183. # sleep(1)
  184. # dio.set_outputs(0b00000000)
  185. # sleep(1)
  186. '''Системные переменные и параметры'''
  187. # dev_tester.sys.get_system_vars()
  188. ''' Установить текущее время с учетом часового пояса'''
  189. # dev.sys.set_rtc()
  190. '''Лог и архив. Настройки лога.'''
  191. # dev.log.get_log_info()
  192. # dev.log.get_random_entries()
  193. # for i in range(8):
  194. # dev.log.set_archive_period(5 + i*2, i)
  195. # dev.log.get_log_info()
  196. # dev.log.get_all_archive()
  197. # dev.log.get_all_log()
  198. # dev.log.log_clear()
  199. # dev.log.archive_clear()
  200. '''Лог'''
  201. # dev.log.get_all_log()
  202. '''Архив'''
  203. # dev.log.get_all_archive()
  204. # print(dev.log.get_archive_entry(0, 1))
  205. '''Сохранение настроек'''
  206. # dev.sys.save_sattings()
  207. '''Настройки'''
  208. '''Регистры модуля'''
  209. # dev.print_inputs()
  210. '''Настройи модуля'''
  211. # for i in range(1, 9):
  212. # dev.set_input_mode(i, 1)
  213. # print('Inputs mode [bit field] :', Fore.GREEN + dev.get_inputs_mode())
  214. # dev.set_input_mode(4, 1)
  215. # dev.print_inputs()
  216. '''Обновление'''
  217. # dio.updater.update('fw.bin', 'MAI_12')
  218. if __name__ == '__main__':
  219. main()