/*! * \file gps_cnav_navigation_message.cc * \brief Implementation of a GPS CNAV Data message decoder as described in IS-GPS-200M * * See https://www.gps.gov/technical/icwg/IS-GPS-200M.pdf Appendix III * \author Javier Arribas, 2015. jarribas(at)cttc.es * * ----------------------------------------------------------------------------- * * GNSS-SDR is a Global Navigation Satellite System software-defined receiver. * This file is part of GNSS-SDR. * * Copyright (C) 2010-2020 (see AUTHORS file for a list of contributors) * SPDX-License-Identifier: GPL-3.0-or-later * * ----------------------------------------------------------------------------- */ #include "gps_cnav_navigation_message.h" #include "gnss_satellite.h" #include // for std::sqrt #include // for std::numeric_limits Gps_CNAV_Navigation_Message::Gps_CNAV_Navigation_Message() { Gnss_Satellite gnss_satellite_ = Gnss_Satellite(); for (uint32_t prn_ = 1; prn_ < 33; prn_++) { satelliteBlock[prn_] = gnss_satellite_.what_block("GPS", prn_); } b_flag_iono_valid = false; b_flag_utc_valid = false; } bool Gps_CNAV_Navigation_Message::read_navigation_bool(const std::bitset& bits, const std::vector>& parameter) const { bool value = bits[GPS_CNAV_DATA_PAGE_BITS - parameter[0].first]; return value; } uint64_t Gps_CNAV_Navigation_Message::read_navigation_unsigned(const std::bitset& bits, const std::vector>& parameter) const { uint64_t value = 0ULL; for (const auto& p : parameter) { for (int32_t j = 0; j < p.second; j++) { value = (value << 1) | static_cast(bits[GPS_CNAV_DATA_PAGE_BITS - p.first - j]); } } return value; } int64_t Gps_CNAV_Navigation_Message::read_navigation_signed(const std::bitset& bits, const std::vector>& parameter) const { int64_t value = (bits[GPS_CNAV_DATA_PAGE_BITS - parameter[0].first] == 1) ? -1LL : 0LL; for (const auto& p : parameter) { for (int32_t j = 0; j < p.second; j++) { value = (value << 1) | static_cast(bits[GPS_CNAV_DATA_PAGE_BITS - p.first - j]); } } return value; } void Gps_CNAV_Navigation_Message::decode_page(const std::bitset& data_bits) { int32_t page_type; bool alert_flag; // common to all messages const auto PRN = static_cast(read_navigation_unsigned(data_bits, CNAV_PRN)); ephemeris_record.PRN = PRN; d_TOW = static_cast(read_navigation_unsigned(data_bits, CNAV_TOW)); d_TOW *= CNAV_TOW_LSB; ephemeris_record.tow = d_TOW; alert_flag = read_navigation_bool(data_bits, CNAV_ALERT_FLAG); ephemeris_record.alert_flag = alert_flag; page_type = static_cast(read_navigation_unsigned(data_bits, CNAV_MSG_TYPE)); switch (page_type) { case 10: // Ephemeris 1/2 ephemeris_record.WN = static_cast(read_navigation_unsigned(data_bits, CNAV_WN)); ephemeris_record.signal_health = static_cast(read_navigation_unsigned(data_bits, CNAV_HEALTH)); ephemeris_record.top = static_cast(read_navigation_unsigned(data_bits, CNAV_TOP1)); ephemeris_record.top *= CNAV_TOP1_LSB; ephemeris_record.URA0 = static_cast(read_navigation_signed(data_bits, CNAV_URA)); ephemeris_record.toe1 = static_cast(read_navigation_unsigned(data_bits, CNAV_TOE1)); ephemeris_record.toe1 *= CNAV_TOE1_LSB; ephemeris_record.delta_A = static_cast(read_navigation_signed(data_bits, CNAV_DELTA_A)); ephemeris_record.delta_A *= CNAV_DELTA_A_LSB; ephemeris_record.Adot = static_cast(read_navigation_signed(data_bits, CNAV_A_DOT)); ephemeris_record.Adot *= CNAV_A_DOT_LSB; ephemeris_record.sqrtA = std::sqrt(CNAV_A_REF + ephemeris_record.delta_A); ephemeris_record.delta_n = static_cast(read_navigation_signed(data_bits, CNAV_DELTA_N0)); ephemeris_record.delta_n *= CNAV_DELTA_N0_LSB; ephemeris_record.delta_ndot = static_cast(read_navigation_signed(data_bits, CNAV_DELTA_N0_DOT)); ephemeris_record.delta_ndot *= CNAV_DELTA_N0_DOT_LSB; ephemeris_record.M_0 = static_cast(read_navigation_signed(data_bits, CNAV_M0)); ephemeris_record.M_0 *= CNAV_M0_LSB; ephemeris_record.ecc = static_cast(read_navigation_unsigned(data_bits, CNAV_E_ECCENTRICITY)); ephemeris_record.ecc *= CNAV_E_ECCENTRICITY_LSB; ephemeris_record.omega = static_cast(read_navigation_signed(data_bits, CNAV_OMEGA)); ephemeris_record.omega *= CNAV_OMEGA_LSB; ephemeris_record.integrity_status_flag = read_navigation_bool(data_bits, CNAV_INTEGRITY_FLAG); ephemeris_record.l2c_phasing_flag = read_navigation_bool(data_bits, CNAV_L2_PHASING_FLAG); b_flag_ephemeris_1 = true; break; case 11: // Ephemeris 2/2 ephemeris_record.toe2 = static_cast(read_navigation_unsigned(data_bits, CNAV_TOE2)); ephemeris_record.toe2 *= CNAV_TOE2_LSB; ephemeris_record.OMEGA_0 = static_cast(read_navigation_signed(data_bits, CNAV_OMEGA0)); ephemeris_record.OMEGA_0 *= CNAV_OMEGA0_LSB; ephemeris_record.delta_OMEGAdot = static_cast(read_navigation_signed(data_bits, CNAV_DELTA_OMEGA_DOT)); ephemeris_record.delta_OMEGAdot *= CNAV_DELTA_OMEGA_DOT_LSB; ephemeris_record.OMEGAdot = CNAV_OMEGA_DOT_REF * GNSS_PI + ephemeris_record.delta_OMEGAdot; ephemeris_record.i_0 = static_cast(read_navigation_signed(data_bits, CNAV_I0)); ephemeris_record.i_0 *= CNAV_I0_LSB; ephemeris_record.idot = static_cast(read_navigation_signed(data_bits, CNAV_I0_DOT)); ephemeris_record.idot *= CNAV_I0_DOT_LSB; ephemeris_record.Cis = static_cast(read_navigation_signed(data_bits, CNAV_CIS)); ephemeris_record.Cis *= CNAV_CIS_LSB; ephemeris_record.Cic = static_cast(read_navigation_signed(data_bits, CNAV_CIC)); ephemeris_record.Cic *= CNAV_CIC_LSB; ephemeris_record.Crs = static_cast(read_navigation_signed(data_bits, CNAV_CRS)); ephemeris_record.Crs *= CNAV_CRS_LSB; ephemeris_record.Crc = static_cast(read_navigation_signed(data_bits, CNAV_CRC)); ephemeris_record.Crc *= CNAV_CRC_LSB; ephemeris_record.Cus = static_cast(read_navigation_signed(data_bits, CNAV_CUS)); ephemeris_record.Cus *= CNAV_CUS_LSB; ephemeris_record.Cuc = static_cast(read_navigation_signed(data_bits, CNAV_CUC)); ephemeris_record.Cuc *= CNAV_CUC_LSB; b_flag_ephemeris_2 = true; break; case 30: // (CLOCK, IONO, GRUP DELAY) // clock ephemeris_record.toc = static_cast(read_navigation_unsigned(data_bits, CNAV_TOC)); ephemeris_record.toc *= CNAV_TOC_LSB; ephemeris_record.URA0 = static_cast(read_navigation_signed(data_bits, CNAV_URA_NED0)); ephemeris_record.URA1 = static_cast(read_navigation_unsigned(data_bits, CNAV_URA_NED1)); ephemeris_record.URA2 = static_cast(read_navigation_unsigned(data_bits, CNAV_URA_NED2)); ephemeris_record.af0 = static_cast(read_navigation_signed(data_bits, CNAV_AF0)); ephemeris_record.af0 *= CNAV_AF0_LSB; ephemeris_record.af1 = static_cast(read_navigation_signed(data_bits, CNAV_AF1)); ephemeris_record.af1 *= CNAV_AF1_LSB; ephemeris_record.af2 = static_cast(read_navigation_signed(data_bits, CNAV_AF2)); ephemeris_record.af2 *= CNAV_AF2_LSB; // group delays // Check if the grup delay values are not available. See IS-GPS-200, Table 30-IV. // Bit string "1000000000000" is -4096 in 2 complement ephemeris_record.TGD = static_cast(read_navigation_signed(data_bits, CNAV_TGD)); if (ephemeris_record.TGD < -4095.9) { ephemeris_record.TGD = 0.0; } ephemeris_record.TGD *= CNAV_TGD_LSB; ephemeris_record.ISCL1 = static_cast(read_navigation_signed(data_bits, CNAV_ISCL1)); if (ephemeris_record.ISCL1 < -4095.9) { ephemeris_record.ISCL1 = 0.0; } ephemeris_record.ISCL1 *= CNAV_ISCL1_LSB; ephemeris_record.ISCL2 = static_cast(read_navigation_signed(data_bits, CNAV_ISCL2)); if (ephemeris_record.ISCL2 < -4095.9) { ephemeris_record.ISCL2 = 0.0; } ephemeris_record.ISCL2 *= CNAV_ISCL2_LSB; ephemeris_record.ISCL5I = static_cast(read_navigation_signed(data_bits, CNAV_ISCL5I)); if (ephemeris_record.ISCL5I < -4095.9) { ephemeris_record.ISCL5I = 0.0; } ephemeris_record.ISCL5I *= CNAV_ISCL5I_LSB; ephemeris_record.ISCL5Q = static_cast(read_navigation_signed(data_bits, CNAV_ISCL5Q)); if (ephemeris_record.ISCL5Q < -4095.9) { ephemeris_record.ISCL5Q = 0.0; } ephemeris_record.ISCL5Q *= CNAV_ISCL5Q_LSB; // iono iono_record.alpha0 = static_cast(read_navigation_signed(data_bits, CNAV_ALPHA0)); iono_record.alpha0 = iono_record.alpha0 * CNAV_ALPHA0_LSB; iono_record.alpha1 = static_cast(read_navigation_signed(data_bits, CNAV_ALPHA1)); iono_record.alpha1 = iono_record.alpha1 * CNAV_ALPHA1_LSB; iono_record.alpha2 = static_cast(read_navigation_signed(data_bits, CNAV_ALPHA2)); iono_record.alpha2 = iono_record.alpha2 * CNAV_ALPHA2_LSB; iono_record.alpha3 = static_cast(read_navigation_signed(data_bits, CNAV_ALPHA3)); iono_record.alpha3 = iono_record.alpha3 * CNAV_ALPHA3_LSB; iono_record.beta0 = static_cast(read_navigation_signed(data_bits, CNAV_BETA0)); iono_record.beta0 = iono_record.beta0 * CNAV_BETA0_LSB; iono_record.beta1 = static_cast(read_navigation_signed(data_bits, CNAV_BETA1)); iono_record.beta1 = iono_record.beta1 * CNAV_BETA1_LSB; iono_record.beta2 = static_cast(read_navigation_signed(data_bits, CNAV_BETA2)); iono_record.beta2 = iono_record.beta2 * CNAV_BETA2_LSB; iono_record.beta3 = static_cast(read_navigation_signed(data_bits, CNAV_BETA3)); iono_record.beta3 = iono_record.beta3 * CNAV_BETA3_LSB; b_flag_iono_valid = true; break; case 33: // (CLOCK & UTC) ephemeris_record.top = static_cast(read_navigation_unsigned(data_bits, CNAV_TOP1)); ephemeris_record.top = ephemeris_record.top * CNAV_TOP1_LSB; ephemeris_record.toc = static_cast(read_navigation_unsigned(data_bits, CNAV_TOC)); ephemeris_record.toc = ephemeris_record.toc * CNAV_TOC_LSB; ephemeris_record.af0 = static_cast(read_navigation_signed(data_bits, CNAV_AF0)); ephemeris_record.af0 = ephemeris_record.af0 * CNAV_AF0_LSB; ephemeris_record.af1 = static_cast(read_navigation_signed(data_bits, CNAV_AF1)); ephemeris_record.af1 = ephemeris_record.af1 * CNAV_AF1_LSB; ephemeris_record.af2 = static_cast(read_navigation_signed(data_bits, CNAV_AF2)); ephemeris_record.af2 = ephemeris_record.af2 * CNAV_AF2_LSB; utc_model_record.A0 = static_cast(read_navigation_signed(data_bits, CNAV_A0)); utc_model_record.A0 = utc_model_record.A0 * CNAV_A0_LSB; utc_model_record.A1 = static_cast(read_navigation_signed(data_bits, CNAV_A1)); utc_model_record.A1 = utc_model_record.A1 * CNAV_A1_LSB; utc_model_record.A2 = static_cast(read_navigation_signed(data_bits, CNAV_A2)); utc_model_record.A2 = utc_model_record.A2 * CNAV_A2_LSB; utc_model_record.DeltaT_LS = static_cast(read_navigation_signed(data_bits, CNAV_DELTA_TLS)); utc_model_record.DeltaT_LS = utc_model_record.DeltaT_LS * CNAV_DELTA_TLS_LSB; utc_model_record.tot = static_cast(read_navigation_signed(data_bits, CNAV_TOT)); utc_model_record.tot = utc_model_record.tot * CNAV_TOT_LSB; utc_model_record.WN_T = static_cast(read_navigation_signed(data_bits, CNAV_WN_OT)); utc_model_record.WN_T = utc_model_record.WN_T * CNAV_WN_OT_LSB; utc_model_record.WN_LSF = static_cast(read_navigation_signed(data_bits, CNAV_WN_LSF)); utc_model_record.WN_LSF = utc_model_record.WN_LSF * CNAV_WN_LSF_LSB; utc_model_record.DN = static_cast(read_navigation_signed(data_bits, CNAV_DN)); utc_model_record.DN = utc_model_record.DN * CNAV_DN_LSB; utc_model_record.DeltaT_LSF = static_cast(read_navigation_signed(data_bits, CNAV_DELTA_TLSF)); utc_model_record.DeltaT_LSF = utc_model_record.DeltaT_LSF * CNAV_DELTA_TLSF_LSB; b_flag_utc_valid = true; break; default: break; } } bool Gps_CNAV_Navigation_Message::have_new_ephemeris() // Check if we have a new ephemeris stored in the galileo navigation class { if (b_flag_ephemeris_1 == true and b_flag_ephemeris_2 == true) { if (ephemeris_record.toe1 == ephemeris_record.toe2) // and ephemeris_record.toe1==ephemeris_record.d_Toc) { // if all ephemeris pages have the same TOE, then they belong to the same block // std::cout << "Ephemeris (1, 2) have been received and belong to the same batch\n"; b_flag_ephemeris_1 = false; // clear the flag b_flag_ephemeris_2 = false; // clear the flag return true; } } return false; } Gps_CNAV_Ephemeris Gps_CNAV_Navigation_Message::get_ephemeris() const { return ephemeris_record; } bool Gps_CNAV_Navigation_Message::have_new_iono() // Check if we have a new iono data stored in the galileo navigation class { if (b_flag_iono_valid == true) { b_flag_iono_valid = false; // clear the flag return true; } return false; } Gps_CNAV_Iono Gps_CNAV_Navigation_Message::get_iono() const { return iono_record; } bool Gps_CNAV_Navigation_Message::have_new_utc_model() // Check if we have a new iono data stored in the galileo navigation class { if (b_flag_utc_valid == true) { b_flag_utc_valid = false; // clear the flag return true; } return false; } Gps_CNAV_Utc_Model Gps_CNAV_Navigation_Message::get_utc_model() { utc_model_record.valid = true; return utc_model_record; }