# include  "common/params.h" 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# include  <dirent.h> 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# include  <sys/file.h> 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <algorithm> 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <cassert> 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# include  <csignal> 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# include  <unordered_map> 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  "common/queue.h" 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# include  "common/swaglog.h" 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# include  "common/util.h" 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  "system/hardware/hw.h" 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								namespace  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								volatile  sig_atomic_t  params_do_exit  =  0 ; 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								void  params_sig_handler ( int  signal )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  params_do_exit  =  1 ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								int  fsync_dir ( const  std : : string  & path )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  int  result  =  - 1 ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  int  fd  =  HANDLE_EINTR ( open ( path . c_str ( ) ,  O_RDONLY ,  0755 ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ( fd  > =  0 )  { 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    result  =  HANDLE_EINTR ( fsync ( fd ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    HANDLE_EINTR ( close ( fd ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  return  result ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								bool  create_params_path ( const  std : : string  & param_path ,  const  std : : string  & key_path )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // Make sure params path exists
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ( ! util : : file_exists ( param_path )  & &  ! util : : create_directories ( param_path ,  0775 ) )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  false ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // See if the symlink exists, otherwise create it
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ( ! util : : file_exists ( key_path ) )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 1) Create temp folder
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 2) Symlink it to temp link
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 3) Move symlink to <params>/d
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    std : : string  tmp_path  =  param_path  +  " /.tmp_XXXXXX " ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // this should be OK since mkdtemp just replaces characters in place
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    char  * tmp_dir  =  mkdtemp ( ( char  * ) tmp_path . c_str ( ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( tmp_dir  = =  NULL )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      return  false ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    std : : string  link_path  =  std : : string ( tmp_dir )  +  " .link " ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( symlink ( tmp_dir ,  link_path . c_str ( ) )  ! =  0 )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      return  false ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // don't return false if it has been created by other
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( rename ( link_path . c_str ( ) ,  key_path . c_str ( ) )  ! =  0  & &  errno  ! =  EEXIST )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      return  false ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  return  true ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								std : : string  ensure_params_path ( const  std : : string  & prefix ,  const  std : : string  & path  =  { } )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  std : : string  params_path  =  path . empty ( )  ?  Path : : params ( )  :  path ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ( ! create_params_path ( params_path ,  params_path  +  prefix ) )  { 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    throw  std : : runtime_error ( util : : string_format ( 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " Failed to ensure params path, errno=%d, path=%s, param_prefix=%s " , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        errno ,  params_path . c_str ( ) ,  prefix . c_str ( ) ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  return  params_path ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								class  FileLock  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  FileLock ( const  std : : string  & fn )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    fd_  =  HANDLE_EINTR ( open ( fn . c_str ( ) ,  O_CREAT ,  0775 ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( fd_  <  0  | |  HANDLE_EINTR ( flock ( fd_ ,  LOCK_EX ) )  <  0 )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      LOGE ( " Failed to lock file %s, errno=%d " ,  fn . c_str ( ) ,  errno ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ~ FileLock ( )  {  close ( fd_ ) ;  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								private : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  int  fd_  =  - 1 ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} ; 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								std : : unordered_map < std : : string ,  uint32_t >  keys  =  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " AccessToken " ,  CLEAR_ON_MANAGER_START  |  DONT_LOG } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " AlwaysOnDM " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " ApiCache_Device " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " AssistNowToken " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " AthenadPid " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " AthenadUploadQueue " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " AthenadRecentlyViewedRoutes " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " BootCount " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " CalibrationParams " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " CameraDebugExpGain " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " CameraDebugExpTime " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " CarBatteryCapacity " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " CarParams " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " CarParamsCache " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " CarParamsPersistent " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " CarParamsPrevRoute " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " CompletedTrainingVersion " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " ControlsReady " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " CurrentBootlog " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " CurrentRoute " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " DisableLogging " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " DisablePowerDown " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " DisableUpdates " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " DisengageOnAccelerator " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " DmModelInitialized " ,  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " DongleId " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " DoReboot " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " DoShutdown " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " DoUninstall " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " ExperimentalLongitudinalEnabled " ,  PERSISTENT  |  DEVELOPMENT_ONLY } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " ExperimentalMode " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " ExperimentalModeConfirmed " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " FirmwareQueryDone " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " ForcePowerDown " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " GitBranch " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " GitCommit " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " GitCommitDate " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " GitDiff " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " GithubSshKeys " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " GithubUsername " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " GitRemote " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " GsmApn " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " GsmMetered " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " GsmRoaming " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " HardwareSerial " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " HasAcceptedTerms " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " IMEI " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " InstallDate " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " IsDriverViewEnabled " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " IsEngaged " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " IsLdwEnabled " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " IsMetric " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " IsOffroad " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " IsOnroad " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " IsRhdDetected " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " IsReleaseBranch " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " IsTakingSnapshot " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " IsTestedBranch " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " JoystickDebugMode " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_OFFROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " LanguageSetting " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " LastAthenaPingTime " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " LastGPSPosition " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " LastManagerExitReason " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " LastOffroadStatusPacket " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_OFFROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " LastPowerDropDetected " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " LastUpdateException " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " LastUpdateTime " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " LiveParameters " ,  PERSISTENT } , 
  
						 
					
						
							
								
									
										
											 
										
											
												Live torque (#25456)
* wip torqued
* add basic logic
* setup in manager
* check sanity and publish msg
* add first order filter to outputs
* wire up controlsd, and update gains
* rename intercept to offset
* add cloudlog, live values are not updated
* fix bugs, do not reset points for now
* fix crashes
* rename to main
* fix bugs, works offline
* fix float in cereal bug
* add latacc filter
* randomly choose points, approx for iid
* add variable decay
* local param to capnp instead of dict
* verify works in replay
* use torqued output in controlsd
* use in controlsd; use points from past routes
* controlsd bugfix
* filter before updating gains, needs to be replaced
* save all points to ensure smooth transition across routes, revert friction factor to 1.5
* add filters to prevent noisy low-speed data points; improve fit sanity
* add engaged buffer
* revert lat_acc thresh
* use paramsd realtime process config
* make latacc-to-torque generic, and overrideable
* move freq to 4Hz, avoid storing in np.array, don't publish points in the message
* float instead of np
* remove constant while storing pts
* rename slope, offset to lat_accet_factor, offset
* resolve issues
* use camelcase in all capnp params
* use camelcase everywhere
* reduce latacc threshold or sanity, add car_sane todo, save points properly
* add and check tag
* write param to disk at end of route
* remove args
* rebase op, cereal
* save on exit
* restore default handler
* cpu usage check
* add to process replay
* handle reset better, reduce unnecessary computation
* always publish raw values - useful for debug
* regen routes
* update refs
* checks on cache restore
* check tuning vals too
* clean that up
* reduce cpu usage
* reduce cpu usage by 75%
* cleanup
* optimize further
* handle reset condition better, don't put points in init, use only in corolla
* bump cereal after rebasing
* update refs
* Update common/params.cc
Co-authored-by: Adeeb Shihadeh <adeebshihadeh@gmail.com>
* remove unnecessary checks
* Update RELEASES.md
Co-authored-by: Adeeb Shihadeh <adeebshihadeh@gmail.com>
											 
										 
										
											3 years ago 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " LiveTorqueParameters " ,  PERSISTENT  |  DONT_LOG } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " LongitudinalPersonality " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " NetworkMetered " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " ObdMultiplexingChanged " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " ObdMultiplexingEnabled " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " Offroad_BadNvme " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_CarUnrecognized " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_ConnectivityNeeded " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_ConnectivityNeededPrompt " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_IsTakingSnapshot " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_NeosUpdate " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_NoFirmware " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_Recalibration " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_StorageMissing " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_TemperatureTooHigh " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_UnofficialHardware " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Offroad_UpdateFailed " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " OpenpilotEnabledToggle " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " PandaHeartbeatLost " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_OFFROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " PandaSomResetTriggered " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_OFFROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " PandaSignatures " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " PrimeType " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " RecordFront " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " RecordFrontLock " ,  PERSISTENT } ,   // for the internal fleet
   
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " ReplayControlsState " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " RouteCount " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " SnoozeUpdate " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_OFFROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " SshEnabled " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " TermsVersion " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " TrainingVersion " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " UbloxAvailable " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " UpdateAvailable " ,  CLEAR_ON_MANAGER_START  |  CLEAR_ON_ONROAD_TRANSITION } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " UpdateFailedCount " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " UpdaterAvailableBranches " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " UpdaterCurrentDescription " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " UpdaterCurrentReleaseNotes " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " UpdaterFetchAvailable " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " UpdaterNewDescription " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " UpdaterNewReleaseNotes " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " UpdaterState " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " UpdaterTargetBranch " ,  CLEAR_ON_MANAGER_START } , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { " UpdaterLastFetchTime " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { " Version " ,  PERSISTENT } , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} ; 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  // namespace
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Params : : Params ( const  std : : string  & path )  { 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  params_prefix  =  " / "  +  util : : getenv ( " OPENPILOT_PREFIX " ,  " d " ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  params_path  =  ensure_params_path ( params_prefix ,  path ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Params : : ~ Params ( )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ( future . valid ( ) )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    future . wait ( ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  assert ( queue . empty ( ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								std : : vector < std : : string >  Params : : allKeys ( )  const  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  std : : vector < std : : string >  ret ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  for  ( auto  & p  :  keys )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ret . push_back ( p . first ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  return  ret ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								bool  Params : : checkKey ( const  std : : string  & key )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  return  keys . find ( key )  ! =  keys . end ( ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								ParamKeyType  Params : : getKeyType ( const  std : : string  & key )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  return  static_cast < ParamKeyType > ( keys [ key ] ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								int  Params : : put ( const  char *  key ,  const  char *  value ,  size_t  value_size )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // Information about safely and atomically writing a file: https://lwn.net/Articles/457667/
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // 1) Create temp file
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // 2) Write data to temp file
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // 3) fsync() the temp file
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // 4) rename the temp file to the real name
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // 5) fsync() the containing directory
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  std : : string  tmp_path  =  params_path  +  " /.tmp_value_XXXXXX " ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  int  tmp_fd  =  mkstemp ( ( char * ) tmp_path . c_str ( ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ( tmp_fd  <  0 )  return  - 1 ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  int  result  =  - 1 ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  do  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // Write value to temp.
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ssize_t  bytes_written  =  HANDLE_EINTR ( write ( tmp_fd ,  value ,  value_size ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( bytes_written  <  0  | |  ( size_t ) bytes_written  ! =  value_size )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      result  =  - 20 ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      break ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // fsync to force persist the changes.
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( ( result  =  fsync ( tmp_fd ) )  <  0 )  break ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    FileLock  file_lock ( params_path  +  " /.lock " ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // Move temp into place.
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( ( result  =  rename ( tmp_path . c_str ( ) ,  getParamPath ( key ) . c_str ( ) ) )  <  0 )  break ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // fsync parent directory
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    result  =  fsync_dir ( getParamPath ( ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }  while  ( false ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  close ( tmp_fd ) ; 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  if  ( result  ! =  0 )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    : : unlink ( tmp_path . c_str ( ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  return  result ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								int  Params : : remove ( const  std : : string  & key )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  FileLock  file_lock ( params_path  +  " /.lock " ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  int  result  =  unlink ( getParamPath ( key ) . c_str ( ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ( result  ! =  0 )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  result ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  return  fsync_dir ( getParamPath ( ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								std : : string  Params : : get ( const  std : : string  & key ,  bool  block )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ( ! block )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  util : : read_file ( getParamPath ( key ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }  else  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // blocking read until successful
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    params_do_exit  =  0 ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    void  ( * prev_handler_sigint ) ( int )  =  std : : signal ( SIGINT ,  params_sig_handler ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    void  ( * prev_handler_sigterm ) ( int )  =  std : : signal ( SIGTERM ,  params_sig_handler ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    std : : string  value ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    while  ( ! params_do_exit )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      if  ( value  =  util : : read_file ( getParamPath ( key ) ) ;  ! value . empty ( ) )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        break ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      util : : sleep_for ( 100 ) ;   // 0.1 s
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    std : : signal ( SIGINT ,  prev_handler_sigint ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    std : : signal ( SIGTERM ,  prev_handler_sigterm ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  value ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								std : : map < std : : string ,  std : : string >  Params : : readAll ( )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  FileLock  file_lock ( params_path  +  " /.lock " ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  return  util : : read_files_in_dir ( getParamPath ( ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								void  Params : : clearAll ( ParamKeyType  key_type )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  FileLock  file_lock ( params_path  +  " /.lock " ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  // 1) delete params of key_type
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // 2) delete files that are not defined in the keys.
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ( DIR  * d  =  opendir ( getParamPath ( ) . c_str ( ) ) )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    struct  dirent  * de  =  NULL ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    while  ( ( de  =  readdir ( d ) ) )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      if  ( de - > d_type  ! =  DT_DIR )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        auto  it  =  keys . find ( de - > d_name ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( it  = =  keys . end ( )  | |  ( it - > second  &  key_type ) )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          unlink ( getParamPath ( de - > d_name ) . c_str ( ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        } 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    closedir ( d ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  fsync_dir ( getParamPath ( ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								void  Params : : putNonBlocking ( const  std : : string  & key ,  const  std : : string  & val )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								   queue . push ( std : : make_pair ( key ,  val ) ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // start thread on demand
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ( ! future . valid ( )  | |  future . wait_for ( std : : chrono : : milliseconds ( 0 ) )  = =  std : : future_status : : ready )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    future  =  std : : async ( std : : launch : : async ,  & Params : : asyncWriteThread ,  this ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								} 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								void  Params : : asyncWriteThread ( )  { 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  // TODO: write the latest one if a key has multiple values in the queue.
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  std : : pair < std : : string ,  std : : string >  p ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  while  ( queue . try_pop ( p ,  0 ) )  { 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // Params::put is Thread-Safe
   
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    put ( p . first ,  p . second ) ; 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}