#!/usr/bin/env python3 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								import  os 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . system . hardware  import  TICI 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								if  TICI : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  from  tinygrad . tensor  import  Tensor 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  from  tinygrad . dtype  import  dtypes 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  from  openpilot . selfdrive . modeld . runners . tinygrad_helpers  import  qcom_tensor_from_opencl_address 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  os . environ [ ' QCOM ' ]  =  ' 1 ' 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								else : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  from  openpilot . selfdrive . modeld . runners . ort_helpers  import  make_onnx_cpu_runner 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								import  time 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								import  pickle 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								import  numpy  as  np 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								import  cereal . messaging  as  messaging 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  cereal  import  car ,  log 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  pathlib  import  Path 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  setproctitle  import  setproctitle 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  cereal . messaging  import  PubMaster ,  SubMaster 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  msgq . visionipc  import  VisionIpcClient ,  VisionStreamType ,  VisionBuf 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  opendbc . car . car_helpers  import  get_demo_car_params 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . common . swaglog  import  cloudlog 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . common . params  import  Params 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . common . filter_simple  import  FirstOrderFilter 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . common . realtime  import  config_realtime_process 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . common . transformations . camera  import  DEVICE_CAMERAS 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . common . transformations . model  import  get_warp_matrix 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . system  import  sentry 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . selfdrive . controls . lib . desire_helper  import  DesireHelper 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . selfdrive . modeld . parse_model_outputs  import  Parser 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . selfdrive . modeld . fill_model_msg  import  fill_model_msg ,  fill_pose_msg ,  PublishState 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . selfdrive . modeld . constants  import  ModelConstants 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . selfdrive . modeld . models . commonmodel_pyx  import  DrivingModelFrame ,  CLContext 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								PROCESS_NAME  =  " selfdrive.modeld.modeld " 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								SEND_RAW_PRED  =  os . getenv ( ' SEND_RAW_PRED ' ) 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								MODEL_PATH  =  Path ( __file__ ) . parent  /  ' models/supercombo.onnx ' 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								MODEL_PKL_PATH  =  Path ( __file__ ) . parent  /  ' models/supercombo_tinygrad.pkl ' 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								METADATA_PATH  =  Path ( __file__ ) . parent  /  ' models/supercombo_metadata.pkl ' 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								class  FrameMeta : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  frame_id :  int  =  0 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  timestamp_sof :  int  =  0 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  timestamp_eof :  int  =  0 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  def  __init__ ( self ,  vipc = None ) : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  vipc  is  not  None : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      self . frame_id ,  self . timestamp_sof ,  self . timestamp_eof  =  vipc . frame_id ,  vipc . timestamp_sof ,  vipc . timestamp_eof 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								class  ModelState : 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  frames :  dict [ str ,  DrivingModelFrame ] 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  inputs :  dict [ str ,  np . ndarray ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  output :  np . ndarray 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  prev_desire :  np . ndarray   # for tracking the rising edge of the pulse 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  def  __init__ ( self ,  context :  CLContext ) : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    self . frames  =  { ' input_imgs ' :  DrivingModelFrame ( context ) ,  ' big_input_imgs ' :  DrivingModelFrame ( context ) } 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    self . prev_desire  =  np . zeros ( ModelConstants . DESIRE_LEN ,  dtype = np . float32 ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # img buffers are managed in openCL transform code 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    self . numpy_inputs  =  { 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      ' desire ' :  np . zeros ( ( 1 ,  ( ModelConstants . FULL_HISTORY_BUFFER_LEN + 1 ) ,  ModelConstants . DESIRE_LEN ) ,  dtype = np . float32 ) , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      ' traffic_convention ' :  np . zeros ( ( 1 ,  ModelConstants . TRAFFIC_CONVENTION_LEN ) ,  dtype = np . float32 ) , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      ' lateral_control_params ' :  np . zeros ( ( 1 ,  ModelConstants . LATERAL_CONTROL_PARAMS_LEN ) ,  dtype = np . float32 ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      ' prev_desired_curv ' :  np . zeros ( ( 1 ,  ( ModelConstants . FULL_HISTORY_BUFFER_LEN + 1 ) ,  ModelConstants . PREV_DESIRED_CURV_LEN ) ,  dtype = np . float32 ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      ' features_buffer ' :  np . zeros ( ( 1 ,  ModelConstants . FULL_HISTORY_BUFFER_LEN ,   ModelConstants . FEATURE_LEN ) ,  dtype = np . float32 ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    with  open ( METADATA_PATH ,  ' rb ' )  as  f : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      model_metadata  =  pickle . load ( f ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    self . input_shapes  =   model_metadata [ ' input_shapes ' ] 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    self . output_slices  =  model_metadata [ ' output_slices ' ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    net_output_size  =  model_metadata [ ' output_shapes ' ] [ ' outputs ' ] [ 1 ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    self . output  =  np . zeros ( net_output_size ,  dtype = np . float32 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    self . parser  =  Parser ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  TICI : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      self . tensor_inputs  =  { k :  Tensor ( v ,  device = ' NPY ' ) . realize ( )  for  k , v  in  self . numpy_inputs . items ( ) } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      with  open ( MODEL_PKL_PATH ,  " rb " )  as  f : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . model_run  =  pickle . load ( f ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      self . onnx_cpu_runner  =  make_onnx_cpu_runner ( MODEL_PATH ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  def  slice_outputs ( self ,  model_outputs :  np . ndarray )  - >  dict [ str ,  np . ndarray ] : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    parsed_model_outputs  =  { k :  model_outputs [ np . newaxis ,  v ]  for  k , v  in  self . output_slices . items ( ) } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  SEND_RAW_PRED : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      parsed_model_outputs [ ' raw_pred ' ]  =  model_outputs . copy ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  parsed_model_outputs 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  def  run ( self ,  buf :  VisionBuf ,  wbuf :  VisionBuf ,  transform :  np . ndarray ,  transform_wide :  np . ndarray , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								                inputs :  dict [ str ,  np . ndarray ] ,  prepare_only :  bool )  - >  dict [ str ,  np . ndarray ]  |  None : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # Model decides when action is completed, so desire input is just a pulse triggered on rising edge 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    inputs [ ' desire ' ] [ 0 ]  =  0 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    new_desire  =  np . where ( inputs [ ' desire ' ]  -  self . prev_desire  >  .99 ,  inputs [ ' desire ' ] ,  0 ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    self . prev_desire [ : ]  =  inputs [ ' desire ' ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    self . numpy_inputs [ ' desire ' ] [ 0 , : - 1 ]  =  self . numpy_inputs [ ' desire ' ] [ 0 , 1 : ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    self . numpy_inputs [ ' desire ' ] [ 0 , - 1 ]  =  new_desire 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    self . numpy_inputs [ ' traffic_convention ' ] [ : ]  =  inputs [ ' traffic_convention ' ] 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    self . numpy_inputs [ ' lateral_control_params ' ] [ : ]  =  inputs [ ' lateral_control_params ' ] 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    imgs_cl  =  { ' input_imgs ' :  self . frames [ ' input_imgs ' ] . prepare ( buf ,  transform . flatten ( ) ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								               ' big_input_imgs ' :  self . frames [ ' big_input_imgs ' ] . prepare ( wbuf ,  transform_wide . flatten ( ) ) } 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  TICI : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      # The imgs tensors are backed by opencl memory, only need init once 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      for  key  in  imgs_cl : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  key  not  in  self . tensor_inputs : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          self . tensor_inputs [ key ]  =  qcom_tensor_from_opencl_address ( imgs_cl [ key ] . mem_address ,  self . input_shapes [ key ] ,  dtype = dtypes . uint8 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      for  key  in  imgs_cl : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								        self . numpy_inputs [ key ]  =  self . frames [ key ] . buffer_from_cl ( imgs_cl [ key ] ) . reshape ( self . input_shapes [ key ] ) . astype ( dtype = np . float32 ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  prepare_only : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      return  None 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  TICI : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      self . output  =  self . model_run ( * * self . tensor_inputs ) . numpy ( ) . flatten ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      self . output  =  self . onnx_cpu_runner . run ( None ,  self . numpy_inputs ) [ 0 ] . flatten ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    outputs  =  self . parser . parse_outputs ( self . slice_outputs ( self . output ) ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    self . numpy_inputs [ ' features_buffer ' ] [ 0 , : - 1 ]  =  self . numpy_inputs [ ' features_buffer ' ] [ 0 , 1 : ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    self . numpy_inputs [ ' features_buffer ' ] [ 0 , - 1 ]  =  outputs [ ' hidden_state ' ] [ 0 ,  : ] 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # TODO model only uses last value now 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    self . numpy_inputs [ ' prev_desired_curv ' ] [ 0 , : - 1 ]  =  self . numpy_inputs [ ' prev_desired_curv ' ] [ 0 , 1 : ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    self . numpy_inputs [ ' prev_desired_curv ' ] [ 0 , - 1 , : ]  =  outputs [ ' desired_curvature ' ] [ 0 ,  : ] 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    return  outputs 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								def  main ( demo = False ) : 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  cloudlog . warning ( " modeld init " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  sentry . set_tag ( " daemon " ,  PROCESS_NAME ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  cloudlog . bind ( daemon = PROCESS_NAME ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  setproctitle ( PROCESS_NAME ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  config_realtime_process ( 7 ,  54 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  cloudlog . warning ( " setting up CL context " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  cl_context  =  CLContext ( ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  cloudlog . warning ( " CL context ready; loading model " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  model  =  ModelState ( cl_context ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  cloudlog . warning ( " models loaded, modeld starting " ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  # visionipc clients 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  while  True : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    available_streams  =  VisionIpcClient . available_streams ( " camerad " ,  block = False ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  available_streams : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      use_extra_client  =  VisionStreamType . VISION_STREAM_WIDE_ROAD  in  available_streams  and  VisionStreamType . VISION_STREAM_ROAD  in  available_streams 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      main_wide_camera  =  VisionStreamType . VISION_STREAM_ROAD  not  in  available_streams 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      break 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    time . sleep ( .1 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  vipc_client_main_stream  =  VisionStreamType . VISION_STREAM_WIDE_ROAD  if  main_wide_camera  else  VisionStreamType . VISION_STREAM_ROAD 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  vipc_client_main  =  VisionIpcClient ( " camerad " ,  vipc_client_main_stream ,  True ,  cl_context ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  vipc_client_extra  =  VisionIpcClient ( " camerad " ,  VisionStreamType . VISION_STREAM_WIDE_ROAD ,  False ,  cl_context ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  cloudlog . warning ( f " vision stream set up, main_wide_camera:  { main_wide_camera } , use_extra_client:  { use_extra_client } " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  while  not  vipc_client_main . connect ( False ) : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    time . sleep ( 0.1 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  while  use_extra_client  and  not  vipc_client_extra . connect ( False ) : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    time . sleep ( 0.1 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  cloudlog . warning ( f " connected main cam with buffer size:  { vipc_client_main . buffer_len }  ( { vipc_client_main . width }  x  { vipc_client_main . height } ) " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  use_extra_client : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    cloudlog . warning ( f " connected extra cam with buffer size:  { vipc_client_extra . buffer_len }  ( { vipc_client_extra . width }  x  { vipc_client_extra . height } ) " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  # messaging 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  pm  =  PubMaster ( [ " modelV2 " ,  " drivingModelData " ,  " cameraOdometry " ] ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  sm  =  SubMaster ( [ " deviceState " ,  " carState " ,  " roadCameraState " ,  " liveCalibration " ,  " driverMonitoringState " ,  " carControl " ] ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  publish_state  =  PublishState ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  params  =  Params ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  # setup filter to track dropped frames 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  frame_dropped_filter  =  FirstOrderFilter ( 0. ,  10. ,  1.  /  ModelConstants . MODEL_FREQ ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  frame_id  =  0 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  last_vipc_frame_id  =  0 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  run_count  =  0 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  model_transform_main  =  np . zeros ( ( 3 ,  3 ) ,  dtype = np . float32 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  model_transform_extra  =  np . zeros ( ( 3 ,  3 ) ,  dtype = np . float32 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  live_calib_seen  =  False 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  buf_main ,  buf_extra  =  None ,  None 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  meta_main  =  FrameMeta ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  meta_extra  =  FrameMeta ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  demo : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    CP  =  get_demo_car_params ( ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  else : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    CP  =  messaging . log_from_bytes ( params . get ( " CarParams " ,  block = True ) ,  car . CarParams ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  cloudlog . info ( " modeld got CarParams:  %s " ,  CP . brand ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  # TODO this needs more thought, use .2s extra for now to estimate other delays 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  steer_delay  =  CP . steerActuatorDelay  +  .2 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  DH  =  DesireHelper ( ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  while  True : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    # Keep receiving frames until we are at least 1 frame ahead of previous extra frame 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    while  meta_main . timestamp_sof  <  meta_extra . timestamp_sof  +  25000000 : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      buf_main  =  vipc_client_main . recv ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      meta_main  =  FrameMeta ( vipc_client_main ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      if  buf_main  is  None : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        break 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  buf_main  is  None : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      cloudlog . debug ( " vipc_client_main no frame " ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      continue 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  use_extra_client : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      # Keep receiving extra frames until frame id matches main camera 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      while  True : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        buf_extra  =  vipc_client_extra . recv ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        meta_extra  =  FrameMeta ( vipc_client_extra ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  buf_extra  is  None  or  meta_main . timestamp_sof  <  meta_extra . timestamp_sof  +  25000000 : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          break 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      if  buf_extra  is  None : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								        cloudlog . debug ( " vipc_client_extra no frame " ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								        continue 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      if  abs ( meta_main . timestamp_sof  -  meta_extra . timestamp_sof )  >  10000000 : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								        cloudlog . error ( f " frames out of sync! main:  { meta_main . frame_id }  ( { meta_main . timestamp_sof  /  1e9 : .5f } ), \
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                         extra :  { meta_extra . frame_id }  ( { meta_extra . timestamp_sof  /  1e9 : .5 f } ) " ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      # Use single camera 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      buf_extra  =  buf_main 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      meta_extra  =  meta_main 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    sm . update ( 0 ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    desire  =  DH . desire 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    is_rhd  =  sm [ " driverMonitoringState " ] . isRHD 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    frame_id  =  sm [ " roadCameraState " ] . frameId 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    v_ego  =  max ( sm [ " carState " ] . vEgo ,  0. ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    lateral_control_params  =  np . array ( [ v_ego ,  steer_delay ] ,  dtype = np . float32 ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  sm . updated [ " liveCalibration " ]  and  sm . seen [ ' roadCameraState ' ]  and  sm . seen [ ' deviceState ' ] : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      device_from_calib_euler  =  np . array ( sm [ " liveCalibration " ] . rpyCalib ,  dtype = np . float32 ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      dc  =  DEVICE_CAMERAS [ ( str ( sm [ ' deviceState ' ] . deviceType ) ,  str ( sm [ ' roadCameraState ' ] . sensor ) ) ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      model_transform_main  =  get_warp_matrix ( device_from_calib_euler ,  dc . ecam . intrinsics  if  main_wide_camera  else  dc . fcam . intrinsics ,  False ) . astype ( np . float32 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      model_transform_extra  =  get_warp_matrix ( device_from_calib_euler ,  dc . ecam . intrinsics ,  True ) . astype ( np . float32 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      live_calib_seen  =  True 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    traffic_convention  =  np . zeros ( 2 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    traffic_convention [ int ( is_rhd ) ]  =  1 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    vec_desire  =  np . zeros ( ModelConstants . DESIRE_LEN ,  dtype = np . float32 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  desire  > =  0  and  desire  <  ModelConstants . DESIRE_LEN : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      vec_desire [ desire ]  =  1 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # tracked dropped frames 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    vipc_dropped_frames  =  max ( 0 ,  meta_main . frame_id  -  last_vipc_frame_id  -  1 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    frames_dropped  =  frame_dropped_filter . update ( min ( vipc_dropped_frames ,  10 ) ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  run_count  <  10 :  # let frame drops warm up 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      frame_dropped_filter . x  =  0. 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      frames_dropped  =  0. 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    run_count  =  run_count  +  1 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    frame_drop_ratio  =  frames_dropped  /  ( 1  +  frames_dropped ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    prepare_only  =  vipc_dropped_frames  >  0 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  prepare_only : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      cloudlog . error ( f " skipping model eval. Dropped  { vipc_dropped_frames }  frames " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    inputs : dict [ str ,  np . ndarray ]  =  { 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      ' desire ' :  vec_desire , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      ' traffic_convention ' :  traffic_convention , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      ' lateral_control_params ' :  lateral_control_params , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    mt1  =  time . perf_counter ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    model_output  =  model . run ( buf_main ,  buf_extra ,  model_transform_main ,  model_transform_extra ,  inputs ,  prepare_only ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    mt2  =  time . perf_counter ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    model_execution_time  =  mt2  -  mt1 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  model_output  is  not  None : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      modelv2_send  =  messaging . new_message ( ' modelV2 ' ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      drivingdata_send  =  messaging . new_message ( ' drivingModelData ' ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      posenet_send  =  messaging . new_message ( ' cameraOdometry ' ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      fill_model_msg ( drivingdata_send ,  modelv2_send ,  model_output ,  v_ego ,  steer_delay , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                     publish_state ,  meta_main . frame_id ,  meta_extra . frame_id ,  frame_id , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								                     frame_drop_ratio ,  meta_main . timestamp_eof ,  model_execution_time ,  live_calib_seen ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      desire_state  =  modelv2_send . modelV2 . meta . desireState 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      l_lane_change_prob  =  desire_state [ log . Desire . laneChangeLeft ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      r_lane_change_prob  =  desire_state [ log . Desire . laneChangeRight ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      lane_change_prob  =  l_lane_change_prob  +  r_lane_change_prob 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      DH . update ( sm [ ' carState ' ] ,  sm [ ' carControl ' ] . latActive ,  lane_change_prob ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      modelv2_send . modelV2 . meta . laneChangeState  =  DH . lane_change_state 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      modelv2_send . modelV2 . meta . laneChangeDirection  =  DH . lane_change_direction 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      drivingdata_send . drivingModelData . meta . laneChangeState  =  DH . lane_change_state 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      drivingdata_send . drivingModelData . meta . laneChangeDirection  =  DH . lane_change_direction 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      fill_pose_msg ( posenet_send ,  model_output ,  meta_main . frame_id ,  vipc_dropped_frames ,  meta_main . timestamp_eof ,  live_calib_seen ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      pm . send ( ' modelV2 ' ,  modelv2_send ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      pm . send ( ' drivingModelData ' ,  drivingdata_send ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      pm . send ( ' cameraOdometry ' ,  posenet_send ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    last_vipc_frame_id  =  meta_main . frame_id 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								if  __name__  ==  " __main__ " : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  try : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    import  argparse 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    parser  =  argparse . ArgumentParser ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    parser . add_argument ( ' --demo ' ,  action = ' store_true ' ,  help = ' A boolean for demo mode. ' ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    args  =  parser . parse_args ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    main ( demo = args . demo ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  except  KeyboardInterrupt : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    cloudlog . warning ( f " child  { PROCESS_NAME }  got SIGINT " ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  except  Exception : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    sentry . capture_exception ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    raise