/*! * \file pvt_solution.cc * \brief Implementation of a base class for a PVT solution * \author Carles Fernandez-Prades, 2015. cfernandez(at)cttc.es * * * ------------------------------------------------------------------------- * * Copyright (C) 2010-2020 (see AUTHORS file for a list of contributors) * * GNSS-SDR is a software defined Global Navigation * Satellite Systems receiver * * This file is part of GNSS-SDR. * * SPDX-License-Identifier: GPL-3.0-or-later * * ------------------------------------------------------------------------- */ #include "pvt_solution.h" #include "MATH_CONSTANTS.h" #include #include #include Pvt_Solution::Pvt_Solution() { d_latitude_d = 0.0; d_longitude_d = 0.0; d_height_m = 0.0; d_speed_over_ground_m_s = 0.0; d_course_over_ground_d = 0.0; d_avg_latitude_d = 0.0; d_avg_longitude_d = 0.0; d_avg_height_m = 0.0; d_flag_averaging = false; b_valid_position = false; d_averaging_depth = 0; d_valid_observations = 0; d_rx_dt_s = 0.0; d_rx_clock_drift_ppm = 0.0; d_pre_2009_file = false; // disabled by default } int Pvt_Solution::cart2geo(double X, double Y, double Z, int elipsoid_selection) { /* Conversion of Cartesian coordinates (X,Y,Z) to geographical coordinates (latitude, longitude, h) on a selected reference ellipsoid. Choices of Reference Ellipsoid for Geographical Coordinates 0. International Ellipsoid 1924 1. International Ellipsoid 1967 2. World Geodetic System 1972 3. Geodetic Reference System 1980 4. World Geodetic System 1984 */ const std::array a = {6378388.0, 6378160.0, 6378135.0, 6378137.0, 6378137.0}; const std::array f = {1.0 / 297.0, 1.0 / 298.247, 1.0 / 298.26, 1.0 / 298.257222101, 1.0 / 298.257223563}; const double lambda = atan2(Y, X); const double ex2 = (2.0 - f[elipsoid_selection]) * f[elipsoid_selection] / ((1.0 - f[elipsoid_selection]) * (1.0 - f[elipsoid_selection])); const double c = a[elipsoid_selection] * sqrt(1.0 + ex2); double phi = atan(Z / ((sqrt(X * X + Y * Y) * (1.0 - (2.0 - f[elipsoid_selection])) * f[elipsoid_selection]))); double h = 0.1; double oldh = 0.0; double N; int iterations = 0; do { oldh = h; N = c / sqrt(1 + ex2 * (cos(phi) * cos(phi))); phi = atan(Z / ((sqrt(X * X + Y * Y) * (1.0 - (2.0 - f[elipsoid_selection]) * f[elipsoid_selection] * N / (N + h))))); h = sqrt(X * X + Y * Y) / cos(phi) - N; iterations = iterations + 1; if (iterations > 100) { DLOG(WARNING) << "Failed to approximate h with desired precision. h-oldh= " << h - oldh; break; } } while (std::abs(h - oldh) > 1.0e-12); d_latitude_d = phi * R2D; d_longitude_d = lambda * R2D; d_height_m = h; return 0; } void Pvt_Solution::set_averaging_depth(int depth) { d_averaging_depth = depth; } void Pvt_Solution::set_averaging_flag(bool flag) { d_flag_averaging = flag; } void Pvt_Solution::perform_pos_averaging() { // MOVING AVERAGE PVT bool avg = d_flag_averaging; if (avg == true) { if (d_hist_longitude_d.size() == static_cast(d_averaging_depth)) { // Pop oldest value d_hist_longitude_d.pop_back(); d_hist_latitude_d.pop_back(); d_hist_height_m.pop_back(); // Push new values d_hist_longitude_d.push_front(d_longitude_d); d_hist_latitude_d.push_front(d_latitude_d); d_hist_height_m.push_front(d_height_m); d_avg_latitude_d = 0.0; d_avg_longitude_d = 0.0; d_avg_height_m = 0.0; for (size_t i = 0; i < d_hist_longitude_d.size(); i++) { d_avg_latitude_d = d_avg_latitude_d + d_hist_latitude_d.at(i); d_avg_longitude_d = d_avg_longitude_d + d_hist_longitude_d.at(i); d_avg_height_m = d_avg_height_m + d_hist_height_m.at(i); } d_avg_latitude_d = d_avg_latitude_d / static_cast(d_averaging_depth); d_avg_longitude_d = d_avg_longitude_d / static_cast(d_averaging_depth); d_avg_height_m = d_avg_height_m / static_cast(d_averaging_depth); b_valid_position = true; } else { // int current_depth=d_hist_longitude_d.size(); // Push new values d_hist_longitude_d.push_front(d_longitude_d); d_hist_latitude_d.push_front(d_latitude_d); d_hist_height_m.push_front(d_height_m); d_avg_latitude_d = d_latitude_d; d_avg_longitude_d = d_longitude_d; d_avg_height_m = d_height_m; b_valid_position = false; } } else { b_valid_position = true; } } double Pvt_Solution::get_time_offset_s() const { return d_rx_dt_s; } void Pvt_Solution::set_time_offset_s(double offset) { d_rx_dt_s = offset; } double Pvt_Solution::get_clock_drift_ppm() const { return d_rx_clock_drift_ppm; } void Pvt_Solution::set_clock_drift_ppm(double clock_drift_ppm) { d_rx_clock_drift_ppm = clock_drift_ppm; } double Pvt_Solution::get_latitude() const { return d_latitude_d; } double Pvt_Solution::get_longitude() const { return d_longitude_d; } double Pvt_Solution::get_height() const { return d_height_m; } double Pvt_Solution::get_speed_over_ground() const { return d_speed_over_ground_m_s; } void Pvt_Solution::set_speed_over_ground(double speed_m_s) { d_speed_over_ground_m_s = speed_m_s; } void Pvt_Solution::set_course_over_ground(double cog_deg) { d_course_over_ground_d = cog_deg; } double Pvt_Solution::get_course_over_ground() const { return d_course_over_ground_d; } double Pvt_Solution::get_avg_latitude() const { return d_avg_latitude_d; } double Pvt_Solution::get_avg_longitude() const { return d_avg_longitude_d; } double Pvt_Solution::get_avg_height() const { return d_avg_height_m; } bool Pvt_Solution::is_averaging() const { return d_flag_averaging; } bool Pvt_Solution::is_valid_position() const { return b_valid_position; } void Pvt_Solution::set_valid_position(bool is_valid) { b_valid_position = is_valid; } void Pvt_Solution::set_rx_pos(const std::array &pos) { d_rx_pos = pos; Pvt_Solution::cart2geo(d_rx_pos[0], d_rx_pos[1], d_rx_pos[2], 4); } std::array Pvt_Solution::get_rx_pos() const { return d_rx_pos; } void Pvt_Solution::set_rx_vel(const std::array &vel) { d_rx_vel = vel; } std::array Pvt_Solution::get_rx_vel() const { return d_rx_vel; } boost::posix_time::ptime Pvt_Solution::get_position_UTC_time() const { return d_position_UTC_time; } void Pvt_Solution::set_position_UTC_time(const boost::posix_time::ptime &pt) { d_position_UTC_time = pt; } int Pvt_Solution::get_num_valid_observations() const { return d_valid_observations; } void Pvt_Solution::set_num_valid_observations(int num) { d_valid_observations = num; } void Pvt_Solution::set_pre_2009_file(bool pre_2009_file) { d_pre_2009_file = pre_2009_file; } bool Pvt_Solution::is_pre_2009() const { return d_pre_2009_file; }