| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | # Serial port management for firmware communication | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2020-02-14 20:47:08 -05:00
										 |  |  | # Copyright (C) 2016-2020  Kevin O'Connor <kevin@koconnor.net> | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | # | 
					
						
							|  |  |  | # This file may be distributed under the terms of the GNU GPLv3 license. | 
					
						
							| 
									
										
										
										
											2017-03-31 20:48:29 -04:00
										 |  |  | import logging, threading | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | import serial | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-09 19:04:30 -05:00
										 |  |  | import msgproto, chelper, util | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-08 21:24:27 -05:00
										 |  |  | class error(Exception): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | class SerialReader: | 
					
						
							| 
									
										
										
										
											2016-07-11 11:41:49 -04:00
										 |  |  |     BITS_PER_BYTE = 10. | 
					
						
							| 
									
										
										
										
											2020-04-01 23:52:55 +02:00
										 |  |  |     def __init__(self, reactor, serialport, baud, rts=True): | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         self.reactor = reactor | 
					
						
							|  |  |  |         self.serialport = serialport | 
					
						
							|  |  |  |         self.baud = baud | 
					
						
							|  |  |  |         # Serial port | 
					
						
							|  |  |  |         self.ser = None | 
					
						
							| 
									
										
										
										
											2020-04-01 23:52:55 +02:00
										 |  |  |         self.rts = rts | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         self.msgparser = msgproto.MessageParser() | 
					
						
							|  |  |  |         # C interface | 
					
						
							|  |  |  |         self.ffi_main, self.ffi_lib = chelper.get_ffi() | 
					
						
							|  |  |  |         self.serialqueue = None | 
					
						
							|  |  |  |         self.default_cmd_queue = self.alloc_command_queue() | 
					
						
							|  |  |  |         self.stats_buf = self.ffi_main.new('char[4096]') | 
					
						
							|  |  |  |         # Threading | 
					
						
							|  |  |  |         self.lock = threading.Lock() | 
					
						
							|  |  |  |         self.background_thread = None | 
					
						
							|  |  |  |         # Message handlers | 
					
						
							| 
									
										
										
										
											2019-06-21 18:44:55 -04:00
										 |  |  |         self.handlers = {} | 
					
						
							|  |  |  |         self.register_response(self._handle_unknown_init, '#unknown') | 
					
						
							|  |  |  |         self.register_response(self.handle_output, '#output') | 
					
						
							| 
									
										
										
										
											2020-02-14 20:47:08 -05:00
										 |  |  |         # Sent message notification tracking | 
					
						
							|  |  |  |         self.last_notify_id = 0 | 
					
						
							|  |  |  |         self.pending_notifications = {} | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def _bg_thread(self): | 
					
						
							|  |  |  |         response = self.ffi_main.new('struct pull_queue_message *') | 
					
						
							|  |  |  |         while 1: | 
					
						
							|  |  |  |             self.ffi_lib.serialqueue_pull(self.serialqueue, response) | 
					
						
							|  |  |  |             count = response.len | 
					
						
							| 
									
										
										
										
											2020-02-14 20:47:08 -05:00
										 |  |  |             if count < 0: | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |                 break | 
					
						
							| 
									
										
										
										
											2020-02-14 20:47:08 -05:00
										 |  |  |             if response.notify_id: | 
					
						
							|  |  |  |                 params = {'#sent_time': response.sent_time, | 
					
						
							|  |  |  |                           '#receive_time': response.receive_time} | 
					
						
							|  |  |  |                 completion = self.pending_notifications.pop(response.notify_id) | 
					
						
							|  |  |  |                 self.reactor.async_complete(completion, params) | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |             params = self.msgparser.parse(response.msg[0:count]) | 
					
						
							|  |  |  |             params['#sent_time'] = response.sent_time | 
					
						
							|  |  |  |             params['#receive_time'] = response.receive_time | 
					
						
							| 
									
										
										
										
											2017-09-15 13:58:05 -04:00
										 |  |  |             hdl = (params['#name'], params.get('oid')) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2019-06-09 19:12:24 -04:00
										 |  |  |                 with self.lock: | 
					
						
							|  |  |  |                     hdl = self.handlers.get(hdl, self.handle_default) | 
					
						
							|  |  |  |                     hdl(params) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |             except: | 
					
						
							|  |  |  |                 logging.exception("Exception in serial callback") | 
					
						
							| 
									
										
										
										
											2020-02-16 15:46:04 -05:00
										 |  |  |     def _get_identify_data(self, eventtime): | 
					
						
							| 
									
										
										
										
											2019-06-21 18:44:55 -04:00
										 |  |  |         # Query the "data dictionary" from the micro-controller | 
					
						
							|  |  |  |         identify_data = "" | 
					
						
							|  |  |  |         while 1: | 
					
						
							| 
									
										
										
										
											2019-06-21 19:15:12 -04:00
										 |  |  |             msg = "identify offset=%d count=%d" % (len(identify_data), 40) | 
					
						
							| 
									
										
										
										
											2020-02-16 15:46:04 -05:00
										 |  |  |             try: | 
					
						
							|  |  |  |                 params = self.send_with_response(msg, 'identify_response') | 
					
						
							|  |  |  |             except error as e: | 
					
						
							|  |  |  |                 logging.exception("Wait for identify_response") | 
					
						
							|  |  |  |                 return None | 
					
						
							| 
									
										
										
										
											2019-06-21 18:44:55 -04:00
										 |  |  |             if params['offset'] == len(identify_data): | 
					
						
							|  |  |  |                 msgdata = params['data'] | 
					
						
							|  |  |  |                 if not msgdata: | 
					
						
							|  |  |  |                     # Done | 
					
						
							|  |  |  |                     return identify_data | 
					
						
							|  |  |  |                 identify_data += msgdata | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def connect(self): | 
					
						
							| 
									
										
										
										
											2016-11-27 17:45:58 -05:00
										 |  |  |         # Initial connection | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         logging.info("Starting serial connect") | 
					
						
							| 
									
										
										
										
											2019-06-21 18:54:56 -04:00
										 |  |  |         start_time = self.reactor.monotonic() | 
					
						
							| 
									
										
										
										
											2016-11-29 17:44:37 -05:00
										 |  |  |         while 1: | 
					
						
							| 
									
										
										
										
											2019-06-21 18:54:56 -04:00
										 |  |  |             connect_time = self.reactor.monotonic() | 
					
						
							| 
									
										
										
										
											2020-03-23 17:26:48 -04:00
										 |  |  |             if connect_time > start_time + 90.: | 
					
						
							| 
									
										
										
										
											2019-06-21 18:54:56 -04:00
										 |  |  |                 raise error("Unable to connect") | 
					
						
							| 
									
										
										
										
											2016-11-29 17:44:37 -05:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2017-05-08 10:29:59 -04:00
										 |  |  |                 if self.baud: | 
					
						
							|  |  |  |                     self.ser = serial.Serial( | 
					
						
							| 
									
										
										
										
											2020-04-01 23:52:55 +02:00
										 |  |  |                         baudrate=self.baud, timeout=0, exclusive=True) | 
					
						
							|  |  |  |                     self.ser.port = self.serialport | 
					
						
							|  |  |  |                     self.ser.rts = self.rts | 
					
						
							|  |  |  |                     self.ser.open() | 
					
						
							| 
									
										
										
										
											2017-05-08 10:29:59 -04:00
										 |  |  |                 else: | 
					
						
							|  |  |  |                     self.ser = open(self.serialport, 'rb+') | 
					
						
							| 
									
										
										
										
											2017-08-14 16:18:54 -04:00
										 |  |  |             except (OSError, IOError, serial.SerialException) as e: | 
					
						
							| 
									
										
										
										
											2017-09-27 11:43:14 -04:00
										 |  |  |                 logging.warn("Unable to open port: %s", e) | 
					
						
							| 
									
										
										
										
											2019-06-21 18:54:56 -04:00
										 |  |  |                 self.reactor.pause(connect_time + 5.) | 
					
						
							| 
									
										
										
										
											2016-11-29 17:44:37 -05:00
										 |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2017-05-08 10:29:59 -04:00
										 |  |  |             if self.baud: | 
					
						
							|  |  |  |                 stk500v2_leave(self.ser, self.reactor) | 
					
						
							| 
									
										
										
										
											2020-09-16 23:49:38 -04:00
										 |  |  |             self.serialqueue = self.ffi_main.gc( | 
					
						
							|  |  |  |                 self.ffi_lib.serialqueue_alloc(self.ser.fileno(), 0), | 
					
						
							|  |  |  |                 self.ffi_lib.serialqueue_free) | 
					
						
							| 
									
										
										
										
											2016-11-29 17:44:37 -05:00
										 |  |  |             self.background_thread = threading.Thread(target=self._bg_thread) | 
					
						
							|  |  |  |             self.background_thread.start() | 
					
						
							|  |  |  |             # Obtain and load the data dictionary from the firmware | 
					
						
							| 
									
										
										
										
											2020-02-16 15:46:04 -05:00
										 |  |  |             completion = self.reactor.register_callback(self._get_identify_data) | 
					
						
							|  |  |  |             identify_data = completion.wait(connect_time + 5.) | 
					
						
							|  |  |  |             if identify_data is not None: | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  |             logging.info("Timeout on serial connect") | 
					
						
							|  |  |  |             self.disconnect() | 
					
						
							| 
									
										
										
										
											2016-11-27 17:45:58 -05:00
										 |  |  |         msgparser = msgproto.MessageParser() | 
					
						
							|  |  |  |         msgparser.process_identify(identify_data) | 
					
						
							|  |  |  |         self.msgparser = msgparser | 
					
						
							| 
									
										
										
										
											2019-06-20 17:36:46 -04:00
										 |  |  |         self.register_response(self.handle_unknown, '#unknown') | 
					
						
							| 
									
										
										
										
											2017-03-03 22:02:27 -05:00
										 |  |  |         # Setup baud adjust | 
					
						
							| 
									
										
										
										
											2018-01-29 10:10:27 -05:00
										 |  |  |         mcu_baud = msgparser.get_constant_float('SERIAL_BAUD', None) | 
					
						
							|  |  |  |         if mcu_baud is not None: | 
					
						
							| 
									
										
										
										
											2016-11-27 17:45:58 -05:00
										 |  |  |             baud_adjust = self.BITS_PER_BYTE / mcu_baud | 
					
						
							|  |  |  |             self.ffi_lib.serialqueue_set_baud_adjust( | 
					
						
							|  |  |  |                 self.serialqueue, baud_adjust) | 
					
						
							| 
									
										
										
										
											2018-05-28 09:42:59 -04:00
										 |  |  |         receive_window = msgparser.get_constant_int('RECEIVE_WINDOW', None) | 
					
						
							|  |  |  |         if receive_window is not None: | 
					
						
							|  |  |  |             self.ffi_lib.serialqueue_set_receive_window( | 
					
						
							|  |  |  |                 self.serialqueue, receive_window) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def connect_file(self, debugoutput, dictionary, pace=False): | 
					
						
							|  |  |  |         self.ser = debugoutput | 
					
						
							|  |  |  |         self.msgparser.process_identify(dictionary, decompress=False) | 
					
						
							| 
									
										
										
										
											2020-09-16 23:49:38 -04:00
										 |  |  |         self.serialqueue = self.ffi_main.gc( | 
					
						
							|  |  |  |             self.ffi_lib.serialqueue_alloc(self.ser.fileno(), 1), | 
					
						
							|  |  |  |             self.ffi_lib.serialqueue_free) | 
					
						
							| 
									
										
										
										
											2017-09-18 21:41:00 -04:00
										 |  |  |     def set_clock_est(self, freq, last_time, last_clock): | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         self.ffi_lib.serialqueue_set_clock_est( | 
					
						
							| 
									
										
										
										
											2017-09-18 21:41:00 -04:00
										 |  |  |             self.serialqueue, freq, last_time, last_clock) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def disconnect(self): | 
					
						
							| 
									
										
										
										
											2017-03-08 22:01:52 -05:00
										 |  |  |         if self.serialqueue is not None: | 
					
						
							|  |  |  |             self.ffi_lib.serialqueue_exit(self.serialqueue) | 
					
						
							|  |  |  |             if self.background_thread is not None: | 
					
						
							|  |  |  |                 self.background_thread.join() | 
					
						
							|  |  |  |             self.background_thread = self.serialqueue = None | 
					
						
							|  |  |  |         if self.ser is not None: | 
					
						
							|  |  |  |             self.ser.close() | 
					
						
							|  |  |  |             self.ser = None | 
					
						
							| 
									
										
										
										
											2020-02-14 20:47:08 -05:00
										 |  |  |         for pn in self.pending_notifications.values(): | 
					
						
							|  |  |  |             pn.complete(None) | 
					
						
							|  |  |  |         self.pending_notifications.clear() | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def stats(self, eventtime): | 
					
						
							|  |  |  |         if self.serialqueue is None: | 
					
						
							|  |  |  |             return "" | 
					
						
							| 
									
										
										
										
											2017-09-18 21:41:00 -04:00
										 |  |  |         self.ffi_lib.serialqueue_get_stats( | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |             self.serialqueue, self.stats_buf, len(self.stats_buf)) | 
					
						
							| 
									
										
										
										
											2017-09-18 21:41:00 -04:00
										 |  |  |         return self.ffi_main.string(self.stats_buf) | 
					
						
							| 
									
										
										
										
											2020-02-19 16:46:06 -05:00
										 |  |  |     def get_reactor(self): | 
					
						
							|  |  |  |         return self.reactor | 
					
						
							| 
									
										
										
										
											2019-06-20 22:08:38 -04:00
										 |  |  |     def get_msgparser(self): | 
					
						
							|  |  |  |         return self.msgparser | 
					
						
							| 
									
										
										
										
											2019-06-21 19:35:34 -04:00
										 |  |  |     def get_default_command_queue(self): | 
					
						
							|  |  |  |         return self.default_cmd_queue | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     # Serial response callbacks | 
					
						
							| 
									
										
										
										
											2019-06-20 17:36:46 -04:00
										 |  |  |     def register_response(self, callback, name, oid=None): | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         with self.lock: | 
					
						
							| 
									
										
										
										
											2019-06-26 10:31:15 -04:00
										 |  |  |             if callback is None: | 
					
						
							|  |  |  |                 del self.handlers[name, oid] | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.handlers[name, oid] = callback | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     # Command sending | 
					
						
							| 
									
										
										
										
											2018-02-27 14:16:16 -05:00
										 |  |  |     def raw_send(self, cmd, minclock, reqclock, cmd_queue): | 
					
						
							| 
									
										
										
										
											2020-02-14 20:47:08 -05:00
										 |  |  |         self.ffi_lib.serialqueue_send(self.serialqueue, cmd_queue, | 
					
						
							|  |  |  |                                       cmd, len(cmd), minclock, reqclock, 0) | 
					
						
							|  |  |  |     def raw_send_wait_ack(self, cmd, minclock, reqclock, cmd_queue): | 
					
						
							|  |  |  |         self.last_notify_id += 1 | 
					
						
							|  |  |  |         nid = self.last_notify_id | 
					
						
							|  |  |  |         completion = self.reactor.completion() | 
					
						
							|  |  |  |         self.pending_notifications[nid] = completion | 
					
						
							|  |  |  |         self.ffi_lib.serialqueue_send(self.serialqueue, cmd_queue, | 
					
						
							|  |  |  |                                       cmd, len(cmd), minclock, reqclock, nid) | 
					
						
							|  |  |  |         params = completion.wait() | 
					
						
							|  |  |  |         if params is None: | 
					
						
							|  |  |  |             raise error("Serial connection closed") | 
					
						
							|  |  |  |         return params | 
					
						
							| 
									
										
										
										
											2018-02-27 14:16:16 -05:00
										 |  |  |     def send(self, msg, minclock=0, reqclock=0): | 
					
						
							|  |  |  |         cmd = self.msgparser.create_command(msg) | 
					
						
							|  |  |  |         self.raw_send(cmd, minclock, reqclock, self.default_cmd_queue) | 
					
						
							| 
									
										
										
										
											2019-06-21 19:15:12 -04:00
										 |  |  |     def send_with_response(self, msg, response): | 
					
						
							|  |  |  |         cmd = self.msgparser.create_command(msg) | 
					
						
							| 
									
										
										
										
											2019-06-26 14:30:02 -04:00
										 |  |  |         src = SerialRetryCommand(self, response) | 
					
						
							| 
									
										
										
										
											2020-02-16 15:46:04 -05:00
										 |  |  |         return src.get_response(cmd, self.default_cmd_queue) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def alloc_command_queue(self): | 
					
						
							| 
									
										
										
										
											2016-11-30 01:58:45 -05:00
										 |  |  |         return self.ffi_main.gc(self.ffi_lib.serialqueue_alloc_commandqueue(), | 
					
						
							|  |  |  |                                 self.ffi_lib.serialqueue_free_commandqueue) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     # Dumping debug lists | 
					
						
							|  |  |  |     def dump_debug(self): | 
					
						
							| 
									
										
										
										
											2017-09-27 11:54:10 -04:00
										 |  |  |         out = [] | 
					
						
							|  |  |  |         out.append("Dumping serial stats: %s" % ( | 
					
						
							|  |  |  |             self.stats(self.reactor.monotonic()),)) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         sdata = self.ffi_main.new('struct pull_queue_message[1024]') | 
					
						
							|  |  |  |         rdata = self.ffi_main.new('struct pull_queue_message[1024]') | 
					
						
							| 
									
										
										
										
											2020-09-16 23:49:38 -04:00
										 |  |  |         scount = self.ffi_lib.serialqueue_extract_old(self.serialqueue, 1, | 
					
						
							|  |  |  |                                                       sdata, len(sdata)) | 
					
						
							|  |  |  |         rcount = self.ffi_lib.serialqueue_extract_old(self.serialqueue, 0, | 
					
						
							|  |  |  |                                                       rdata, len(rdata)) | 
					
						
							| 
									
										
										
										
											2017-09-27 11:54:10 -04:00
										 |  |  |         out.append("Dumping send queue %d messages" % (scount,)) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         for i in range(scount): | 
					
						
							|  |  |  |             msg = sdata[i] | 
					
						
							|  |  |  |             cmds = self.msgparser.dump(msg.msg[0:msg.len]) | 
					
						
							| 
									
										
										
										
											2017-09-27 11:54:10 -04:00
										 |  |  |             out.append("Sent %d %f %f %d: %s" % ( | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |                 i, msg.receive_time, msg.sent_time, msg.len, ', '.join(cmds))) | 
					
						
							| 
									
										
										
										
											2017-09-27 11:54:10 -04:00
										 |  |  |         out.append("Dumping receive queue %d messages" % (rcount,)) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         for i in range(rcount): | 
					
						
							|  |  |  |             msg = rdata[i] | 
					
						
							|  |  |  |             cmds = self.msgparser.dump(msg.msg[0:msg.len]) | 
					
						
							| 
									
										
										
										
											2017-09-27 11:54:10 -04:00
										 |  |  |             out.append("Receive: %d %f %f %d: %s" % ( | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |                 i, msg.receive_time, msg.sent_time, msg.len, ', '.join(cmds))) | 
					
						
							| 
									
										
										
										
											2017-09-27 11:54:10 -04:00
										 |  |  |         return '\n'.join(out) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     # Default message handlers | 
					
						
							| 
									
										
										
										
											2019-06-21 18:44:55 -04:00
										 |  |  |     def _handle_unknown_init(self, params): | 
					
						
							|  |  |  |         logging.debug("Unknown message %d (len %d) while identifying", | 
					
						
							|  |  |  |                       params['#msgid'], len(params['#msg'])) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def handle_unknown(self, params): | 
					
						
							| 
									
										
										
										
											2017-09-27 11:43:14 -04:00
										 |  |  |         logging.warn("Unknown message type %d: %s", | 
					
						
							|  |  |  |                      params['#msgid'], repr(params['#msg'])) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def handle_output(self, params): | 
					
						
							| 
									
										
										
										
											2017-09-27 11:43:14 -04:00
										 |  |  |         logging.info("%s: %s", params['#name'], params['#msg']) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def handle_default(self, params): | 
					
						
							| 
									
										
										
										
											2017-09-27 11:43:14 -04:00
										 |  |  |         logging.warn("got %s", params) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:46:04 -05:00
										 |  |  | # Class to send a query command and return the received response | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | class SerialRetryCommand: | 
					
						
							| 
									
										
										
										
											2019-06-26 14:30:02 -04:00
										 |  |  |     def __init__(self, serial, name, oid=None): | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         self.serial = serial | 
					
						
							|  |  |  |         self.name = name | 
					
						
							| 
									
										
										
										
											2017-03-05 15:00:15 -05:00
										 |  |  |         self.oid = oid | 
					
						
							| 
									
										
										
										
											2020-02-16 15:46:04 -05:00
										 |  |  |         self.last_params = None | 
					
						
							| 
									
										
										
										
											2019-06-26 14:30:02 -04:00
										 |  |  |         self.serial.register_response(self.handle_callback, name, oid) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def handle_callback(self, params): | 
					
						
							| 
									
										
										
										
											2020-02-16 15:46:04 -05:00
										 |  |  |         self.last_params = params | 
					
						
							|  |  |  |     def get_response(self, cmd, cmd_queue, minclock=0): | 
					
						
							|  |  |  |         retries = 5 | 
					
						
							|  |  |  |         retry_delay = .010 | 
					
						
							| 
									
										
										
										
											2019-06-26 14:30:02 -04:00
										 |  |  |         while 1: | 
					
						
							| 
									
										
										
										
											2020-02-16 15:46:04 -05:00
										 |  |  |             self.serial.raw_send_wait_ack(cmd, minclock, minclock, cmd_queue) | 
					
						
							|  |  |  |             params = self.last_params | 
					
						
							| 
									
										
										
										
											2019-06-26 14:30:02 -04:00
										 |  |  |             if params is not None: | 
					
						
							|  |  |  |                 self.serial.register_response(None, self.name, self.oid) | 
					
						
							|  |  |  |                 return params | 
					
						
							| 
									
										
										
										
											2020-02-16 15:46:04 -05:00
										 |  |  |             if retries <= 0: | 
					
						
							| 
									
										
										
										
											2019-06-26 14:30:02 -04:00
										 |  |  |                 self.serial.register_response(None, self.name, self.oid) | 
					
						
							| 
									
										
										
										
											2020-02-16 15:46:04 -05:00
										 |  |  |                 raise error("Unable to obtain '%s' response" % (self.name,)) | 
					
						
							|  |  |  |             reactor = self.serial.reactor | 
					
						
							|  |  |  |             reactor.pause(reactor.monotonic() + retry_delay) | 
					
						
							|  |  |  |             retries -= 1 | 
					
						
							|  |  |  |             retry_delay *= 2. | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | # Attempt to place an AVR stk500v2 style programmer into normal mode | 
					
						
							| 
									
										
										
										
											2016-11-28 13:14:56 -05:00
										 |  |  | def stk500v2_leave(ser, reactor): | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     logging.debug("Starting stk500v2 leave programmer sequence") | 
					
						
							| 
									
										
										
										
											2016-12-09 19:04:30 -05:00
										 |  |  |     util.clear_hupcl(ser.fileno()) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     origbaud = ser.baudrate | 
					
						
							|  |  |  |     # Request a dummy speed first as this seems to help reset the port | 
					
						
							| 
									
										
										
										
											2016-06-11 16:35:48 -04:00
										 |  |  |     ser.baudrate = 2400 | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     ser.read(1) | 
					
						
							|  |  |  |     # Send stk500v2 leave programmer sequence | 
					
						
							|  |  |  |     ser.baudrate = 115200 | 
					
						
							| 
									
										
										
										
											2017-02-06 13:31:34 -05:00
										 |  |  |     reactor.pause(reactor.monotonic() + 0.100) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     ser.read(4096) | 
					
						
							|  |  |  |     ser.write('\x1b\x01\x00\x01\x0e\x11\x04') | 
					
						
							| 
									
										
										
										
											2017-02-06 13:31:34 -05:00
										 |  |  |     reactor.pause(reactor.monotonic() + 0.050) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     res = ser.read(4096) | 
					
						
							| 
									
										
										
										
											2017-09-27 11:43:14 -04:00
										 |  |  |     logging.debug("Got %s from stk500v2", repr(res)) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     ser.baudrate = origbaud | 
					
						
							| 
									
										
										
										
											2017-03-08 22:26:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 23:52:55 +02:00
										 |  |  | def cheetah_reset(serialport, reactor): | 
					
						
							|  |  |  |     # Fysetc Cheetah v1.2 boards have a weird stateful circuitry for | 
					
						
							|  |  |  |     # configuring the bootloader. This sequence takes care of disabling it for | 
					
						
							|  |  |  |     # sure. | 
					
						
							|  |  |  |     # Open the serial port with RTS asserted | 
					
						
							|  |  |  |     ser = serial.Serial(baudrate=2400, timeout=0, exclusive=True) | 
					
						
							|  |  |  |     ser.port = serialport | 
					
						
							|  |  |  |     ser.rts = True | 
					
						
							|  |  |  |     ser.open() | 
					
						
							|  |  |  |     ser.read(1) | 
					
						
							|  |  |  |     reactor.pause(reactor.monotonic() + 0.100) | 
					
						
							|  |  |  |     # Toggle DTR | 
					
						
							|  |  |  |     ser.dtr = True | 
					
						
							|  |  |  |     reactor.pause(reactor.monotonic() + 0.100) | 
					
						
							|  |  |  |     ser.dtr = False | 
					
						
							|  |  |  |     # Deassert RTS | 
					
						
							|  |  |  |     reactor.pause(reactor.monotonic() + 0.100) | 
					
						
							|  |  |  |     ser.rts = False | 
					
						
							|  |  |  |     reactor.pause(reactor.monotonic() + 0.100) | 
					
						
							|  |  |  |     # Toggle DTR again | 
					
						
							|  |  |  |     ser.dtr = True | 
					
						
							|  |  |  |     reactor.pause(reactor.monotonic() + 0.100) | 
					
						
							|  |  |  |     ser.dtr = False | 
					
						
							|  |  |  |     reactor.pause(reactor.monotonic() + 0.100) | 
					
						
							|  |  |  |     ser.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-08 22:26:10 -05:00
										 |  |  | # Attempt an arduino style reset on a serial port | 
					
						
							|  |  |  | def arduino_reset(serialport, reactor): | 
					
						
							| 
									
										
										
										
											2017-10-12 21:21:49 -04:00
										 |  |  |     # First try opening the port at a different baud | 
					
						
							| 
									
										
										
										
											2019-02-10 08:51:29 -08:00
										 |  |  |     ser = serial.Serial(serialport, 2400, timeout=0, exclusive=True) | 
					
						
							| 
									
										
										
										
											2017-03-08 22:26:10 -05:00
										 |  |  |     ser.read(1) | 
					
						
							| 
									
										
										
										
											2017-03-31 20:48:29 -04:00
										 |  |  |     reactor.pause(reactor.monotonic() + 0.100) | 
					
						
							| 
									
										
										
										
											2017-10-12 21:21:49 -04:00
										 |  |  |     # Then toggle DTR | 
					
						
							| 
									
										
										
										
											2017-03-08 22:26:10 -05:00
										 |  |  |     ser.dtr = True | 
					
						
							| 
									
										
										
										
											2017-03-31 20:48:29 -04:00
										 |  |  |     reactor.pause(reactor.monotonic() + 0.100) | 
					
						
							| 
									
										
										
										
											2017-03-08 22:26:10 -05:00
										 |  |  |     ser.dtr = False | 
					
						
							| 
									
										
										
										
											2017-03-31 20:48:29 -04:00
										 |  |  |     reactor.pause(reactor.monotonic() + 0.100) | 
					
						
							| 
									
										
										
										
											2017-03-08 22:26:10 -05:00
										 |  |  |     ser.close() |