/*! * \file galileo_telemetry_decoder_gs.cc * \brief Implementation of a Galileo unified INAV and FNAV message demodulator * block * \author Javier Arribas 2018. jarribas(at)cttc.es * \author Carles Fernandez, 2021. cfernandez(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 "galileo_telemetry_decoder_gs.h" #include "Galileo_E1.h" // for GALILEO_E1_CODE_PERIOD_MS #include "Galileo_E5a.h" // for GALILEO_E5A_CODE_PERIOD_MS #include "Galileo_E5b.h" // for GALILEO_E5B_CODE_PERIOD_MS #include "Galileo_E6.h" // for GALILEO_E6_CODE_PERIOD_MS #include "display.h" // for colours in terminal: TEXT_BLUE, TEXT_RESET, ... #include "galileo_almanac_helper.h" // for Galileo_Almanac_Helper #include "galileo_ephemeris.h" // for Galileo_Ephemeris #include "galileo_has_page.h" // For Galileo_HAS_page #include "galileo_iono.h" // for Galileo_Iono #include "galileo_utc_model.h" // for Galileo_Utc_Model #include "gnss_sdr_make_unique.h" // for std::make_unique in C++11 #include "gnss_synchro.h" // for Gnss_Synchro #include "tlm_crc_stats.h" // for Tlm_CRC_Stats #include "tlm_utils.h" // for save_tlm_matfile, tlm_remove_file #include "viterbi_decoder.h" // for Viterbi_Decoder #include // for LOG, DLOG #include // for gr::io_signature::make #include // for pmt::make_any #include // for pmt::mp #include // for std::array #include // for std::fmod, std::abs #include // for size_t #include // for std::exception #include // for std::cout #define CRC_ERROR_LIMIT 6 galileo_telemetry_decoder_gs_sptr galileo_make_telemetry_decoder_gs(const Gnss_Satellite &satellite, const Tlm_Conf &conf, int frame_type) { return galileo_telemetry_decoder_gs_sptr(new galileo_telemetry_decoder_gs(satellite, conf, frame_type)); } galileo_telemetry_decoder_gs::galileo_telemetry_decoder_gs( const Gnss_Satellite &satellite, const Tlm_Conf &conf, int frame_type) : gr::block("galileo_telemetry_decoder_gs", gr::io_signature::make(1, 1, sizeof(Gnss_Synchro)), gr::io_signature::make(1, 1, sizeof(Gnss_Synchro))), d_dump_filename(conf.dump_filename), d_delta_t(0), d_sample_counter(0ULL), d_preamble_index(0ULL), d_last_valid_preamble(0), d_frame_type(frame_type), d_CRC_error_counter(0), d_channel(0), d_flag_even_word_arrived(0), d_stat(0), d_TOW_at_Preamble_ms(0), d_TOW_at_current_symbol_ms(0), d_band('1'), d_sent_tlm_failed_msg(false), d_flag_frame_sync(false), d_flag_PLL_180_deg_phase_locked(false), d_flag_preamble(false), d_dump(conf.dump), d_dump_mat(conf.dump_mat), d_remove_dat(conf.remove_dat), d_first_eph_sent(false), d_cnav_dummy_page(false), d_print_cnav_page(true), d_enable_navdata_monitor(conf.enable_navdata_monitor), d_dump_crc_stats(conf.dump_crc_stats), d_enable_reed_solomon_inav(false), d_valid_timetag(false) { // prevent telemetry symbols accumulation in output buffers this->set_max_noutput_items(1); // Ephemeris data port out this->message_port_register_out(pmt::mp("telemetry")); // Control messages to tracking block this->message_port_register_out(pmt::mp("telemetry_to_trk")); // register Gal E6 messages HAS out this->message_port_register_out(pmt::mp("E6_HAS_from_TLM")); if (d_enable_navdata_monitor) { // register nav message monitor out this->message_port_register_out(pmt::mp("Nav_msg_from_TLM")); } d_satellite = Gnss_Satellite(satellite.get_system(), satellite.get_PRN()); // Viterbi decoder vars const int32_t nn = 2; // Coding rate 1/n const int32_t KK = 7; // Constraint Length const std::array g_encoder{{121, 91}}; // Polynomial G1 and G2 d_mm = KK - 1; DLOG(INFO) << "Initializing GALILEO UNIFIED TELEMETRY DECODER"; if (d_dump_crc_stats) { // initialize the telemetry CRC statistics class d_Tlm_CRC_Stats = std::make_unique(); d_Tlm_CRC_Stats->initialize(conf.dump_crc_stats_filename); } else { d_Tlm_CRC_Stats = nullptr; } switch (d_frame_type) { case 1: // INAV { d_PRN_code_period_ms = GALILEO_E1_CODE_PERIOD_MS; // for Galileo E5b is also 4 ms d_bits_per_preamble = GALILEO_INAV_PREAMBLE_LENGTH_BITS; // set the preamble d_samples_per_preamble = GALILEO_INAV_PREAMBLE_LENGTH_BITS; d_preamble_period_symbols = GALILEO_INAV_PREAMBLE_PERIOD_SYMBOLS; d_required_symbols = GALILEO_INAV_PAGE_SYMBOLS + d_samples_per_preamble; // preamble bits to sampled symbols d_preamble_samples = std::vector(d_samples_per_preamble); d_frame_length_symbols = GALILEO_INAV_PAGE_PART_SYMBOLS - GALILEO_INAV_PREAMBLE_LENGTH_BITS; d_codelength = static_cast(d_frame_length_symbols); d_datalength = (d_codelength / nn) - d_mm; d_max_symbols_without_valid_frame = GALILEO_INAV_PAGE_SYMBOLS * 30; // rise alarm 60 seconds without valid tlm if (conf.enable_reed_solomon == true) { d_enable_reed_solomon_inav = true; d_inav_nav.enable_reed_solomon(); } break; } case 2: // FNAV { d_PRN_code_period_ms = static_cast(GALILEO_E5A_CODE_PERIOD_MS * GALILEO_E5A_I_SECONDARY_CODE_LENGTH); d_bits_per_preamble = GALILEO_FNAV_PREAMBLE_LENGTH_BITS; // set the preamble d_samples_per_preamble = GALILEO_FNAV_PREAMBLE_LENGTH_BITS; d_preamble_period_symbols = GALILEO_FNAV_SYMBOLS_PER_PAGE; d_required_symbols = static_cast(GALILEO_FNAV_SYMBOLS_PER_PAGE) + d_samples_per_preamble; // preamble bits to sampled symbols d_preamble_samples = std::vector(d_samples_per_preamble); d_frame_length_symbols = GALILEO_FNAV_SYMBOLS_PER_PAGE - GALILEO_FNAV_PREAMBLE_LENGTH_BITS; d_codelength = static_cast(d_frame_length_symbols); d_datalength = (d_codelength / nn) - d_mm; d_max_symbols_without_valid_frame = GALILEO_FNAV_SYMBOLS_PER_PAGE * 5; // rise alarm 100 seconds without valid tlm break; } case 3: // CNAV { d_PRN_code_period_ms = GALILEO_E6_CODE_PERIOD_MS; d_bits_per_preamble = GALILEO_CNAV_PREAMBLE_LENGTH_BITS; d_samples_per_preamble = GALILEO_CNAV_PREAMBLE_LENGTH_BITS; d_preamble_period_symbols = GALILEO_CNAV_SYMBOLS_PER_PAGE; d_required_symbols = static_cast(GALILEO_CNAV_SYMBOLS_PER_PAGE) + d_samples_per_preamble; d_preamble_samples = std::vector(d_samples_per_preamble); d_frame_length_symbols = GALILEO_CNAV_SYMBOLS_PER_PAGE - GALILEO_CNAV_PREAMBLE_LENGTH_BITS; d_codelength = static_cast(d_frame_length_symbols); d_datalength = (d_codelength / nn) - d_mm; d_max_symbols_without_valid_frame = GALILEO_CNAV_SYMBOLS_PER_PAGE * 60; break; } default: d_bits_per_preamble = 0; d_samples_per_preamble = 0; d_preamble_period_symbols = 0; d_PRN_code_period_ms = 0U; d_required_symbols = 0U; d_frame_length_symbols = 0U; d_codelength = 0; d_datalength = 0; d_max_symbols_without_valid_frame = 0; std::cout << "Galileo unified telemetry decoder error: Unknown frame type\n"; } d_page_part_symbols = std::vector(d_frame_length_symbols); for (int32_t i = 0; i < d_bits_per_preamble; i++) { switch (d_frame_type) { case 1: // INAV { if (GALILEO_INAV_PREAMBLE[i] == '1') { d_preamble_samples[i] = 1; } else { d_preamble_samples[i] = -1; } break; } case 2: // FNAV for E5a-I { if (GALILEO_FNAV_PREAMBLE[i] == '1') { d_preamble_samples[i] = 1; } else { d_preamble_samples[i] = -1; } break; } case 3: // CNAV for E6 { if (GALILEO_CNAV_PREAMBLE[i] == '1') { d_preamble_samples[i] = 1; } else { d_preamble_samples[i] = -1; } break; } } } d_symbol_history.set_capacity(d_required_symbols + 1); d_inav_nav.init_PRN(d_satellite.get_PRN()); // Instantiate the Viterbi decoder d_viterbi = std::make_unique(KK, nn, d_datalength, g_encoder); } galileo_telemetry_decoder_gs::~galileo_telemetry_decoder_gs() { DLOG(INFO) << "Galileo Telemetry decoder block (channel " << d_channel << ") destructor called."; size_t pos = 0; if (d_dump_file.is_open() == true) { pos = d_dump_file.tellp(); try { d_dump_file.close(); } catch (const std::exception &ex) { LOG(WARNING) << "Exception in destructor closing the dump file " << ex.what(); } if (pos == 0) { if (!tlm_remove_file(d_dump_filename)) { LOG(WARNING) << "Error deleting temporary file"; } } } if (d_dump && (pos != 0) && d_dump_mat) { save_tlm_matfile(d_dump_filename); if (d_remove_dat) { if (!tlm_remove_file(d_dump_filename)) { LOG(WARNING) << "Error deleting temporary file"; } } } } void galileo_telemetry_decoder_gs::deinterleaver(int32_t rows, int32_t cols, const float *in, float *out) { for (int32_t r = 0; r < rows; r++) { for (int32_t c = 0; c < cols; c++) { out[c * rows + r] = in[r * cols + c]; } } } void galileo_telemetry_decoder_gs::decode_INAV_word(float *page_part_symbols, int32_t frame_length) { // 1. De-interleave std::vector page_part_symbols_soft_value(frame_length); deinterleaver(GALILEO_INAV_INTERLEAVER_ROWS, GALILEO_INAV_INTERLEAVER_COLS, page_part_symbols, page_part_symbols_soft_value.data()); // 2. Viterbi decoder // 2.1 Take into account the NOT gate in G2 polynomial (Galileo ICD Figure 13, FEC encoder) for (int32_t i = 0; i < frame_length; i++) { if ((i + 1) % 2 == 0) { page_part_symbols_soft_value[i] = -page_part_symbols_soft_value[i]; } } const int32_t decoded_length = frame_length / 2; std::vector page_part_bits(decoded_length); d_viterbi->decode(page_part_bits, page_part_symbols_soft_value); // 3. Call the Galileo page decoder std::string page_String; page_String.reserve(decoded_length); for (int32_t i = 0; i < decoded_length; i++) { if (page_part_bits[i] > 0) { page_String.push_back('1'); } else { page_String.push_back('0'); } } if (d_enable_navdata_monitor) { d_nav_msg_packet.nav_message = page_String; } if (page_part_bits[0] == 1) { // DECODE COMPLETE WORD (even + odd) and TEST CRC d_inav_nav.split_page(page_String, d_flag_even_word_arrived); if (d_inav_nav.get_flag_CRC_test() == true) { if (d_band == '1') { DLOG(INFO) << "Galileo E1 CRC correct in channel " << d_channel << " from satellite " << d_satellite; } else if (d_band == '7') { DLOG(INFO) << "Galileo E5b CRC correct in channel " << d_channel << " from satellite " << d_satellite; } } else { if (d_band == '1') { DLOG(INFO) << "Galileo E1 CRC error in channel " << d_channel << " from satellite " << d_satellite; } else if (d_band == '7') { DLOG(INFO) << "Galileo E5b CRC error in channel " << d_channel << " from satellite " << d_satellite; } } d_flag_even_word_arrived = 0; } else { // STORE HALF WORD (even page) d_inav_nav.split_page(page_String, d_flag_even_word_arrived); d_flag_even_word_arrived = 1; } // 4. Push the new navigation data to the queues if (d_inav_nav.have_new_ephemeris() == true) { // get object for this SV (mandatory) const std::shared_ptr tmp_obj = std::make_shared(d_inav_nav.get_ephemeris()); if (d_band == '1') { std::cout << "New Galileo E1 I/NAV message received in channel " << d_channel << ": ephemeris from satellite " << d_satellite << '\n'; } else if (d_band == '7') { std::cout << TEXT_BLUE << "New Galileo E5b I/NAV message received in channel " << d_channel << ": ephemeris from satellite " << d_satellite << TEXT_RESET << '\n'; } this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); d_first_eph_sent = true; // do not send reduced CED anymore, since we have the full ephemeris set } else { // If we still do not have ephemeris, check if we have a reduced CED if ((d_band == '1') && !d_first_eph_sent && (d_inav_nav.have_new_reduced_ced() == true)) { const std::shared_ptr tmp_obj = std::make_shared(d_inav_nav.get_reduced_ced()); std::cout << "New Galileo E1 I/NAV reduced CED message received in channel " << d_channel << " from satellite " << d_satellite << '\n'; this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); } } if (d_inav_nav.have_new_iono_and_GST() == true) { // get object for this SV (mandatory) const std::shared_ptr tmp_obj = std::make_shared(d_inav_nav.get_iono()); if (d_band == '1') { std::cout << "New Galileo E1 I/NAV message received in channel " << d_channel << ": iono/GST model parameters from satellite " << d_satellite << '\n'; } else if (d_band == '7') { std::cout << TEXT_BLUE << "New Galileo E5b I/NAV message received in channel " << d_channel << ": iono/GST model parameters from satellite " << d_satellite << TEXT_RESET << '\n'; } this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); } if (d_inav_nav.have_new_utc_model() == true) { // get object for this SV (mandatory) const std::shared_ptr tmp_obj = std::make_shared(d_inav_nav.get_utc_model()); if (d_band == '1') { std::cout << "New Galileo E1 I/NAV message received in channel " << d_channel << ": UTC model parameters from satellite " << d_satellite << '\n'; } else if (d_band == '7') { std::cout << TEXT_BLUE << "New Galileo E5b I/NAV message received in channel " << d_channel << ": UTC model parameters from satellite " << d_satellite << TEXT_RESET << '\n'; } this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); d_delta_t = tmp_obj->A_0G + tmp_obj->A_1G * (static_cast(d_TOW_at_current_symbol_ms) / 1000.0 - tmp_obj->t_0G + 604800 * (std::fmod(static_cast(d_inav_nav.get_Galileo_week() - tmp_obj->WN_0G), 64.0))); DLOG(INFO) << "delta_t=" << d_delta_t << "[s]"; } if (d_inav_nav.have_new_almanac() == true) { const std::shared_ptr tmp_obj = std::make_shared(d_inav_nav.get_almanac()); this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); // debug if (d_band == '1') { std::cout << "Galileo E1 I/NAV almanac received in channel " << d_channel << " from satellite " << d_satellite << '\n'; } else if (d_band == '7') { std::cout << "Galileo E5b I/NAV almanac received in channel " << d_channel << " from satellite " << d_satellite << '\n'; } DLOG(INFO) << "Current parameters:"; DLOG(INFO) << "d_TOW_at_current_symbol_ms=" << d_TOW_at_current_symbol_ms; DLOG(INFO) << "d_nav.WN_0=" << d_inav_nav.get_Galileo_week(); } } void galileo_telemetry_decoder_gs::decode_FNAV_word(float *page_symbols, int32_t frame_length) { // 1. De-interleave std::vector page_symbols_soft_value(frame_length); deinterleaver(GALILEO_FNAV_INTERLEAVER_ROWS, GALILEO_FNAV_INTERLEAVER_COLS, page_symbols, page_symbols_soft_value.data()); // 2. Viterbi decoder // 2.1 Take into account the NOT gate in G2 polynomial (Galileo ICD Figure 13, FEC encoder) for (int32_t i = 0; i < frame_length; i++) { if ((i + 1) % 2 == 0) { page_symbols_soft_value[i] = -page_symbols_soft_value[i]; } } const int32_t decoded_length = frame_length / 2; std::vector page_bits(decoded_length); d_viterbi->decode(page_bits, page_symbols_soft_value); // 3. Call the Galileo page decoder std::string page_String; page_String.reserve(decoded_length); for (int32_t i = 0; i < decoded_length; i++) { if (page_bits[i] > 0) { page_String.push_back('1'); } else { page_String.push_back('0'); } } if (d_enable_navdata_monitor) { d_nav_msg_packet.nav_message = page_String; } // DECODE COMPLETE WORD (even + odd) and TEST CRC d_fnav_nav.split_page(page_String); if (d_fnav_nav.get_flag_CRC_test() == true) { DLOG(INFO) << "Galileo E5a CRC correct in channel " << d_channel << " from satellite " << d_satellite; } else { DLOG(INFO) << "Galileo E5a CRC error in channel " << d_channel << " from satellite " << d_satellite; } // 4. Push the new navigation data to the queues if (d_fnav_nav.have_new_ephemeris() == true) { const std::shared_ptr tmp_obj = std::make_shared(d_fnav_nav.get_ephemeris()); std::cout << TEXT_MAGENTA << "New Galileo E5a F/NAV message received in channel " << d_channel << ": ephemeris from satellite " << d_satellite << TEXT_RESET << '\n'; this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); } if (d_fnav_nav.have_new_iono_and_GST() == true) { const std::shared_ptr tmp_obj = std::make_shared(d_fnav_nav.get_iono()); std::cout << TEXT_MAGENTA << "New Galileo E5a F/NAV message received in channel " << d_channel << ": iono/GST model parameters from satellite " << d_satellite << TEXT_RESET << '\n'; this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); } if (d_fnav_nav.have_new_utc_model() == true) { const std::shared_ptr tmp_obj = std::make_shared(d_fnav_nav.get_utc_model()); std::cout << TEXT_MAGENTA << "New Galileo E5a F/NAV message received in channel " << d_channel << ": UTC model parameters from satellite " << d_satellite << TEXT_RESET << '\n'; this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); } } void galileo_telemetry_decoder_gs::decode_CNAV_word(float *page_symbols, int32_t page_length) { // 1. De-interleave std::vector page_symbols_soft_value(page_length); deinterleaver(GALILEO_CNAV_INTERLEAVER_ROWS, GALILEO_CNAV_INTERLEAVER_COLS, page_symbols, page_symbols_soft_value.data()); // 2. Viterbi decoder // 2.1 Take into account the NOT gate in G2 polynomial (Galileo ICD Figure 13, FEC encoder) for (int32_t i = 0; i < page_length; i++) { if ((i + 1) % 2 == 0) { page_symbols_soft_value[i] = -page_symbols_soft_value[i]; } } const int32_t decoded_length = page_length / 2; std::vector page_bits(decoded_length); d_viterbi->decode(page_bits, page_symbols_soft_value); // 3. Call the Galileo page decoder std::string page_String; page_String.reserve(decoded_length); for (int32_t i = 0; i < decoded_length; i++) { if (page_bits[i] > 0) { page_String.push_back('1'); } else { page_String.push_back('0'); } } d_cnav_nav.read_HAS_page(page_String); // 4. If we have a new HAS page, read it if (d_cnav_nav.have_new_HAS_page() == true) { bool is_page_dummy = d_cnav_nav.is_HAS_page_dummy(); if (is_page_dummy == true) { d_print_cnav_page = true; // Only print the message once if (is_page_dummy != d_cnav_dummy_page) { d_cnav_dummy_page = is_page_dummy; std::cout << TEXT_MAGENTA << "Receiving Galileo E6 CNAV dummy pages in channel " << d_channel << " from satellite " << d_satellite << TEXT_RESET << '\n'; } } else { const std::shared_ptr tmp_obj = std::make_shared(d_cnav_nav.get_HAS_encoded_page()); this->message_port_pub(pmt::mp("E6_HAS_from_TLM"), pmt::make_any(tmp_obj)); if (d_print_cnav_page == true) { d_print_cnav_page = false; // only print the first page std::cout << TEXT_MAGENTA << "Receiving Galileo E6 HAS pages" << (d_cnav_nav.is_HAS_in_test_mode() == true ? " (test mode) " : " ") << "in channel " << d_channel << " from satellite " << d_satellite << TEXT_RESET << '\n'; } } } } void galileo_telemetry_decoder_gs::set_satellite(const Gnss_Satellite &satellite) { gr::thread::scoped_lock lock(d_setlock); d_satellite = Gnss_Satellite(satellite.get_system(), satellite.get_PRN()); d_last_valid_preamble = d_sample_counter; d_sent_tlm_failed_msg = false; DLOG(INFO) << "Setting decoder Finite State Machine to satellite " << d_satellite; DLOG(INFO) << "Navigation Satellite set to " << d_satellite; } void galileo_telemetry_decoder_gs::reset() { gr::thread::scoped_lock lock(d_setlock); d_flag_frame_sync = false; d_TOW_at_current_symbol_ms = 0; d_TOW_at_Preamble_ms = 0; d_fnav_nav.set_flag_TOW_set(false); d_inav_nav.set_flag_TOW_set(false); d_inav_nav.set_TOW0_flag(false); d_last_valid_preamble = d_sample_counter; d_sent_tlm_failed_msg = false; d_stat = 0; d_viterbi->reset(); if (d_enable_reed_solomon_inav == true) { d_inav_nav.enable_reed_solomon(); } DLOG(INFO) << "Telemetry decoder reset for satellite " << d_satellite; } void galileo_telemetry_decoder_gs::set_channel(int32_t channel) { d_channel = channel; DLOG(INFO) << "Navigation channel set to " << channel; // ############# ENABLE DATA FILE LOG ################# if (d_dump == true) { if (d_dump_file.is_open() == false) { try { d_dump_filename.append(std::to_string(d_channel)); d_dump_filename.append(".dat"); d_dump_file.exceptions(std::ifstream::failbit | std::ifstream::badbit); d_dump_file.open(d_dump_filename.c_str(), std::ios::out | std::ios::binary); LOG(INFO) << "Telemetry decoder dump enabled on channel " << d_channel << " Log file: " << d_dump_filename.c_str(); } catch (const std::ifstream::failure &e) { LOG(WARNING) << "channel " << d_channel << " Exception opening trk dump file " << e.what(); } } } if (d_dump_crc_stats) { // set the channel number for the telemetry CRC statistics // disable the telemetry CRC statistics if there is a problem opening the output file d_dump_crc_stats = d_Tlm_CRC_Stats->set_channel(d_channel); } } int galileo_telemetry_decoder_gs::general_work(int noutput_items __attribute__((unused)), gr_vector_int &ninput_items __attribute__((unused)), gr_vector_const_void_star &input_items, gr_vector_void_star &output_items) { auto **out = reinterpret_cast(&output_items[0]); // Get the output buffer pointer const auto **in = reinterpret_cast(&input_items[0]); // Get the input buffer pointer Gnss_Synchro current_symbol{}; // structure to save the synchronization information and send the output object to the next block // 1. Copy the current tracking output current_symbol = in[0][0]; d_band = current_symbol.Signal[0]; // add new symbol to the symbol queue switch (d_frame_type) { case 1: // INAV { d_symbol_history.push_back(current_symbol.Prompt_I); break; } case 2: // FNAV { d_symbol_history.push_back(current_symbol.Prompt_Q); break; } case 3: // CNAV { d_symbol_history.push_back(current_symbol.Prompt_I); break; } default: { LOG(WARNING) << "Frame type " << d_frame_type << " is not defined"; d_symbol_history.push_back(current_symbol.Prompt_I); break; } } d_sample_counter++; // count for the processed symbols // Time Tags from signal source (optional feature) std::vector tags_vec; this->get_tags_in_range(tags_vec, 0, this->nitems_read(0), this->nitems_read(0) + 1); // telemetry decoder consumes symbols one-by-one if (!tags_vec.empty()) { for (const auto &it : tags_vec) { try { if (pmt::any_ref(it.value).type().hash_code() == typeid(const std::shared_ptr).hash_code()) { const auto timetag = boost::any_cast>(pmt::any_ref(it.value)); // std::cout << "Old tow: " << d_current_timetag.tow_ms << " new tow: " << timetag->tow_ms << "\n"; d_current_timetag = *timetag; d_valid_timetag = true; } else { std::cout << "hash code not match\n"; } } catch (const boost::bad_any_cast &e) { std::cout << "msg Bad any_cast: " << e.what(); } } } else { if (d_valid_timetag == true) { // propagate timetag to current symbol // todo: tag rx_time is set only in the time channel. The tracking tag does not have valid rx_time (it is not required since it is associated to the current symbol) // d_current_timetag.rx_time+=d_PRN_code_period_ms d_current_timetag.tow_ms += d_PRN_code_period_ms; if (d_current_timetag.tow_ms >= 604800000) { d_current_timetag.tow_ms -= 604800000; d_current_timetag.week++; } } } consume_each(1); d_flag_preamble = false; // check if there is a problem with the telemetry of the current satellite if (d_sent_tlm_failed_msg == false) { if ((d_sample_counter - d_last_valid_preamble) > d_max_symbols_without_valid_frame) { const int message = 1; // bad telemetry DLOG(INFO) << "sent msg sat " << this->d_satellite; this->message_port_pub(pmt::mp("telemetry_to_trk"), pmt::make_any(message)); d_sent_tlm_failed_msg = true; } } // ******* frame sync ****************** switch (d_stat) { case 0: // no preamble information { // correlate with preamble int32_t corr_value = 0; if (d_symbol_history.size() > d_required_symbols) { // ******* preamble correlation ******** for (int32_t i = 0; i < d_samples_per_preamble; i++) { if (d_symbol_history[i] < 0.0) // symbols clipping { corr_value -= d_preamble_samples[i]; } else { corr_value += d_preamble_samples[i]; } } if (std::abs(corr_value) >= d_samples_per_preamble) { d_preamble_index = d_sample_counter; // record the preamble sample stamp LOG(INFO) << "Preamble detection for Galileo satellite " << this->d_satellite << " in channel " << this->d_channel; d_stat = 1; // enter into frame pre-detection status } } break; } case 1: // possible preamble lock { // correlate with preamble int32_t corr_value = 0; if (d_symbol_history.size() > d_required_symbols) { // ******* preamble correlation ******** for (int32_t i = 0; i < d_samples_per_preamble; i++) { if (d_symbol_history[i] < 0.0) // symbols clipping { corr_value -= d_preamble_samples[i]; } else { corr_value += d_preamble_samples[i]; } } if (std::abs(corr_value) >= d_samples_per_preamble) { // check preamble separation const auto preamble_diff = static_cast(d_sample_counter - d_preamble_index); if (std::abs(preamble_diff - d_preamble_period_symbols) == 0) { // try to decode frame DLOG(INFO) << "Starting page decoder for Galileo satellite " << this->d_satellite; d_preamble_index = d_sample_counter; // record the preamble sample stamp d_CRC_error_counter = 0; if (corr_value < 0) { d_flag_PLL_180_deg_phase_locked = true; } else { d_flag_PLL_180_deg_phase_locked = false; } d_stat = 2; } else { if (preamble_diff > d_preamble_period_symbols) { d_stat = 0; // start again } } } } break; } case 2: // preamble acquired { if (d_sample_counter == d_preamble_index + static_cast(d_preamble_period_symbols)) { // call the decoder // NEW Galileo page part is received // 0. fetch the symbols into an array if (d_flag_PLL_180_deg_phase_locked == false) // normal PLL lock { for (uint32_t i = 0; i < d_frame_length_symbols; i++) { d_page_part_symbols[i] = d_symbol_history[i + d_samples_per_preamble]; // because last symbol of the preamble is just received now! } } else // 180 deg. inverted carrier phase PLL lock { for (uint32_t i = 0; i < d_frame_length_symbols; i++) { d_page_part_symbols[i] = -d_symbol_history[i + d_samples_per_preamble]; // because last symbol of the preamble is just received now! } } switch (d_frame_type) { case 1: // INAV decode_INAV_word(d_page_part_symbols.data(), d_frame_length_symbols); break; case 2: // FNAV decode_FNAV_word(d_page_part_symbols.data(), d_frame_length_symbols); break; case 3: // CNAV decode_CNAV_word(d_page_part_symbols.data(), d_frame_length_symbols); break; default: return -1; break; } bool crc_ok = (d_inav_nav.get_flag_CRC_test() || d_fnav_nav.get_flag_CRC_test() || d_cnav_nav.get_flag_CRC_test()); if (d_dump_crc_stats) { // update CRC statistics d_Tlm_CRC_Stats->update_CRC_stats(crc_ok); } d_preamble_index = d_sample_counter; // record the preamble sample stamp (t_P) if (crc_ok) { d_CRC_error_counter = 0; d_flag_preamble = true; // valid preamble indicator (initialized to false every work()) gr::thread::scoped_lock lock(d_setlock); d_last_valid_preamble = d_sample_counter; if (!d_flag_frame_sync) { d_flag_frame_sync = true; DLOG(INFO) << " Frame sync SAT " << this->d_satellite; } } else { d_CRC_error_counter++; if ((d_CRC_error_counter > CRC_ERROR_LIMIT) && (d_frame_type != 3)) { DLOG(INFO) << "Lost of frame sync SAT " << this->d_satellite; gr::thread::scoped_lock lock(d_setlock); d_flag_frame_sync = false; d_stat = 0; d_TOW_at_current_symbol_ms = 0; d_TOW_at_Preamble_ms = 0; d_fnav_nav.set_flag_TOW_set(false); d_inav_nav.set_flag_TOW_set(false); } } } break; } } // UPDATE GNSS SYNCHRO DATA // 2. Add the telemetry decoder information if (this->d_flag_preamble == true) // update TOW at the preamble instant { switch (d_frame_type) { case 1: // INAV { if (d_inav_nav.get_flag_TOW_set() == true) { if (d_inav_nav.is_TOW5_set() == true) // page 5 arrived and decoded, so we are in the odd page (since Tow refers to the even page, we have to add 1 sec) { // TOW_5 refers to the even preamble, but when we decode it we are in the odd part, so 1 second later plus the decoding delay d_TOW_at_Preamble_ms = static_cast(d_inav_nav.get_TOW5() * 1000.0); d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast(GALILEO_INAV_PAGE_PART_MS + (d_required_symbols + 1) * d_PRN_code_period_ms); d_inav_nav.set_TOW5_flag(false); // timetag debug if (d_valid_timetag == true) { int decoder_delay_ms = static_cast(GALILEO_INAV_PAGE_PART_MS + (d_required_symbols + 1) * d_PRN_code_period_ms); int rx_tow_at_preamble = d_current_timetag.tow_ms - decoder_delay_ms; if (rx_tow_at_preamble < 0) { rx_tow_at_preamble += 604800000; } uint32_t predicted_tow_at_preamble_ms = 1000 * (rx_tow_at_preamble / 1000); // floor to integer number of seconds std::cout << "TOW at PREAMBLE: " << d_TOW_at_Preamble_ms << " predicted TOW at preamble: " << predicted_tow_at_preamble_ms << " [ms]\n"; } } else if (d_inav_nav.is_TOW6_set() == true) // page 6 arrived and decoded, so we are in the odd page (since Tow refers to the even page, we have to add 1 sec) { // TOW_6 refers to the even preamble, but when we decode it we are in the odd part, so 1 second later plus the decoding delay d_TOW_at_Preamble_ms = static_cast(d_inav_nav.get_TOW6() * 1000.0); d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast(GALILEO_INAV_PAGE_PART_MS + (d_required_symbols + 1) * d_PRN_code_period_ms); d_inav_nav.set_TOW6_flag(false); // timetag debug if (d_valid_timetag == true) { int decoder_delay_ms = static_cast(GALILEO_INAV_PAGE_PART_MS + (d_required_symbols + 1) * d_PRN_code_period_ms); int rx_tow_at_preamble = d_current_timetag.tow_ms - decoder_delay_ms; if (rx_tow_at_preamble < 0) { rx_tow_at_preamble += 604800000; } uint32_t predicted_tow_at_preamble_ms = 1000 * (rx_tow_at_preamble / 1000); // floor to integer number of seconds std::cout << "TOW at PREAMBLE: " << d_TOW_at_Preamble_ms << " predicted TOW at preamble: " << predicted_tow_at_preamble_ms << " [ms]\n"; } } else if (d_inav_nav.is_TOW0_set() == true) // page 0 arrived and decoded { // TOW_0 refers to the even preamble, but when we decode it we are in the odd part, so 1 second later plus the decoding delay d_TOW_at_Preamble_ms = static_cast(d_inav_nav.get_TOW0() * 1000.0); d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast(GALILEO_INAV_PAGE_PART_MS + (d_required_symbols + 1) * d_PRN_code_period_ms); d_inav_nav.set_TOW0_flag(false); // timetag debug if (d_valid_timetag == true) { int decoder_delay_ms = static_cast(GALILEO_INAV_PAGE_PART_MS + (d_required_symbols + 1) * d_PRN_code_period_ms); int rx_tow_at_preamble = d_current_timetag.tow_ms - decoder_delay_ms; if (rx_tow_at_preamble < 0) { rx_tow_at_preamble += 604800000; } uint32_t predicted_tow_at_preamble_ms = 1000 * (rx_tow_at_preamble / 1000); // floor to integer number of seconds std::cout << "TOW at PREAMBLE: " << d_TOW_at_Preamble_ms << " predicted TOW at preamble: " << predicted_tow_at_preamble_ms << " [ms]\n"; } } else { // this page has no timing information d_TOW_at_current_symbol_ms += d_PRN_code_period_ms; } } if (d_enable_navdata_monitor && !d_nav_msg_packet.nav_message.empty()) { d_nav_msg_packet.system = std::string(1, current_symbol.System); d_nav_msg_packet.signal = std::string(current_symbol.Signal); d_nav_msg_packet.prn = static_cast(current_symbol.PRN); d_nav_msg_packet.tow_at_current_symbol_ms = static_cast(d_TOW_at_current_symbol_ms); const std::shared_ptr tmp_obj = std::make_shared(d_nav_msg_packet); this->message_port_pub(pmt::mp("Nav_msg_from_TLM"), pmt::make_any(tmp_obj)); d_nav_msg_packet.nav_message = ""; } break; } case 2: // FNAV { if (d_fnav_nav.get_flag_TOW_set() == true) { if (d_fnav_nav.is_TOW1_set() == true) { d_TOW_at_Preamble_ms = static_cast(d_fnav_nav.get_TOW1() * 1000.0); d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast((d_required_symbols + 1) * GALILEO_FNAV_CODES_PER_SYMBOL * GALILEO_E5A_CODE_PERIOD_MS); // d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast((GALILEO_FNAV_CODES_PER_PAGE + GALILEO_FNAV_CODES_PER_PREAMBLE) * GALILEO_E5a_CODE_PERIOD_MS); d_fnav_nav.set_TOW1_flag(false); } else if (d_fnav_nav.is_TOW2_set() == true) { d_TOW_at_Preamble_ms = static_cast(d_fnav_nav.get_TOW2() * 1000.0); // d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast((GALILEO_FNAV_CODES_PER_PAGE + GALILEO_FNAV_CODES_PER_PREAMBLE) * GALILEO_E5a_CODE_PERIOD_MS); d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast((d_required_symbols + 1) * GALILEO_FNAV_CODES_PER_SYMBOL * GALILEO_E5A_CODE_PERIOD_MS); d_fnav_nav.set_TOW2_flag(false); } else if (d_fnav_nav.is_TOW3_set() == true) { d_TOW_at_Preamble_ms = static_cast(d_fnav_nav.get_TOW3() * 1000.0); // d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast((GALILEO_FNAV_CODES_PER_PAGE + GALILEO_FNAV_CODES_PER_PREAMBLE) * GALILEO_E5a_CODE_PERIOD_MS); d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast((d_required_symbols + 1) * GALILEO_FNAV_CODES_PER_SYMBOL * GALILEO_E5A_CODE_PERIOD_MS); d_fnav_nav.set_TOW3_flag(false); } else if (d_fnav_nav.is_TOW4_set() == true) { d_TOW_at_Preamble_ms = static_cast(d_fnav_nav.get_TOW4() * 1000.0); // d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast((GALILEO_FNAV_CODES_PER_PAGE + GALILEO_FNAV_CODES_PER_PREAMBLE) * GALILEO_E5a_CODE_PERIOD_MS); d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast((d_required_symbols + 1) * GALILEO_FNAV_CODES_PER_SYMBOL * GALILEO_E5A_CODE_PERIOD_MS); d_fnav_nav.set_TOW4_flag(false); } else { d_TOW_at_current_symbol_ms += static_cast(GALILEO_FNAV_CODES_PER_SYMBOL * GALILEO_E5A_CODE_PERIOD_MS); } if (d_enable_navdata_monitor && !d_nav_msg_packet.nav_message.empty()) { d_nav_msg_packet.system = std::string(1, current_symbol.System); d_nav_msg_packet.signal = std::string(current_symbol.Signal); d_nav_msg_packet.prn = static_cast(current_symbol.PRN); d_nav_msg_packet.tow_at_current_symbol_ms = static_cast(d_TOW_at_current_symbol_ms); const std::shared_ptr tmp_obj = std::make_shared(d_nav_msg_packet); this->message_port_pub(pmt::mp("Nav_msg_from_TLM"), pmt::make_any(tmp_obj)); d_nav_msg_packet.nav_message = ""; } break; } } case 3: // CNAV { // TODO } } } else // if there is not a new preamble, we define the TOW of the current symbol { switch (d_frame_type) { case 1: // INAV { if (d_inav_nav.get_flag_TOW_set() == true) { d_TOW_at_current_symbol_ms += d_PRN_code_period_ms; } break; } case 2: // FNAV { if (d_fnav_nav.get_flag_TOW_set() == true) { d_TOW_at_current_symbol_ms += d_PRN_code_period_ms; } break; } case 3: // CNAV { // TODO d_TOW_at_current_symbol_ms += d_PRN_code_period_ms; // this is not the TOW! break; } } } switch (d_frame_type) { case 1: // INAV { if (d_inav_nav.get_flag_TOW_set() == true) { if (d_inav_nav.get_flag_GGTO() == true) // all GGTO parameters arrived { d_delta_t = d_inav_nav.get_A0G() + d_inav_nav.get_A1G() * (static_cast(d_TOW_at_current_symbol_ms) / 1000.0 - d_inav_nav.get_t0G() + 604800.0 * (std::fmod(static_cast(d_inav_nav.get_Galileo_week() - d_inav_nav.get_WN0G()), 64.0))); } current_symbol.Flag_valid_word = true; } break; } case 2: // FNAV { if (d_fnav_nav.get_flag_TOW_set() == true) { current_symbol.Flag_valid_word = true; } break; } case 3: // CNAV { // TODO break; } } if (d_inav_nav.get_flag_TOW_set() == true || d_fnav_nav.get_flag_TOW_set() == true || d_cnav_nav.get_flag_CRC_test() == true) { current_symbol.TOW_at_current_symbol_ms = d_TOW_at_current_symbol_ms; // todo: Galileo to GPS time conversion should be moved to observable block. // current_symbol.TOW_at_current_symbol_ms -= d_delta_t; // Galileo to GPS TOW if (d_flag_PLL_180_deg_phase_locked == true) { // correct the accumulated phase for the Costas loop phase shift, if required current_symbol.Carrier_phase_rads += GNSS_PI; current_symbol.Flag_PLL_180_deg_phase_locked = true; } else { current_symbol.Flag_PLL_180_deg_phase_locked = false; } if (d_dump == true) { // MULTIPLEXED FILE RECORDING - Record results to file try { double tmp_double; uint64_t tmp_ulong_int; int32_t tmp_int; tmp_double = static_cast(d_TOW_at_current_symbol_ms) / 1000.0; d_dump_file.write(reinterpret_cast(&tmp_double), sizeof(double)); tmp_ulong_int = current_symbol.Tracking_sample_counter; d_dump_file.write(reinterpret_cast(&tmp_ulong_int), sizeof(uint64_t)); tmp_double = static_cast(d_TOW_at_Preamble_ms) / 1000.0; d_dump_file.write(reinterpret_cast(&tmp_double), sizeof(double)); switch (d_frame_type) { case 1: tmp_int = (current_symbol.Prompt_I > 0.0 ? 1 : -1); break; case 2: tmp_int = (current_symbol.Prompt_Q > 0.0 ? 1 : -1); break; case 3: tmp_int = (current_symbol.Prompt_I > 0.0 ? 1 : -1); break; default: tmp_int = 0; break; } d_dump_file.write(reinterpret_cast(&tmp_int), sizeof(int32_t)); tmp_int = static_cast(current_symbol.PRN); d_dump_file.write(reinterpret_cast(&tmp_int), sizeof(int32_t)); } catch (const std::ifstream::failure &e) { LOG(WARNING) << "Exception writing navigation data dump file " << e.what(); } } // 3. Make the output (copy the object contents to the GNURadio reserved memory) *out[0] = current_symbol; return 1; } return 0; }