@ -24,7 +24,7 @@ SOURCES = ['lead0', 'lead1', 'cruise', 'e2e'] 
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					X_DIM  =  3 X_DIM  =  3  
			
		
	
		
		
			
				
					
					U_DIM  =  1 U_DIM  =  1  
			
		
	
		
		
			
				
					
					PARAM_DIM  =  4 PARAM_DIM  =  6  
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					COST_E_DIM  =  5 COST_E_DIM  =  5  
			
		
	
		
		
			
				
					
					COST_DIM  =  COST_E_DIM  +  1 COST_DIM  =  COST_E_DIM  +  1  
			
		
	
		
		
			
				
					
					CONSTR_DIM  =  4 CONSTR_DIM  =  4  
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -37,6 +37,7 @@ J_EGO_COST = 5.0 
			
		
	
		
		
			
				
					
					A_CHANGE_COST  =  200. A_CHANGE_COST  =  200.  
			
		
	
		
		
			
				
					
					DANGER_ZONE_COST  =  100. DANGER_ZONE_COST  =  100.  
			
		
	
		
		
			
				
					
					CRASH_DISTANCE  =  .5 CRASH_DISTANCE  =  .5  
			
		
	
		
		
			
				
					
					LEAD_DANGER_FACTOR  =  0.75  
			
		
	
		
		
			
				
					
					LIMIT_COST  =  1e6 LIMIT_COST  =  1e6  
			
		
	
		
		
			
				
					
					ACADOS_SOLVER_TYPE  =  ' SQP_RTI ' ACADOS_SOLVER_TYPE  =  ' SQP_RTI '  
			
		
	
		
		
			
				
					
					
 
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -58,8 +59,8 @@ STOP_DISTANCE = 6.0 
			
		
	
		
		
			
				
					
					def  get_stopped_equivalence_factor ( v_lead ) : def  get_stopped_equivalence_factor ( v_lead ) :  
			
		
	
		
		
			
				
					
					  return  ( v_lead * * 2 )  /  ( 2  *  COMFORT_BRAKE )    return  ( v_lead * * 2 )  /  ( 2  *  COMFORT_BRAKE )   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					def  get_safe_obstacle_distance ( v_ego ) : def  get_safe_obstacle_distance ( v_ego ,  t_follow = T_FOLLOW ) :  
			
				
				
			
		
	
		
		
			
				
					
					  return  ( v_ego * * 2 )  /  ( 2  *  COMFORT_BRAKE )  +  T_FOLLOW *  v_ego  +  STOP_DISTANCE    return  ( v_ego * * 2 )  /  ( 2  *  COMFORT_BRAKE )  +  t_follow *  v_ego  +  STOP_DISTANCE   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					def  desired_follow_distance ( v_ego ,  v_lead ) : def  desired_follow_distance ( v_ego ,  v_lead ) :  
			
		
	
		
		
			
				
					
					  return  get_safe_obstacle_distance ( v_ego )  -  get_stopped_equivalence_factor ( v_lead )    return  get_safe_obstacle_distance ( v_ego )  -  get_stopped_equivalence_factor ( v_lead )   
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -90,7 +91,9 @@ def gen_long_model(): 
			
		
	
		
		
			
				
					
					  a_max  =  SX . sym ( ' a_max ' )    a_max  =  SX . sym ( ' a_max ' )   
			
		
	
		
		
			
				
					
					  x_obstacle  =  SX . sym ( ' x_obstacle ' )    x_obstacle  =  SX . sym ( ' x_obstacle ' )   
			
		
	
		
		
			
				
					
					  prev_a  =  SX . sym ( ' prev_a ' )    prev_a  =  SX . sym ( ' prev_a ' )   
			
		
	
		
		
			
				
					
					  model . p  =  vertcat ( a_min ,  a_max ,  x_obstacle ,  prev_a )    lead_t_follow  =  SX . sym ( ' lead_t_follow ' )   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  lead_danger_factor  =  SX . sym ( ' lead_danger_factor ' )   
			
		
	
		
		
			
				
					
					  model . p  =  vertcat ( a_min ,  a_max ,  x_obstacle ,  prev_a ,  lead_t_follow ,  lead_danger_factor )   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  # dynamics model    # dynamics model   
			
		
	
		
		
			
				
					
					  f_expl  =  vertcat ( v_ego ,  a_ego ,  j_ego )    f_expl  =  vertcat ( v_ego ,  a_ego ,  j_ego )   
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -124,11 +127,13 @@ def gen_long_ocp(): 
			
		
	
		
		
			
				
					
					  a_min ,  a_max  =  ocp . model . p [ 0 ] ,  ocp . model . p [ 1 ]    a_min ,  a_max  =  ocp . model . p [ 0 ] ,  ocp . model . p [ 1 ]   
			
		
	
		
		
			
				
					
					  x_obstacle  =  ocp . model . p [ 2 ]    x_obstacle  =  ocp . model . p [ 2 ]   
			
		
	
		
		
			
				
					
					  prev_a  =  ocp . model . p [ 3 ]    prev_a  =  ocp . model . p [ 3 ]   
			
		
	
		
		
			
				
					
					  lead_t_follow  =  ocp . model . p [ 4 ]   
			
		
	
		
		
			
				
					
					  lead_danger_factor  =  ocp . model . p [ 5 ]   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ocp . cost . yref  =  np . zeros ( ( COST_DIM ,  ) )    ocp . cost . yref  =  np . zeros ( ( COST_DIM ,  ) )   
			
		
	
		
		
			
				
					
					  ocp . cost . yref_e  =  np . zeros ( ( COST_E_DIM ,  ) )    ocp . cost . yref_e  =  np . zeros ( ( COST_E_DIM ,  ) )   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  desired_dist_comfort  =  get_safe_obstacle_distance ( v_ego )    desired_dist_comfort  =  get_safe_obstacle_distance ( v_ego ,  lead_t_follow )   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  # The main cost in normal operation is how close you are to the "desired" distance    # The main cost in normal operation is how close you are to the "desired" distance   
			
		
	
		
		
			
				
					
					  # from an obstacle at every timestep. This obstacle can be a lead car    # from an obstacle at every timestep. This obstacle can be a lead car   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -149,12 +154,12 @@ def gen_long_ocp(): 
			
		
	
		
		
			
				
					
					  constraints  =  vertcat ( v_ego ,    constraints  =  vertcat ( v_ego ,   
			
		
	
		
		
			
				
					
					                        ( a_ego  -  a_min ) ,                          ( a_ego  -  a_min ) ,   
			
		
	
		
		
			
				
					
					                        ( a_max  -  a_ego ) ,                          ( a_max  -  a_ego ) ,   
			
		
	
		
		
			
				
					
					                        ( ( x_obstacle  -  x_ego )  -  ( 3 / 4 ) *  ( desired_dist_comfort ) )  /  ( v_ego  +  10. ) )                          ( ( x_obstacle  -  x_ego )  -  lead_danger_factor *  ( desired_dist_comfort ) )  /  ( v_ego  +  10. ) )   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  ocp . model . con_h_expr  =  constraints    ocp . model . con_h_expr  =  constraints   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  x0  =  np . zeros ( X_DIM )    x0  =  np . zeros ( X_DIM )   
			
		
	
		
		
			
				
					
					  ocp . constraints . x0  =  x0    ocp . constraints . x0  =  x0   
			
		
	
		
		
			
				
					
					  ocp . parameter_values  =  np . array ( [ - 1.2 ,  1.2 ,  0.0 ,  0.0 ] )    ocp . parameter_values  =  np . array ( [ - 1.2 ,  1.2 ,  0.0 ,  0.0 ,  T_FOLLOW ,  LEAD_DANGER_FACTOR ] )   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  # We put all constraint cost weights to 0 and only set them at runtime    # We put all constraint cost weights to 0 and only set them at runtime   
			
		
	
		
		
			
				
					
					  cost_weights  =  np . zeros ( CONSTR_DIM )    cost_weights  =  np . zeros ( CONSTR_DIM )   
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -249,7 +254,7 @@ class LongitudinalMpc: 
			
		
	
		
		
			
				
					
					      constraint_cost_weights  =  [ LIMIT_COST ,  LIMIT_COST ,  LIMIT_COST ,  DANGER_ZONE_COST ]        constraint_cost_weights  =  [ LIMIT_COST ,  LIMIT_COST ,  LIMIT_COST ,  DANGER_ZONE_COST ]   
			
		
	
		
		
			
				
					
					    elif  self . mode  ==  ' blended ' :      elif  self . mode  ==  ' blended ' :   
			
		
	
		
		
			
				
					
					      cost_weights  =  [ 0. ,  0.2 ,  0.25 ,  1.0 ,  0.0 ,  1.0 ]        cost_weights  =  [ 0. ,  0.2 ,  0.25 ,  1.0 ,  0.0 ,  1.0 ]   
			
		
	
		
		
			
				
					
					      constraint_cost_weights  =  [ LIMIT_COST ,  LIMIT_COST ,  LIMIT_COST ,  5.0 ]        constraint_cost_weights  =  [ LIMIT_COST ,  LIMIT_COST ,  LIMIT_COST ,  50 .0 ]   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    elif  self . mode  ==  ' e2e ' :      elif  self . mode  ==  ' e2e ' :   
			
		
	
		
		
			
				
					
					      cost_weights  =  [ 0. ,  0.2 ,  0.25 ,  1. ,  0.0 ,  .1 ]        cost_weights  =  [ 0. ,  0.2 ,  0.25 ,  1. ,  0.0 ,  .1 ]   
			
		
	
		
		
			
				
					
					      constraint_cost_weights  =  [ LIMIT_COST ,  LIMIT_COST ,  LIMIT_COST ,  0.0 ]        constraint_cost_weights  =  [ LIMIT_COST ,  LIMIT_COST ,  LIMIT_COST ,  0.0 ]   
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -317,6 +322,7 @@ class LongitudinalMpc: 
			
		
	
		
		
			
				
					
					    if  self . mode  ==  ' acc ' :      if  self . mode  ==  ' acc ' :   
			
		
	
		
		
			
				
					
					      self . params [ : , 0 ]  =  MIN_ACCEL  if  self . status  else  self . cruise_min_a        self . params [ : , 0 ]  =  MIN_ACCEL  if  self . status  else  self . cruise_min_a   
			
		
	
		
		
			
				
					
					      self . params [ : , 1 ]  =  self . cruise_max_a        self . params [ : , 1 ]  =  self . cruise_max_a   
			
		
	
		
		
			
				
					
					      self . params [ : , 5 ]  =  LEAD_DANGER_FACTOR   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					      # Fake an obstacle for cruise, this ensures smooth acceleration to set speed        # Fake an obstacle for cruise, this ensures smooth acceleration to set speed   
			
		
	
		
		
			
				
					
					      # when the leads are no factor.        # when the leads are no factor.   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -335,6 +341,7 @@ class LongitudinalMpc: 
			
		
	
		
		
			
				
					
					    elif  self . mode  ==  ' blended ' :      elif  self . mode  ==  ' blended ' :   
			
		
	
		
		
			
				
					
					      self . params [ : , 0 ]  =  MIN_ACCEL        self . params [ : , 0 ]  =  MIN_ACCEL   
			
		
	
		
		
			
				
					
					      self . params [ : , 1 ]  =  MAX_ACCEL        self . params [ : , 1 ]  =  MAX_ACCEL   
			
		
	
		
		
			
				
					
					      self . params [ : , 5 ]  =  1.0   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					      x_obstacles  =  np . column_stack ( [ lead_0_obstacle ,        x_obstacles  =  np . column_stack ( [ lead_0_obstacle ,   
			
		
	
		
		
			
				
					
					                                     lead_1_obstacle ] )                                       lead_1_obstacle ] )   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -344,7 +351,8 @@ class LongitudinalMpc: 
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					      x_and_cruise  =  np . column_stack ( [ x ,  cruise_target ] )        x_and_cruise  =  np . column_stack ( [ x ,  cruise_target ] )   
			
		
	
		
		
			
				
					
					      x  =  np . min ( x_and_cruise ,  axis = 1 )        x  =  np . min ( x_and_cruise ,  axis = 1 )   
			
		
	
		
		
			
				
					
					      self . source  =  ' e2e '          
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					      self . source  =  ' e2e '  if  x_and_cruise [ 0 , 0 ]  <  x_and_cruise [ 0 , 1 ]  else  ' cruise '   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    else :      else :   
			
		
	
		
		
			
				
					
					      raise  NotImplementedError ( f ' Planner mode  { self . mode }  not recognized in planner update ' )        raise  NotImplementedError ( f ' Planner mode  { self . mode }  not recognized in planner update ' )   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -359,6 +367,7 @@ class LongitudinalMpc: 
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    self . params [ : , 2 ]  =  np . min ( x_obstacles ,  axis = 1 )      self . params [ : , 2 ]  =  np . min ( x_obstacles ,  axis = 1 )   
			
		
	
		
		
			
				
					
					    self . params [ : , 3 ]  =  np . copy ( self . prev_a )      self . params [ : , 3 ]  =  np . copy ( self . prev_a )   
			
		
	
		
		
			
				
					
					    self . params [ : , 4 ]  =  T_FOLLOW   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    self . run ( )      self . run ( )   
			
		
	
		
		
			
				
					
					    if  ( np . any ( lead_xv_0 [ : , 0 ]  -  self . x_sol [ : , 0 ]  <  CRASH_DISTANCE )  and      if  ( np . any ( lead_xv_0 [ : , 0 ]  -  self . x_sol [ : , 0 ]  <  CRASH_DISTANCE )  and   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -367,10 +376,23 @@ class LongitudinalMpc: 
			
		
	
		
		
			
				
					
					    else :      else :   
			
		
	
		
		
			
				
					
					      self . crash_cnt  =  0        self . crash_cnt  =  0   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    # Check if it got within lead comfort range   
			
		
	
		
		
			
				
					
					    # TODO This should be done cleaner   
			
		
	
		
		
			
				
					
					    if  self . mode  ==  ' blended ' :   
			
		
	
		
		
			
				
					
					      if  any ( ( lead_0_obstacle  -  get_safe_obstacle_distance ( self . x_sol [ : , 1 ] ,  T_FOLLOW ) ) -  self . x_sol [ : , 0 ]  <  0.0 ) :   
			
		
	
		
		
			
				
					
					        self . source  =  ' lead0 '   
			
		
	
		
		
			
				
					
					      if  any ( ( lead_1_obstacle  -  get_safe_obstacle_distance ( self . x_sol [ : , 1 ] ,  T_FOLLOW ) ) -  self . x_sol [ : , 0 ]  <  0.0 )  and  \  
			
		
	
		
		
			
				
					
					         ( lead_1_obstacle [ 0 ]  -  lead_0_obstacle [ 0 ] ) :   
			
		
	
		
		
			
				
					
					        self . source  =  ' lead1 '   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					        
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  def  update_with_xva ( self ,  x ,  v ,  a ) :    def  update_with_xva ( self ,  x ,  v ,  a ) :   
			
		
	
		
		
			
				
					
					    self . params [ : , 0 ]  =  - 10.      self . params [ : , 0 ]  =  - 10.   
			
		
	
		
		
			
				
					
					    self . params [ : , 1 ]  =  10.      self . params [ : , 1 ]  =  10.   
			
		
	
		
		
			
				
					
					    self . params [ : , 2 ]  =  1e5      self . params [ : , 2 ]  =  1e5   
			
		
	
		
		
			
				
					
					    self . params [ : , 4 ]  =  T_FOLLOW   
			
		
	
		
		
			
				
					
					    self . params [ : , 5 ]  =  LEAD_DANGER_FACTOR   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    # v, and a are in local frame, but x is wrt the x[0] position      # v, and a are in local frame, but x is wrt the x[0] position   
			
		
	
		
		
			
				
					
					    # In >90degree turns, x goes to 0 (and may even be -ve)      # In >90degree turns, x goes to 0 (and may even be -ve)