/*! * \file galileo_telemetry_decoder_cc.cc * \brief Implementation of a Galileo unified INAV and FNAV message demodulator block * \author Javier Arribas 2018. jarribas(at)cttc.es * * ------------------------------------------------------------------------- * * Copyright (C) 2010-2018 (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. * * GNSS-SDR is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * GNSS-SDR is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with GNSS-SDR. If not, see . * * ------------------------------------------------------------------------- */ #include "galileo_telemetry_decoder_cc.h" #include "control_message_factory.h" #include "convolutional.h" #include "display.h" #include "gnss_synchro.h" #include #include #include #include #include #define CRC_ERROR_LIMIT 6 using google::LogMessage; galileo_telemetry_decoder_cc_sptr galileo_make_telemetry_decoder_cc(const Gnss_Satellite &satellite, int frame_type, bool dump) { return galileo_telemetry_decoder_cc_sptr(new galileo_telemetry_decoder_cc(satellite, frame_type, dump)); } void galileo_telemetry_decoder_cc::viterbi_decoder(double *page_part_symbols, int32_t *page_part_bits) { Viterbi(page_part_bits, out0, state0, out1, state1, page_part_symbols, KK, nn, DataLength); } void galileo_telemetry_decoder_cc::deinterleaver(int32_t rows, int32_t cols, double *in, double *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]; } } } galileo_telemetry_decoder_cc::galileo_telemetry_decoder_cc( const Gnss_Satellite &satellite, int frame_type, bool dump) : gr::block("galileo_telemetry_decoder_cc", gr::io_signature::make(1, 1, sizeof(Gnss_Synchro)), gr::io_signature::make(1, 1, sizeof(Gnss_Synchro))) { // Ephemeris data port out this->message_port_register_out(pmt::mp("telemetry")); // initialize internal vars d_dump = dump; d_satellite = Gnss_Satellite(satellite.get_system(), satellite.get_PRN()); d_frame_type = frame_type; LOG(INFO) << "Initializing GALILEO UNIFIED TELEMETRY DECODER"; switch (d_frame_type) { case 1: // INAV { d_PRN_code_period_ms = static_cast(GALILEO_E1_CODE_PERIOD_MS); d_samples_per_symbol = Galileo_E1_B_SAMPLES_PER_SYMBOL; d_bits_per_preamble = GALILEO_INAV_PREAMBLE_LENGTH_BITS; // set the preamble d_samples_per_preamble = GALILEO_INAV_PREAMBLE_LENGTH_BITS * d_samples_per_symbol; d_preamble_period_symbols = GALILEO_INAV_PREAMBLE_PERIOD_SYMBOLS; d_required_symbols = static_cast(GALILEO_INAV_PAGE_SYMBOLS) + d_samples_per_preamble; // preamble bits to sampled symbols d_preamble_samples = static_cast(volk_gnsssdr_malloc(d_samples_per_preamble * sizeof(int32_t), volk_gnsssdr_get_alignment())); d_secondary_code_samples = nullptr; d_frame_length_symbols = GALILEO_INAV_PAGE_PART_SYMBOLS - GALILEO_INAV_PREAMBLE_LENGTH_BITS; CodeLength = GALILEO_INAV_PAGE_PART_SYMBOLS - GALILEO_INAV_PREAMBLE_LENGTH_BITS; DataLength = (CodeLength / nn) - mm; break; } case 2: // FNAV { d_PRN_code_period_ms = static_cast(GALILEO_E5a_CODE_PERIOD_MS); d_samples_per_symbol = GALILEO_FNAV_CODES_PER_SYMBOL; d_bits_per_preamble = GALILEO_FNAV_PREAMBLE_LENGTH_BITS; // set the preamble d_samples_per_preamble = GALILEO_FNAV_PREAMBLE_LENGTH_BITS * d_samples_per_symbol; d_preamble_period_symbols = GALILEO_FNAV_CODES_PER_PAGE; d_required_symbols = static_cast(GALILEO_FNAV_SYMBOLS_PER_PAGE) * d_samples_per_symbol + d_samples_per_preamble; // preamble bits to sampled symbols d_preamble_samples = static_cast(volk_gnsssdr_malloc(d_samples_per_preamble * sizeof(int32_t), volk_gnsssdr_get_alignment())); d_secondary_code_samples = static_cast(volk_gnsssdr_malloc(Galileo_E5a_I_SECONDARY_CODE_LENGTH * sizeof(int32_t), volk_gnsssdr_get_alignment())); d_frame_length_symbols = GALILEO_FNAV_SYMBOLS_PER_PAGE - GALILEO_FNAV_PREAMBLE_LENGTH_BITS; CodeLength = GALILEO_FNAV_SYMBOLS_PER_PAGE - GALILEO_FNAV_PREAMBLE_LENGTH_BITS; DataLength = (CodeLength / nn) - mm; for (int32_t i = 0; i < Galileo_E5a_I_SECONDARY_CODE_LENGTH; i++) { if (Galileo_E5a_I_SECONDARY_CODE.at(i) == '1') { d_secondary_code_samples[i] = 1; } else { d_secondary_code_samples[i] = -1; } } break; } default: d_bits_per_preamble = 0; d_samples_per_preamble = 0; d_preamble_period_symbols = 0; d_preamble_samples = nullptr; d_secondary_code_samples = nullptr; d_samples_per_symbol = 0U; d_PRN_code_period_ms = 0U; d_required_symbols = 0U; d_frame_length_symbols = 0.0; CodeLength = 0; DataLength = 0; std::cout << "Galileo unified telemetry decoder error: Unknown frame type " << std::endl; } d_page_part_symbols = static_cast(volk_gnsssdr_malloc(d_frame_length_symbols * sizeof(double), volk_gnsssdr_get_alignment())); int32_t n = 0; for (int32_t i = 0; i < d_bits_per_preamble; i++) { switch (d_frame_type) { case 1: // INAV { if (GALILEO_INAV_PREAMBLE.at(i) == '1') { for (uint32_t j = 0; j < d_samples_per_symbol; j++) { d_preamble_samples[n] = 1; n++; } } else { for (uint32_t j = 0; j < d_samples_per_symbol; j++) { d_preamble_samples[n] = -1; n++; } } break; } case 2: // FNAV for E5a-I { // Galileo E5a data channel (E5a-I) still has a secondary code int m = 0; if (GALILEO_FNAV_PREAMBLE.at(i) == '1') { for (uint32_t j = 0; j < d_samples_per_symbol; j++) { d_preamble_samples[n] = d_secondary_code_samples[m]; n++; m++; m = m % Galileo_E5a_I_SECONDARY_CODE_LENGTH; } } else { for (uint32_t j = 0; j < d_samples_per_symbol; j++) { d_preamble_samples[n] = -d_secondary_code_samples[m]; n++; m++; m = m % Galileo_E5a_I_SECONDARY_CODE_LENGTH; } } break; } } } d_sample_counter = 0ULL; d_stat = 0; d_preamble_index = 0ULL; d_flag_frame_sync = false; d_flag_parity = false; d_TOW_at_current_symbol_ms = 0; d_TOW_at_Preamble_ms = 0; delta_t = 0; d_CRC_error_counter = 0; flag_even_word_arrived = 0; d_flag_preamble = false; d_channel = 0; flag_TOW_set = false; // vars for Viterbi decoder int32_t max_states = 1 << mm; // 2^mm g_encoder[0] = 121; // Polynomial G1 g_encoder[1] = 91; // Polynomial G2 out0 = static_cast(volk_gnsssdr_malloc(max_states * sizeof(int32_t), volk_gnsssdr_get_alignment())); out1 = static_cast(volk_gnsssdr_malloc(max_states * sizeof(int32_t), volk_gnsssdr_get_alignment())); state0 = static_cast(volk_gnsssdr_malloc(max_states * sizeof(int32_t), volk_gnsssdr_get_alignment())); state1 = static_cast(volk_gnsssdr_malloc(max_states * sizeof(int32_t), volk_gnsssdr_get_alignment())); // create appropriate transition matrices nsc_transit(out0, state0, 0, g_encoder, KK, nn); nsc_transit(out1, state1, 1, g_encoder, KK, nn); } galileo_telemetry_decoder_cc::~galileo_telemetry_decoder_cc() { volk_gnsssdr_free(d_preamble_samples); if (d_frame_type == 2) { volk_gnsssdr_free(d_secondary_code_samples); } volk_gnsssdr_free(d_page_part_symbols); volk_gnsssdr_free(out0); volk_gnsssdr_free(out1); volk_gnsssdr_free(state0); volk_gnsssdr_free(state1); if (d_dump_file.is_open() == true) { try { d_dump_file.close(); } catch (const std::exception &ex) { LOG(WARNING) << "Exception in destructor closing the dump file " << ex.what(); } } } void galileo_telemetry_decoder_cc::decode_INAV_word(double *page_part_symbols, int32_t frame_length) { // 1. De-interleave auto *page_part_symbols_deint = static_cast(volk_gnsssdr_malloc(frame_length * sizeof(double), volk_gnsssdr_get_alignment())); deinterleaver(GALILEO_INAV_INTERLEAVER_ROWS, GALILEO_INAV_INTERLEAVER_COLS, page_part_symbols, page_part_symbols_deint); // 2. Viterbi decoder // 2.1 Take into account the NOT gate in G2 polynomial (Galileo ICD Figure 13, FEC encoder) // 2.2 Take into account the possible inversion of the polarity due to PLL lock at 180º for (int32_t i = 0; i < frame_length; i++) { if ((i + 1) % 2 == 0) { page_part_symbols_deint[i] = -page_part_symbols_deint[i]; } } auto *page_part_bits = static_cast(volk_gnsssdr_malloc((frame_length / 2) * sizeof(int32_t), volk_gnsssdr_get_alignment())); viterbi_decoder(page_part_symbols_deint, page_part_bits); volk_gnsssdr_free(page_part_symbols_deint); // 3. Call the Galileo page decoder std::string page_String; for (int32_t i = 0; i < (frame_length / 2); i++) { if (page_part_bits[i] > 0) { page_String.push_back('1'); } else { page_String.push_back('0'); } } if (page_part_bits[0] == 1) { // DECODE COMPLETE WORD (even + odd) and TEST CRC d_inav_nav.split_page(page_String, flag_even_word_arrived); if (d_inav_nav.flag_CRC_test == true) { LOG(INFO) << "Galileo E1 CRC correct in channel " << d_channel << " from satellite " << d_satellite; } else { LOG(INFO) << "Galileo E1 CRC error in channel " << d_channel << " from satellite " << d_satellite; } flag_even_word_arrived = 0; } else { // STORE HALF WORD (even page) d_inav_nav.split_page(page_String.c_str(), flag_even_word_arrived); flag_even_word_arrived = 1; } volk_gnsssdr_free(page_part_bits); // 4. Push the new navigation data to the queues if (d_inav_nav.have_new_ephemeris() == true) { // get object for this SV (mandatory) std::shared_ptr tmp_obj = std::make_shared(d_inav_nav.get_ephemeris()); std::cout << "New Galileo E1 I/NAV message received in channel " << d_channel << ": ephemeris from satellite " << d_satellite << std::endl; 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) std::shared_ptr tmp_obj = std::make_shared(d_inav_nav.get_iono()); std::cout << "New Galileo E1 I/NAV message received in channel " << d_channel << ": iono/GST model parameters from satellite " << d_satellite << std::endl; 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) std::shared_ptr tmp_obj = std::make_shared(d_inav_nav.get_utc_model()); std::cout << "New Galileo E1 I/NAV message received in channel " << d_channel << ": UTC model parameters from satellite " << d_satellite << std::endl; this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); delta_t = tmp_obj->A_0G_10 + tmp_obj->A_1G_10 * (static_cast(d_TOW_at_current_symbol_ms) / 1000.0 - tmp_obj->t_0G_10 + 604800 * (fmod((d_inav_nav.WN_0 - tmp_obj->WN_0G_10), 64))); DLOG(INFO) << "delta_t=" << delta_t << "[s]"; } if (d_inav_nav.have_new_almanac() == true) { 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 std::cout << "Galileo E1 I/NAV almanac received in channel " << d_channel << " from satellite " << d_satellite << std::endl; 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.WN_0; } } void galileo_telemetry_decoder_cc::decode_FNAV_word(double *page_symbols, int32_t frame_length) { // 1. De-interleave auto *page_symbols_deint = static_cast(volk_gnsssdr_malloc(frame_length * sizeof(double), volk_gnsssdr_get_alignment())); deinterleaver(GALILEO_FNAV_INTERLEAVER_ROWS, GALILEO_FNAV_INTERLEAVER_COLS, page_symbols, page_symbols_deint); // 2. Viterbi decoder // 2.1 Take into account the NOT gate in G2 polynomial (Galileo ICD Figure 13, FEC encoder) // 2.2 Take into account the possible inversion of the polarity due to PLL lock at 180� for (int32_t i = 0; i < frame_length; i++) { if ((i + 1) % 2 == 0) { page_symbols_deint[i] = -page_symbols_deint[i]; } } auto *page_bits = static_cast(volk_gnsssdr_malloc((frame_length / 2) * sizeof(int32_t), volk_gnsssdr_get_alignment())); viterbi_decoder(page_symbols_deint, page_bits); volk_gnsssdr_free(page_symbols_deint); // 3. Call the Galileo page decoder std::string page_String; for (int32_t i = 0; i < frame_length; i++) { if (page_bits[i] > 0) { page_String.push_back('1'); } else { page_String.push_back('0'); } } volk_gnsssdr_free(page_bits); // DECODE COMPLETE WORD (even + odd) and TEST CRC d_fnav_nav.split_page(page_String); if (d_fnav_nav.flag_CRC_test == true) { LOG(INFO) << "Galileo E5a CRC correct in channel " << d_channel << " from satellite " << d_satellite; } else { LOG(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) { 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 << std::endl; this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); } if (d_fnav_nav.have_new_iono_and_GST() == true) { 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 << std::endl; this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); } if (d_fnav_nav.have_new_utc_model() == true) { 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 << std::endl; this->message_port_pub(pmt::mp("telemetry"), pmt::make_any(tmp_obj)); } } void galileo_telemetry_decoder_cc::set_satellite(const Gnss_Satellite &satellite) { d_satellite = Gnss_Satellite(satellite.get_system(), satellite.get_PRN()); DLOG(INFO) << "Setting decoder Finite State Machine to satellite " << d_satellite; DLOG(INFO) << "Navigation Satellite set to " << d_satellite; } void galileo_telemetry_decoder_cc::set_channel(int32_t channel) { d_channel = channel; LOG(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 = "telemetry"; 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(); } } } } int galileo_telemetry_decoder_cc::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) { int32_t corr_value = 0; int32_t preamble_diff = 0; 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]; // add new symbol to the symbol queue d_symbol_history.push_back(current_symbol.Prompt_I); d_sample_counter++; // count for the processed samples consume_each(1); d_flag_preamble = false; if (d_symbol_history.size() > d_required_symbols) { // TODO Optimize me! // ******* preamble correlation ******** for (int32_t i = 0; i < d_samples_per_preamble; i++) { if (d_symbol_history.at(i) < 0.0) // symbols clipping { corr_value -= d_preamble_samples[i]; } else { corr_value += d_preamble_samples[i]; } } } // ******* frame sync ****************** switch (d_stat) { case 0: // no preamble information { if (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; d_stat = 1; // enter into frame pre-detection status } break; } case 1: // possible preamble lock { if (abs(corr_value) >= d_samples_per_preamble) { // check preamble separation preamble_diff = static_cast(d_sample_counter - d_preamble_index); if (abs(preamble_diff - d_preamble_period_symbols) == 0) { // try to decode frame LOG(INFO) << "Starting page decoder for Galileo satellite " << this->d_satellite; d_preamble_index = d_sample_counter; // record the preamble sample stamp 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 switch (d_frame_type) { case 1: // INAV // NEW Galileo page part is received // 0. fetch the symbols into an array if (corr_value > 0) //normal PLL lock { for (uint32_t i = 0; i < d_frame_length_symbols; i++) { d_page_part_symbols[i] = d_symbol_history.at(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.at(i + d_samples_per_preamble); // because last symbol of the preamble is just received now! } } decode_INAV_word(d_page_part_symbols, d_frame_length_symbols); break; case 2: // FNAV // NEW Galileo page part is received // 0. fetch the symbols into an array if (corr_value > 0) //normal PLL lock { int k = 0; for (uint32_t i = 0; i < d_frame_length_symbols; i++) { d_page_part_symbols[i] = 0; for (uint32_t m = 0; m < d_samples_per_symbol; m++) { d_page_part_symbols[i] += static_cast(d_secondary_code_samples[k]) * d_symbol_history.at(i * d_samples_per_symbol + d_samples_per_preamble + m); // because last symbol of the preamble is just received now! k++; k = k % Galileo_E5a_I_SECONDARY_CODE_LENGTH; } } } else //180 deg. inverted carrier phase PLL lock { int k = 0; for (uint32_t i = 0; i < d_frame_length_symbols; i++) { d_page_part_symbols[i] = 0; for (uint32_t m = 0; m < d_samples_per_symbol; m++) //integrate samples into symbols { d_page_part_symbols[i] -= static_cast(d_secondary_code_samples[k]) * d_symbol_history.at(i * d_samples_per_symbol + d_samples_per_preamble + m); // because last symbol of the preamble is just received now! k++; k = k % Galileo_E5a_I_SECONDARY_CODE_LENGTH; } } } decode_FNAV_word(d_page_part_symbols, d_frame_length_symbols); break; default: return -1; break; } if (d_inav_nav.flag_CRC_test == true or d_fnav_nav.flag_CRC_test == true) { d_CRC_error_counter = 0; d_flag_preamble = true; // valid preamble indicator (initialized to false every work()) d_preamble_index = d_sample_counter; // record the preamble sample stamp (t_P) if (!d_flag_frame_sync) { d_flag_frame_sync = true; DLOG(INFO) << " Frame sync SAT " << this->d_satellite; } } else { d_CRC_error_counter++; d_preamble_index = d_sample_counter; // record the preamble sample stamp if (d_CRC_error_counter > CRC_ERROR_LIMIT) { LOG(INFO) << "Lost of frame sync SAT " << this->d_satellite; d_flag_frame_sync = false; d_stat = 0; d_TOW_at_current_symbol_ms = 0; d_TOW_at_Preamble_ms = 0; d_fnav_nav.flag_TOW_set = false; d_inav_nav.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.flag_TOW_set == true) { if (d_inav_nav.flag_TOW_5 == 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.TOW_5 * 1000.0); d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast(GALILEO_INAV_PAGE_PART_MS + (d_required_symbols + 1) * GALILEO_E1_CODE_PERIOD_MS); d_inav_nav.flag_TOW_5 = false; } else if (d_inav_nav.flag_TOW_6 == 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.TOW_6 * 1000.0); d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast(GALILEO_INAV_PAGE_PART_MS + (d_required_symbols + 1) * GALILEO_E1_CODE_PERIOD_MS); d_inav_nav.flag_TOW_6 = false; } else { // this page has no timing information d_TOW_at_current_symbol_ms += static_cast(GALILEO_E1_CODE_PERIOD_MS); // + GALILEO_INAV_PAGE_PART_SYMBOLS*GALILEO_E1_CODE_PERIOD; } } break; } case 2: // FNAV { if (d_fnav_nav.flag_TOW_set == true) { if (d_fnav_nav.flag_TOW_1 == true) { d_TOW_at_Preamble_ms = static_cast(d_fnav_nav.FNAV_TOW_1 * 1000.0); d_TOW_at_current_symbol_ms = d_TOW_at_Preamble_ms + static_cast((d_required_symbols + 1) * 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.flag_TOW_1 = false; } else if (d_fnav_nav.flag_TOW_2 == true) { d_TOW_at_Preamble_ms = static_cast(d_fnav_nav.FNAV_TOW_2 * 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_E5a_CODE_PERIOD_MS); d_fnav_nav.flag_TOW_2 = false; } else if (d_fnav_nav.flag_TOW_3 == true) { d_TOW_at_Preamble_ms = static_cast(d_fnav_nav.FNAV_TOW_3 * 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_E5a_CODE_PERIOD_MS); d_fnav_nav.flag_TOW_3 = false; } else if (d_fnav_nav.flag_TOW_4 == true) { d_TOW_at_Preamble_ms = static_cast(d_fnav_nav.FNAV_TOW_4 * 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_E5a_CODE_PERIOD_MS); d_fnav_nav.flag_TOW_4 = false; } else { d_TOW_at_current_symbol_ms += static_cast(GALILEO_E5a_CODE_PERIOD_MS); } break; } } } } 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.flag_TOW_set == true) { d_TOW_at_current_symbol_ms += d_PRN_code_period_ms; } break; } case 2: // FNAV { if (d_fnav_nav.flag_TOW_set == true) { d_TOW_at_current_symbol_ms += d_PRN_code_period_ms; } break; } } } // remove used symbols from history // todo: Use circular buffer here if (d_symbol_history.size() > d_required_symbols) { d_symbol_history.pop_front(); } switch (d_frame_type) { case 1: // INAV { if (d_inav_nav.flag_TOW_set) { if (d_inav_nav.flag_GGTO_1 == true and d_inav_nav.flag_GGTO_2 == true and d_inav_nav.flag_GGTO_3 == true and d_inav_nav.flag_GGTO_4 == true) // all GGTO parameters arrived { delta_t = d_inav_nav.A_0G_10 + d_inav_nav.A_1G_10 * (static_cast(d_TOW_at_current_symbol_ms) / 1000.0 - d_inav_nav.t_0G_10 + 604800.0 * (fmod((d_inav_nav.WN_0 - d_inav_nav.WN_0G_10), 64.0))); } current_symbol.Flag_valid_word = true; } break; } case 2: // FNAV { if (d_fnav_nav.flag_TOW_set) { current_symbol.Flag_valid_word = true; } break; } } if (d_inav_nav.flag_TOW_set or d_fnav_nav.flag_TOW_set) { 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 -= delta_t; //Galileo to GPS TOW if (d_dump == true) { // MULTIPLEXED FILE RECORDING - Record results to file try { double tmp_double; uint64_t tmp_ulong_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)); } catch (const std::ifstream::failure &e) { LOG(WARNING) << "Exception writing observables 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; }