|  |  | @ -70,7 +70,7 @@ static VectorXd rotate_std(const MatrixXdr& rot_matrix, const VectorXd& std_in) | 
			
		
	
		
		
			
				
					
					|  |  |  |   return rotate_cov(rot_matrix, std_in.array().square().matrix().asDiagonal()).diagonal().array().sqrt(); |  |  |  |   return rotate_cov(rot_matrix, std_in.array().square().matrix().asDiagonal()).diagonal().array().sqrt(); | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | Localizer::Localizer() { |  |  |  | Localizer::Localizer(LocalizerGnssSource gnss_source) { | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |   this->kf = std::make_unique<LiveKalman>(); |  |  |  |   this->kf = std::make_unique<LiveKalman>(); | 
			
		
	
		
		
			
				
					
					|  |  |  |   this->reset_kalman(); |  |  |  |   this->reset_kalman(); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | @ -84,10 +84,9 @@ Localizer::Localizer() { | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   VectorXd ecef_pos = this->kf->get_x().segment<STATE_ECEF_POS_LEN>(STATE_ECEF_POS_START); |  |  |  |   VectorXd ecef_pos = this->kf->get_x().segment<STATE_ECEF_POS_LEN>(STATE_ECEF_POS_START); | 
			
		
	
		
		
			
				
					
					|  |  |  |   this->converter = std::make_unique<LocalCoord>((ECEF) { .x = ecef_pos[0], .y = ecef_pos[1], .z = ecef_pos[2] }); |  |  |  |   this->converter = std::make_unique<LocalCoord>((ECEF) { .x = ecef_pos[0], .y = ecef_pos[1], .z = ecef_pos[2] }); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   this->configure_gnss_source(gnss_source); | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | Localizer::Localizer(bool has_ublox) : Localizer() { ublox_available = has_ublox; } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | void Localizer::build_live_location(cereal::LiveLocationKalman::Builder& fix) { |  |  |  | void Localizer::build_live_location(cereal::LiveLocationKalman::Builder& fix) { | 
			
		
	
		
		
			
				
					
					|  |  |  |   VectorXd predicted_state = this->kf->get_x(); |  |  |  |   VectorXd predicted_state = this->kf->get_x(); | 
			
		
	
		
		
			
				
					
					|  |  |  |   MatrixXdr predicted_cov = this->kf->get_P(); |  |  |  |   MatrixXdr predicted_cov = this->kf->get_P(); | 
			
		
	
	
		
		
			
				
					|  |  | @ -325,12 +324,7 @@ void Localizer::handle_gps(double current_time, const cereal::GpsLocationData::R | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   VectorXd ecef_pos = this->converter->ned2ecef({ 0.0, 0.0, 0.0 }).to_vector(); |  |  |  |   VectorXd ecef_pos = this->converter->ned2ecef({ 0.0, 0.0, 0.0 }).to_vector(); | 
			
		
	
		
		
			
				
					
					|  |  |  |   VectorXd ecef_vel = this->converter->ned2ecef({ log.getVNED()[0], log.getVNED()[1], log.getVNED()[2] }).to_vector() - ecef_pos; |  |  |  |   VectorXd ecef_vel = this->converter->ned2ecef({ log.getVNED()[0], log.getVNED()[1], log.getVNED()[2] }).to_vector() - ecef_pos; | 
			
		
	
		
		
			
				
					
					|  |  |  |   float ecef_pos_std; |  |  |  |   float ecef_pos_std = std::sqrt(this->gps_variance_factor * std::pow(log.getAccuracy(), 2) + this->gps_vertical_variance_factor * std::pow(log.getVerticalAccuracy(), 2)); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |   if (ublox_available) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     ecef_pos_std = std::sqrt(std::pow(log.getAccuracy(), 2) + std::pow(log.getVerticalAccuracy(), 2)); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   } else { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     ecef_pos_std = std::sqrt(3 * std::pow(log.getVerticalAccuracy(), 2)); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   } |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |   MatrixXdr ecef_pos_R = Vector3d::Constant(std::pow(this->gps_std_factor * ecef_pos_std, 2)).asDiagonal(); |  |  |  |   MatrixXdr ecef_pos_R = Vector3d::Constant(std::pow(this->gps_std_factor * ecef_pos_std, 2)).asDiagonal(); | 
			
		
	
		
		
			
				
					
					|  |  |  |   MatrixXdr ecef_vel_R = Vector3d::Constant(std::pow(this->gps_std_factor * log.getSpeedAccuracy(), 2)).asDiagonal(); |  |  |  |   MatrixXdr ecef_vel_R = Vector3d::Constant(std::pow(this->gps_std_factor * log.getSpeedAccuracy(), 2)).asDiagonal(); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | @ -372,10 +366,7 @@ void Localizer::handle_gnss(double current_time, const cereal::GnssMeasurements: | 
			
		
	
		
		
			
				
					
					|  |  |  |   } |  |  |  |   } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   double sensor_time = log.getMeasTime() * 1e-9; |  |  |  |   double sensor_time = log.getMeasTime() * 1e-9; | 
			
		
	
		
		
			
				
					
					|  |  |  |   if (ublox_available) |  |  |  |   sensor_time -= this->gps_time_offset; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     sensor_time -= GPS_UBLOX_SENSOR_TIME_OFFSET; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   else |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     sensor_time -= GPS_QUECTEL_SENSOR_TIME_OFFSET; |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   auto ecef_pos_v = log.getPositionECEF().getValue(); |  |  |  |   auto ecef_pos_v = log.getPositionECEF().getValue(); | 
			
		
	
		
		
			
				
					
					|  |  |  |   VectorXd ecef_pos = Vector3d(ecef_pos_v[0], ecef_pos_v[1], ecef_pos_v[2]); |  |  |  |   VectorXd ecef_pos = Vector3d(ecef_pos_v[0], ecef_pos_v[1], ecef_pos_v[2]); | 
			
		
	
	
		
		
			
				
					|  |  | @ -664,16 +655,33 @@ void Localizer::determine_gps_mode(double current_time) { | 
			
		
	
		
		
			
				
					
					|  |  |  |   } |  |  |  |   } | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | void Localizer::configure_gnss_source(LocalizerGnssSource source) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   this->gnss_source = source; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   if (source == LocalizerGnssSource::UBLOX) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     this->gps_std_factor = 10.0; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     this->gps_variance_factor = 1.0; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     this->gps_vertical_variance_factor = 1.0; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     this->gps_time_offset = GPS_UBLOX_SENSOR_TIME_OFFSET; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   } else { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     this->gps_std_factor = 2.0; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     this->gps_variance_factor = 0.0; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     this->gps_vertical_variance_factor = 3.0; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     this->gps_time_offset = GPS_QUECTEL_SENSOR_TIME_OFFSET; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   } | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | int Localizer::locationd_thread() { |  |  |  | int Localizer::locationd_thread() { | 
			
		
	
		
		
			
				
					
					|  |  |  |   ublox_available = Params().getBool("UbloxAvailable", true); |  |  |  |   LocalizerGnssSource source; | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |   const char* gps_location_socket; |  |  |  |   const char* gps_location_socket; | 
			
		
	
		
		
			
				
					
					|  |  |  |   if (ublox_available) { |  |  |  |   if (Params().getBool("UbloxAvailable", true)) { | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     source = LocalizerGnssSource::UBLOX; | 
			
		
	
		
		
			
				
					
					|  |  |  |     gps_location_socket = "gpsLocationExternal"; |  |  |  |     gps_location_socket = "gpsLocationExternal"; | 
			
		
	
		
		
			
				
					
					|  |  |  |     this->gps_std_factor = 10.0; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   } else { |  |  |  |   } else { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     source = LocalizerGnssSource::QCOM; | 
			
		
	
		
		
			
				
					
					|  |  |  |     gps_location_socket = "gpsLocation"; |  |  |  |     gps_location_socket = "gpsLocation"; | 
			
		
	
		
		
			
				
					
					|  |  |  |     this->gps_std_factor = 2.0; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   } |  |  |  |   } | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   this->configure_gnss_source(source); | 
			
		
	
		
		
			
				
					
					|  |  |  |   const std::initializer_list<const char *> service_list = {gps_location_socket, "cameraOdometry", "liveCalibration", |  |  |  |   const std::initializer_list<const char *> service_list = {gps_location_socket, "cameraOdometry", "liveCalibration", | 
			
		
	
		
		
			
				
					
					|  |  |  |                                                           "carState", "carParams", "accelerometer", "gyroscope"}; |  |  |  |                                                           "carState", "carParams", "accelerometer", "gyroscope"}; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | 
 |