|  |  |  | #!/usr/bin/env python3
 | 
					
						
							|  |  |  | import re
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import cereal.messaging as messaging
 | 
					
						
							|  |  |  | from selfdrive.car.isotp_parallel_query import IsoTpParallelQuery
 | 
					
						
							|  |  |  | from selfdrive.car.fw_query_definitions import StdQueries
 | 
					
						
							|  |  |  | from system.swaglog import cloudlog
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VIN_UNKNOWN = "0" * 17
 | 
					
						
							|  |  |  | VIN_RE = "[A-HJ-NPR-Z0-9]{17}"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def is_valid_vin(vin: str):
 | 
					
						
							|  |  |  |   return re.fullmatch(VIN_RE, vin) is not None
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_vin(logcan, sendcan, bus, timeout=0.1, retry=5, debug=False):
 | 
					
						
							|  |  |  |   addrs = [0x7e0, 0x7e2, 0x18da10f1, 0x18da0ef1]  # engine, VMCU, 29-bit engine, PGM-FI
 | 
					
						
							|  |  |  |   for i in range(retry):
 | 
					
						
							|  |  |  |     for request, response in ((StdQueries.UDS_VIN_REQUEST, StdQueries.UDS_VIN_RESPONSE), (StdQueries.OBD_VIN_REQUEST, StdQueries.OBD_VIN_RESPONSE)):
 | 
					
						
							|  |  |  |       try:
 | 
					
						
							|  |  |  |         query = IsoTpParallelQuery(sendcan, logcan, bus, addrs, [request, ], [response, ], debug=debug)
 | 
					
						
							|  |  |  |         for (addr, rx_addr), vin in query.get_data(timeout).items():
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           # Honda Bosch response starts with a length, trim to correct length
 | 
					
						
							|  |  |  |           if vin.startswith(b'\x11'):
 | 
					
						
							|  |  |  |             vin = vin[1:18]
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           return addr[0], rx_addr, vin.decode()
 | 
					
						
							|  |  |  |         cloudlog.error(f"vin query retry ({i+1}) ...")
 | 
					
						
							|  |  |  |       except Exception:
 | 
					
						
							|  |  |  |         cloudlog.exception("VIN query exception")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return 0, 0, VIN_UNKNOWN
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__":
 | 
					
						
							|  |  |  |   import argparse
 | 
					
						
							|  |  |  |   import time
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   parser = argparse.ArgumentParser(description='Get VIN of the car')
 | 
					
						
							|  |  |  |   parser.add_argument('--debug', action='store_true')
 | 
					
						
							|  |  |  |   parser.add_argument('--bus', type=int, default=1)
 | 
					
						
							|  |  |  |   parser.add_argument('--timeout', type=float, default=0.1)
 | 
					
						
							|  |  |  |   parser.add_argument('--retry', type=int, default=5)
 | 
					
						
							|  |  |  |   args = parser.parse_args()
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sendcan = messaging.pub_sock('sendcan')
 | 
					
						
							|  |  |  |   logcan = messaging.sub_sock('can')
 | 
					
						
							|  |  |  |   time.sleep(1)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   addr, vin_rx_addr, vin = get_vin(logcan, sendcan, args.bus, args.timeout, args.retry, debug=args.debug)
 | 
					
						
							|  |  |  |   print(f'TX: {hex(addr)}, RX: {hex(vin_rx_addr)}, VIN: {vin}')
 |