#!/usr/bin/env python3 import os import sys from typing import Dict, List from common.basedir import BASEDIR # messages reserved for CAN based ignition (see can_ignition_hook function in panda/board/drivers/can) # (addr, len) CAN_IGNITION_MSGS = { 'gm': [(0x1F1, 8), (0x160, 5)], #'tesla' : [(0x348, 8)], } def _get_fingerprints(): # read all the folders in selfdrive/car and return a dict where: # - keys are all the car names that which we have a fingerprint dict for # - values are dicts of fingeprints for each trim fingerprints = {} for car_folder in [x[0] for x in os.walk(BASEDIR + '/selfdrive/car')]: car_name = car_folder.split('/')[-1] try: fingerprints[car_name] = __import__(f'selfdrive.car.{car_name}.values', fromlist=['FINGERPRINTS']).FINGERPRINTS except (ImportError, OSError, AttributeError): pass return fingerprints def check_fingerprint_consistency(f1, f2): # return false if it finds a fingerprint fully included in another # max message worth checking is 1800, as above that they usually come too infrequently and not # usable for fingerprinting max_msg = 1800 is_f1_in_f2 = True for k in f1: if (k not in f2 or f1[k] != f2[k]) and k < max_msg: is_f1_in_f2 = False is_f2_in_f1 = True for k in f2: if (k not in f1 or f2[k] != f1[k]) and k < max_msg: is_f2_in_f1 = False return not is_f1_in_f2 and not is_f2_in_f1 def check_can_ignition_conflicts(fingerprints, brands): # loops through all the fingerprints and exits if CAN ignition dedicated messages # are found in unexpected fingerprints for brand_can, msgs_can in CAN_IGNITION_MSGS.items(): for i, f in enumerate(fingerprints): for msg_can in msgs_can: if brand_can != brands[i] and msg_can[0] in f and msg_can[1] == f[msg_can[0]]: print("CAN ignition dedicated msg %d with len %d found in %s fingerprints!" % (msg_can[0], msg_can[1], brands[i])) print("TEST FAILED") sys.exit(1) if __name__ == "__main__": fingerprints = _get_fingerprints() fingerprints_flat: List[Dict] = [] car_names = [] brand_names = [] for brand in fingerprints: for car in fingerprints[brand]: fingerprints_flat += fingerprints[brand][car] for i in range(len(fingerprints[brand][car])): car_names.append(car) brand_names.append(brand) # first check if CAN ignition specific messages are unexpectedly included in other fingerprints check_can_ignition_conflicts(fingerprints_flat, brand_names) valid = True for idx1, f1 in enumerate(fingerprints_flat): for idx2, f2 in enumerate(fingerprints_flat): if idx1 < idx2 and not check_fingerprint_consistency(f1, f2): valid = False print(f"Those two fingerprints are inconsistent {car_names[idx1]} {car_names[idx2]}") print("") print(', '.join("%d: %d" % v for v in sorted(f1.items()))) print("") print(', '.join("%d: %d" % v for v in sorted(f2.items()))) print("") print(f"Found {len(fingerprints_flat)} individual fingerprints") if not valid or len(fingerprints_flat) == 0: print("TEST FAILED") sys.exit(1) else: print("TEST SUCESSFUL")