#!/usr/bin/env python3 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								import  argparse 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								import  binascii 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								import  time 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  collections  import  defaultdict 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  typing  import  Optional 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								import  cereal . messaging  as  messaging 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . selfdrive . debug . can_table  import  can_table 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . tools . lib . logreader  import  LogIterable 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . tools . lib . srreader  import  SegmentRangeReader 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								RED  =  ' \033 [91m ' 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								CLEAR  =  ' \033 [0m ' 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								def  update ( msgs ,  bus ,  dat ,  low_to_high ,  high_to_low ,  quiet = False ) : 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  for  x  in  msgs : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  x . which ( )  !=  ' can ' : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      continue 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  y  in  x . can : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      if  y . src  ==  bus : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								        dat [ y . address ]  =  y . dat 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								        i  =  int . from_bytes ( y . dat ,  byteorder = ' big ' ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								        l_h  =  low_to_high [ y . address ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        h_l  =  high_to_low [ y . address ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        change  =  None 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( i  |  l_h )  !=  l_h : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          low_to_high [ y . address ]  =  i  |  l_h 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          change  =  " + " 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( ~ i  |  h_l )  !=  h_l : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          high_to_low [ y . address ]  =  ~ i  |  h_l 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          change  =  " - " 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  change  and  not  quiet : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          print ( f " { time . monotonic ( ) : .2f } \t { hex ( y . address ) }  ( { y . address } ) \t { change } { binascii . hexlify ( y . dat ) } " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								def  can_printer ( bus = 0 ,  init_msgs = None ,  new_msgs = None ,  table = False ) : 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  logcan  =  messaging . sub_sock ( ' can ' ,  timeout = 10 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  dat  =  defaultdict ( int ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  low_to_high  =  defaultdict ( int ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  high_to_low  =  defaultdict ( int ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  if  init_msgs  is  not  None : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    update ( init_msgs ,  bus ,  dat ,  low_to_high ,  high_to_low ,  quiet = True ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  low_to_high_init  =  low_to_high . copy ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  high_to_low_init  =  high_to_low . copy ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  if  new_msgs  is  not  None : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    update ( new_msgs ,  bus ,  dat ,  low_to_high ,  high_to_low ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  else : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # Live mode 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    print ( f " Waiting for messages on bus  { bus } " ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    try : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      while  1 : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        can_recv  =  messaging . drain_sock ( logcan ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        update ( can_recv ,  bus ,  dat ,  low_to_high ,  high_to_low ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        time . sleep ( 0.02 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    except  KeyboardInterrupt : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      pass 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  print ( " \n \n " ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  tables  =  " " 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  for  addr  in  sorted ( dat . keys ( ) ) : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    init  =  low_to_high_init [ addr ]  &  high_to_low_init [ addr ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    now  =  low_to_high [ addr ]  &  high_to_low [ addr ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    d  =  now  &  ~ init 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  d  ==  0 : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      continue 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    b  =  d . to_bytes ( len ( dat [ addr ] ) ,  byteorder = ' big ' ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    byts  =  ' ' . join ( [ ( c  if  c  ==  ' 0 '  else  f ' { RED } { c } { CLEAR } ' )  for  c  in  str ( binascii . hexlify ( b ) ) [ 2 : - 1 ] ] ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    header  =  f " { hex ( addr ) . ljust ( 6 ) } ( { str ( addr ) . ljust ( 4 ) } ) " 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    print ( header ,  byts ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    tables  + =  f " { header } \n " 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    tables  + =  can_table ( b )  +  " \n \n " 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  if  table : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    print ( tables ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								if  __name__  ==  " __main__ " : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  desc  =  """ Collects messages and prints when a new bit transition is observed. 
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  This  is  very  useful  to  find  signals  based  on  user  triggered  actions ,  such  as  blinkers  and  seatbelt . 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  Leave  the  script  running  until  no  new  transitions  are  seen ,  then  perform  the  action . """ 
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser  =  argparse . ArgumentParser ( description = desc , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                                   formatter_class = argparse . ArgumentDefaultsHelpFormatter ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser . add_argument ( " --bus " ,  type = int ,  help = " CAN bus to print out " ,  default = 0 ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  parser . add_argument ( " --table " ,  action = " store_true " ,  help = " Print a cabana-like table " ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  parser . add_argument ( " init " ,  type = str ,  nargs = ' ? ' ,  help = " Route or segment to initialize with. Use empty quotes to compare against all zeros. " ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  parser . add_argument ( " comp " ,  type = str ,  nargs = ' ? ' ,  help = " Route or segment to compare against init " ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  args  =  parser . parse_args ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  init_lr :  Optional [ LogIterable ]  =  None 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  new_lr :  Optional [ LogIterable ]  =  None 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  if  args . init : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  args . init  ==  ' ' : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      init_lr  =  [ ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      init_lr  =  SegmentRangeReader ( args . init ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  if  args . comp : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    new_lr  =  SegmentRangeReader ( args . comp ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  can_printer ( args . bus ,  init_msgs = init_lr ,  new_msgs = new_lr ,  table = args . table )