summaryrefslogtreecommitdiff
path: root/gr-dtv/lib/dvbt/dvbt_reference_signals_impl.cc
diff options
context:
space:
mode:
Diffstat (limited to 'gr-dtv/lib/dvbt/dvbt_reference_signals_impl.cc')
-rw-r--r--gr-dtv/lib/dvbt/dvbt_reference_signals_impl.cc1280
1 files changed, 1280 insertions, 0 deletions
diff --git a/gr-dtv/lib/dvbt/dvbt_reference_signals_impl.cc b/gr-dtv/lib/dvbt/dvbt_reference_signals_impl.cc
new file mode 100644
index 0000000000..d64b403915
--- /dev/null
+++ b/gr-dtv/lib/dvbt/dvbt_reference_signals_impl.cc
@@ -0,0 +1,1280 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2015 Free Software Foundation, Inc.
+ *
+ * This 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, or (at your option)
+ * any later version.
+ *
+ * This software 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 this software; see the file COPYING. If not, write to
+ * the Free Software Foundation, Inc., 51 Franklin Street,
+ * Boston, MA 02110-1301, USA.
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include <gnuradio/io_signature.h>
+#include "dvbt_reference_signals_impl.h"
+#include <complex>
+#include <stdio.h>
+#include <gnuradio/expj.h>
+#include <gnuradio/math.h>
+
+namespace gr {
+ namespace dtv {
+
+ //Number of symbols in a frame
+ const int dvbt_pilot_gen::d_symbols_per_frame = SYMBOLS_PER_FRAME;
+ //Number of frames in a superframe
+ const int dvbt_pilot_gen::d_frames_per_superframe = FRAMES_PER_SUPERFRAME;
+
+ // 2k mode
+ // Scattered pilots # of carriers
+ const int dvbt_pilot_gen::d_spilot_carriers_size_2k = SCATTERED_PILOT_SIZE_2k;
+ // Continual pilots # of carriers and positions
+ const int dvbt_pilot_gen::d_cpilot_carriers_size_2k = CONTINUAL_PILOT_SIZE_2k;
+ const int dvbt_pilot_gen::d_cpilot_carriers_2k[dvbt_pilot_gen::d_cpilot_carriers_size_2k] = {
+ 0, 48, 54, 87, 141, 156, 192, \
+ 201, 255, 279, 282, 333, 432, 450, \
+ 483, 525, 531, 618, 636, 714, 759, \
+ 765, 780, 804, 873, 888, 918, 939, \
+ 942, 969, 984, 1050, 1101, 1107, 1110, \
+ 1137, 1140, 1146, 1206, 1269, 1323, 1377, \
+ 1491, 1683, 1704
+ };
+ // TPS pilots # of carriers and positions
+ const int dvbt_pilot_gen::d_tps_carriers_size_2k = TPS_PILOT_SIZE_2k;
+ const int dvbt_pilot_gen::d_tps_carriers_2k[dvbt_pilot_gen::d_tps_carriers_size_2k] = {
+ 34, 50, 209, 346, 413, \
+ 569, 595, 688, 790, 901, \
+ 1073, 1219, 1262, 1286, 1469, \
+ 1594, 1687
+ };
+
+ // 8k mode
+ // Scattered pilots # of carriers
+ const int dvbt_pilot_gen::d_spilot_carriers_size_8k = SCATTERED_PILOT_SIZE_8k;
+ // Continual pilots # of carriers and positions
+ const int dvbt_pilot_gen::d_cpilot_carriers_size_8k = CONTINUAL_PILOT_SIZE_8k;
+ const int dvbt_pilot_gen::d_cpilot_carriers_8k[dvbt_pilot_gen::d_cpilot_carriers_size_8k] = {
+ 0, 48, 54, 87, 141, 156, 192,
+ 201, 255, 279, 282, 333, 432, 450,
+ 483, 525, 531, 618, 636, 714, 759,
+ 765, 780, 804, 873, 888, 918, 939,
+ 942, 969, 984, 1050, 1101, 1107, 1110,
+ 1137, 1140, 1146, 1206, 1269, 1323, 1377,
+ 1491, 1683, 1704, 1752, 1758, 1791, 1845,
+ 1860, 1896, 1905, 1959, 1983, 1986, 2037,
+ 2136, 2154, 2187, 2229, 2235, 2322, 2340,
+ 2418, 2463, 2469, 2484, 2508, 2577, 2592,
+ 2622, 2643, 2646, 2673, 2688, 2754, 2805,
+ 2811, 2814, 2841, 2844, 2850, 2910, 2973,
+ 3027, 3081, 3195, 3387, 3408, 3456, 3462,
+ 3495, 3549, 3564, 3600, 3609, 3663, 3687,
+ 3690, 3741, 3840, 3858, 3891, 3933, 3939,
+ 4026, 4044, 4122, 4167, 4173, 4188, 4212,
+ 4281, 4296, 4326, 4347, 4350, 4377, 4392,
+ 4458, 4509, 4515, 4518, 4545, 4548, 4554,
+ 4614, 4677, 4731, 4785, 4899, 5091, 5112,
+ 5160, 5166, 5199, 5253, 5268, 5304, 5313,
+ 5367, 5391, 5394, 5445, 5544, 5562, 5595,
+ 5637, 5643, 5730, 5748, 5826, 5871, 5877,
+ 5892, 5916, 5985, 6000, 6030, 6051, 6054,
+ 6081, 6096, 6162, 6213, 6219, 6222, 6249,
+ 6252, 6258, 6318, 6381, 6435, 6489, 6603,
+ 6795, 6816
+ };
+ // TPS pilots # of carriers and positions
+ const int dvbt_pilot_gen::d_tps_carriers_size_8k = TPS_PILOT_SIZE_8k;
+ const int dvbt_pilot_gen::d_tps_carriers_8k[dvbt_pilot_gen::d_tps_carriers_size_8k] = {
+ 34, 50, 209, 346, 413, 569, 595, 688, \
+ 790, 901, 1073, 1219, 1262, 1286, 1469, 1594, \
+ 1687, 1738, 1754, 1913, 2050, 2117, 2273, 2299, \
+ 2392, 2494, 2605, 2777, 2923, 2966, 2990, 3173, \
+ 3298, 3391, 3442, 3458, 3617, 3754, 3821, 3977, \
+ 4003, 4096, 4198, 4309, 4481, 4627, 4670, 4694, \
+ 4877, 5002, 5095, 5146, 5162, 5321, 5458, 5525, \
+ 5681, 5707, 5800, 5902, 6013, 6185, 6331, 6374, \
+ 6398, 6581, 6706, 6799
+ };
+
+ // TPS sync sequence for odd and even frames
+ const int dvbt_pilot_gen::d_tps_sync_size = 16; // TODO
+ const int dvbt_pilot_gen::d_tps_sync_even[d_tps_sync_size] = {
+ 0, 0, 1, 1, 0, 1, 0, 1, 1, 1, 1, 0, 1, 1, 1, 0
+ };
+ const int dvbt_pilot_gen::d_tps_sync_odd[d_tps_sync_size] = {
+ 1, 1, 0, 0, 1, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 1
+ };
+
+ /*
+ * Constructor of class
+ */
+ dvbt_pilot_gen::dvbt_pilot_gen(const dvbt_configure &c) : config(c),
+ d_spilot_index(0),
+ d_cpilot_index(0),
+ d_tpilot_index(0),
+ d_symbol_index(0),
+ d_symbol_index_known(0),
+ d_frame_index(0),
+ d_superframe_index(0),
+ d_freq_offset_max(8),
+ d_trigger_index(0),
+ d_payload_index(0),
+ d_chanestim_index(0),
+ d_prev_mod_symbol_index(0),
+ d_mod_symbol_index(0)
+ {
+ //Determine parameters from config file
+ d_Kmin = config.d_Kmin;
+ d_Kmax = config.d_Kmax;
+ d_fft_length = config.d_fft_length;
+ d_payload_length = config.d_payload_length;
+ d_zeros_on_left = config.d_zeros_on_left;
+ d_zeros_on_right = config.d_zeros_on_right;
+ d_cp_length = config.d_cp_length;
+
+ //Set-up pilot data depending on transmission mode
+ if (config.d_transmission_mode == T2k) {
+ d_spilot_carriers_size = d_spilot_carriers_size_2k;
+ d_cpilot_carriers_size = d_cpilot_carriers_size_2k;
+ d_cpilot_carriers = d_cpilot_carriers_2k;
+ d_tps_carriers_size = d_tps_carriers_size_2k;
+ d_tps_carriers = d_tps_carriers_2k;
+ }
+ else if (config.d_transmission_mode == T8k) {
+ d_spilot_carriers_size = d_spilot_carriers_size_8k;
+ d_cpilot_carriers_size = d_cpilot_carriers_size_8k;
+ d_cpilot_carriers = d_cpilot_carriers_8k;
+ d_tps_carriers_size = d_tps_carriers_size_8k;
+ d_tps_carriers = d_tps_carriers_8k;
+ }
+ else {
+ d_spilot_carriers_size = d_spilot_carriers_size_2k;
+ d_cpilot_carriers_size = d_cpilot_carriers_size_2k;
+ d_cpilot_carriers = d_cpilot_carriers_2k;
+ d_tps_carriers_size = d_tps_carriers_size_2k;
+ d_tps_carriers = d_tps_carriers_2k;
+ }
+
+ //allocate PRBS buffer
+ d_wk = new char[d_Kmax - d_Kmin + 1];
+ if (d_wk == NULL) {
+ std::cout << "Cannot allocate memory for d_wk" << std::endl;
+ exit(1);
+ }
+ // Generate wk sequence
+ generate_prbs();
+
+ // allocate buffer for scattered pilots
+ d_spilot_carriers_val = new gr_complex[d_Kmax - d_Kmin + 1];
+ if (d_spilot_carriers_val == NULL) {
+ std::cout << "Cannot allocate memory for d_tps_carriers_val" << std::endl;
+ delete [] d_wk;
+ exit(1);
+ }
+
+ // allocate buffer for channel gains (for each useful carrier)
+ d_channel_gain = new gr_complex[d_Kmax - d_Kmin + 1];
+ if (d_channel_gain == NULL) {
+ std::cout << "Cannot allocate memory for d_tps_carriers_val" << std::endl;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ exit(1);
+ }
+
+ // Allocate buffer for continual pilots phase diffs
+ d_known_phase_diff = new float[d_cpilot_carriers_size - 1];
+ if (d_known_phase_diff == NULL) {
+ std::cout << "Cannot allocate memory for d_tps_carriers_val" << std::endl;
+ delete [] d_channel_gain;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ exit(1);
+ }
+
+ // Obtain phase diff for all continual pilots
+ for (int i = 0; i < (d_cpilot_carriers_size - 1); i++) {
+ d_known_phase_diff[i] = \
+ norm(get_cpilot_value(d_cpilot_carriers[i + 1]) - get_cpilot_value(d_cpilot_carriers[i]));
+ }
+
+ d_cpilot_phase_diff = new float[d_cpilot_carriers_size - 1];
+ if (d_cpilot_phase_diff == NULL) {
+ std::cout << "Cannot allocate memory for d_tps_carriers_val" << std::endl;
+ delete [] d_known_phase_diff;
+ delete [] d_channel_gain;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ exit(1);
+ }
+
+ // Allocate buffer for derotated input symbol
+ d_derot_in = new gr_complex[d_fft_length];
+ if (d_derot_in == NULL) {
+ std::cout << "Cannot allocate memory for d_derot_in" << std::endl;
+ delete [] d_cpilot_phase_diff;
+ delete [] d_known_phase_diff;
+ delete [] d_channel_gain;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ exit(1);
+ }
+
+ // allocate buffer for first tps symbol constellation
+ d_tps_carriers_val = new gr_complex[d_tps_carriers_size];
+ if (d_tps_carriers_val == NULL) {
+ std::cout << "Cannot allocate memory for d_tps_carriers_val" << std::endl;
+ delete [] d_derot_in;
+ delete [] d_cpilot_phase_diff;
+ delete [] d_known_phase_diff;
+ delete [] d_channel_gain;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ exit(1);
+ }
+
+ // allocate tps data buffer
+ d_tps_data = new unsigned char[d_symbols_per_frame];
+ if (d_tps_data == NULL) {
+ std::cout << "Cannot allocate memory for d_tps_data" << std::endl;
+ delete [] d_tps_carriers_val;
+ delete [] d_derot_in;
+ delete [] d_cpilot_phase_diff;
+ delete [] d_known_phase_diff;
+ delete [] d_channel_gain;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ exit(1);
+ }
+
+ d_prev_tps_symbol = new gr_complex[d_tps_carriers_size];
+ if (d_prev_tps_symbol == NULL) {
+ std::cout << "Cannot allocate memory for d_tps_data" << std::endl;
+ delete [] d_tps_data;
+ delete [] d_tps_carriers_val;
+ delete [] d_derot_in;
+ delete [] d_cpilot_phase_diff;
+ delete [] d_known_phase_diff;
+ delete [] d_channel_gain;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ exit(1);
+ }
+ memset(d_prev_tps_symbol, 0, d_tps_carriers_size * sizeof(gr_complex));
+
+ d_tps_symbol = new gr_complex[d_tps_carriers_size];
+ if (d_tps_symbol == NULL) {
+ std::cout << "Cannot allocate memory for d_tps_data" << std::endl;
+ delete [] d_prev_tps_symbol;
+ delete [] d_tps_data;
+ delete [] d_tps_carriers_val;
+ delete [] d_derot_in;
+ delete [] d_cpilot_phase_diff;
+ delete [] d_known_phase_diff;
+ delete [] d_channel_gain;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ exit(1);
+ }
+ memset(d_tps_symbol, 0, d_tps_carriers_size * sizeof(gr_complex));
+
+ // Init receive TPS data vector
+ for (int i = 0; i < d_symbols_per_frame; i++) {
+ d_rcv_tps_data.push_back(0);
+ }
+
+ // Init TPS sync sequence
+ for (int i = 0; i < d_tps_sync_size; i++) {
+ d_tps_sync_evenv.push_back(d_tps_sync_even[i]);
+ d_tps_sync_oddv.push_back(d_tps_sync_odd[i]);
+ }
+
+ // Allocate buffer for channel estimation carriers
+ d_chanestim_carriers = new int[d_Kmax - d_Kmin + 1];
+ if (d_chanestim_carriers == NULL) {
+ std::cout << "Cannot allocate memory for d_tps_data" << std::endl;
+ delete [] d_tps_symbol;
+ delete [] d_prev_tps_symbol;
+ delete [] d_tps_data;
+ delete [] d_tps_carriers_val;
+ delete [] d_derot_in;
+ delete [] d_cpilot_phase_diff;
+ delete [] d_known_phase_diff;
+ delete [] d_channel_gain;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ exit(1);
+ }
+
+ // Allocate buffer for payload carriers
+ d_payload_carriers = new int[d_Kmax - d_Kmin + 1];
+ if (d_payload_carriers == NULL) {
+ std::cout << "Cannot allocate memory for d_tps_data" << std::endl;
+ delete [] d_chanestim_carriers;
+ delete [] d_tps_symbol;
+ delete [] d_prev_tps_symbol;
+ delete [] d_tps_data;
+ delete [] d_tps_carriers_val;
+ delete [] d_derot_in;
+ delete [] d_cpilot_phase_diff;
+ delete [] d_known_phase_diff;
+ delete [] d_channel_gain;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ exit(1);
+ }
+
+ // Reset the pilot generator
+ reset_pilot_generator();
+ // Format TPS data with current values
+ format_tps_data();
+ }
+
+ /*
+ * Destructor of class
+ */
+ dvbt_pilot_gen::~dvbt_pilot_gen()
+ {
+ delete [] d_payload_carriers;
+ delete [] d_chanestim_carriers;
+ delete [] d_tps_symbol;
+ delete [] d_prev_tps_symbol;
+ delete [] d_tps_data;
+ delete [] d_tps_carriers_val;
+ delete [] d_derot_in;
+ delete [] d_cpilot_phase_diff;
+ delete [] d_known_phase_diff;
+ delete [] d_channel_gain;
+ delete [] d_spilot_carriers_val;
+ delete [] d_wk;
+ }
+
+ /*
+ * Generate PRBS sequence
+ * X^11 + X^2 + 1
+ * en 300 744 - section 4.5.2
+ */
+ void
+ dvbt_pilot_gen::generate_prbs()
+ {
+ // init PRBS register with 1s
+ unsigned int reg_prbs = (1 << 11) - 1;
+
+ for (int k = 0; k < (d_Kmax - d_Kmin + 1); k++) {
+ d_wk[k] = (char)(reg_prbs & 0x01);
+ int new_bit = ((reg_prbs >> 2) ^ (reg_prbs >> 0)) & 0x01;
+ reg_prbs = (reg_prbs >> 1) | (new_bit << 10);
+ }
+ }
+
+ /*
+ * Generate shortened BCH(67, 53) codes from TPS data
+ * Extend the code with 60 bits and use BCH(127, 113)
+ */
+ void
+ dvbt_pilot_gen::generate_bch_code()
+ {
+ //TODO
+ //DO other way: if (feedback == 1) reg = reg ^ polymomial
+ //else nothing
+
+ //(n, k) = (127, 113) = (60+67, 60+53)
+ unsigned int reg_bch = 0;
+ unsigned char data_in[113];
+
+ //fill in 60 zeros
+ memset(&data_in[0], 0, 60);
+ //fill in TPS data - start bit not included
+ memcpy(&data_in[60], &d_tps_data[1], 53);
+
+ //X^14+X^9+X^8+X^6+X^5+X^4+X^2+X+1
+ for (int i = 0; i < 113; i++) {
+ int feedback = 0x1 & (data_in[i] ^ reg_bch);
+ reg_bch = reg_bch >> 1;
+ reg_bch |= feedback << 13;
+ reg_bch = reg_bch \
+ ^ (feedback << 12) ^ (feedback << 11) \
+ ^ (feedback << 9) ^ (feedback << 8) \
+ ^ (feedback << 7) ^ (feedback << 5) \
+ ^ (feedback << 4);
+ }
+
+ for (int i = 0; i < 14; i++) {
+ d_tps_data[i + 54] = 0x1 & (reg_bch >> i);
+ }
+ }
+
+ int
+ dvbt_pilot_gen::verify_bch_code(std::deque<char> data)
+ {
+ int ret = 0;
+
+ //TODO
+ //DO other way: if (feedback == 1) reg = reg ^ polymomial
+ //else nothing
+
+ //(n, k) = (127, 113) = (60+67, 60+53)
+ unsigned int reg_bch = 0;
+ unsigned char data_in[113];
+
+ //fill in 60 zeros
+ memset(&data_in[0], 0, 60);
+ //fill in TPS data - start bit not included
+ //memcpy(&data_in[60], &data[1], 53);
+ for (int i = 0; i < 53; i++)
+ data_in[60 + i] = data[1 + i];
+
+ //X^14+X^9+X^8+X^6+X^5+X^4+X^2+X+1
+ for (int i = 0; i < 113; i++) {
+ int feedback = 0x1 & (data_in[i] ^ reg_bch);
+ reg_bch = reg_bch >> 1;
+ reg_bch |= feedback << 13;
+ reg_bch = reg_bch \
+ ^ (feedback << 12) ^ (feedback << 11) \
+ ^ (feedback << 9) ^ (feedback << 8) \
+ ^ (feedback << 7) ^ (feedback << 5) \
+ ^ (feedback << 4);
+ }
+
+ for (int i = 0; i < 14; i++) {
+ if ((unsigned int)data[i + 54] != (0x1 & (reg_bch >> i))) {
+ ret = -1;
+ break;
+ }
+ }
+
+ return ret;
+ }
+
+ void
+ dvbt_pilot_gen::set_symbol_index(int sindex)
+ {
+ d_symbol_index = sindex;
+ }
+
+ int
+ dvbt_pilot_gen::get_symbol_index()
+ {
+ return d_symbol_index;
+ }
+
+ void
+ dvbt_pilot_gen::set_tps_data()
+ {
+ }
+
+ void
+ dvbt_pilot_gen::get_tps_data()
+ {
+ }
+
+ /*
+ * Reset pilot generator
+ */
+ void
+ dvbt_pilot_gen::reset_pilot_generator()
+ {
+ d_spilot_index = 0; d_cpilot_index = 0; d_tpilot_index = 0;
+ d_payload_index = 0; d_chanestim_index = 0;
+ d_symbol_index = 0; d_frame_index = 0; d_superframe_index = 0;
+ d_symbol_index_known = 0;
+ d_equalizer_ready = 0;
+ }
+
+ /*
+ * Init scattered pilot generator
+ */
+ int
+ dvbt_pilot_gen::get_current_spilot(int sindex) const
+ {
+ //TODO - can be optimized for same symbol_index
+ return (d_Kmin + 3 * (sindex % 4) + 12 * d_spilot_index);
+ }
+
+ gr_complex
+ dvbt_pilot_gen::get_spilot_value(int spilot)
+ {
+ // TODO - can be calculated at the beginning
+ return gr_complex(4 * 2 * (0.5 - d_wk[spilot]) / 3, 0);
+ }
+
+ void
+ dvbt_pilot_gen::set_spilot_value(int spilot, gr_complex val)
+ {
+ d_spilot_carriers_val[spilot] = val;
+ }
+
+ void
+ dvbt_pilot_gen::set_channel_gain(int spilot, gr_complex val)
+ {
+ // Gain gval=rxval/txval
+ d_channel_gain[spilot] = gr_complex((4 * 2 * (0.5 - d_wk[spilot]) / 3), 0) / val;
+ }
+ void
+ dvbt_pilot_gen::advance_spilot(int sindex)
+ {
+ //TODO - do in a simpler way?
+ int size = d_spilot_carriers_size;
+
+ if (sindex == 0) {
+ size = d_spilot_carriers_size + 1;
+ }
+
+ // TODO - fix this - what value should we use?
+ ++d_spilot_index;
+ d_spilot_index = d_spilot_index % size;
+ }
+
+ int
+ dvbt_pilot_gen::get_first_spilot()
+ {
+ d_spilot_index = 0;
+
+ return (d_Kmin + 3 * (d_symbol_index % 4));
+ }
+
+ int
+ dvbt_pilot_gen::get_last_spilot() const
+ {
+ int size = d_spilot_carriers_size - 1;
+
+ if (d_symbol_index == 0) {
+ size = d_spilot_carriers_size;
+ }
+
+ return (d_Kmin + 3 * (d_symbol_index % 4) + 12 * size);
+ }
+
+ int
+ dvbt_pilot_gen::get_next_spilot()
+ {
+ int pilot = (d_Kmin + 3 * (d_symbol_index % 4) + 12 * (++d_spilot_index));
+
+ if (pilot > d_Kmax) {
+ pilot = d_Kmax;
+ }
+
+ return pilot;
+ }
+
+ int
+ dvbt_pilot_gen::process_spilot_data(const gr_complex * in)
+ {
+ // This is channel estimator
+ // Interpolate the gain between carriers to obtain
+ // gain for non pilot carriers - we use linear interpolation
+
+ /*************************************************************/
+ // Find out the OFDM symbol index (value 0 to 3) sent
+ // in current block by correlating scattered symbols with
+ // current block - result is (symbol index % 4)
+ /*************************************************************/
+ float max = 0; float sum = 0;
+
+ for (int scount = 0; scount < 4; scount++) {
+ d_spilot_index = 0; d_cpilot_index = 0;
+ d_chanestim_index = 0;
+
+ for (int k = 0; k < (d_Kmax - d_Kmin + 1); k++) {
+ // Keep data for channel estimation
+ if (k == get_current_spilot(scount)) {
+ set_chanestim_carrier(k);
+ advance_spilot(scount);
+ advance_chanestim();
+ }
+ }
+
+ gr_complex c = gr_complex(0.0, 0.0);
+
+ // This should be of range 0 to d_chanestim_index bit for now we use just a
+ // small number of spilots to obtain the symbol index
+ for (int j = 0; j < 10; j++) {
+ c += get_spilot_value(d_chanestim_carriers[j]) * conj(in[d_zeros_on_left + d_chanestim_carriers[j]]);
+ }
+ sum = norm(c);
+
+ if (sum > max) {
+ max = sum;
+ d_mod_symbol_index = scount;
+ }
+ }
+
+ /*************************************************************/
+ // Keep data for channel estimator
+ // This method interpolates scattered measurements across one OFDM symbol
+ // It does not use measurements from the previous OFDM symbols (does not use history)
+ // as it may have encountered a phase change for the current phase only
+ /*************************************************************/
+
+ d_spilot_index = 0; d_cpilot_index = 0;
+ d_chanestim_index = 0;
+
+ for (int k = 0; k < (d_Kmax - d_Kmin + 1); k++) {
+ // Keep data for channel estimation
+ if (k == get_current_spilot(d_mod_symbol_index)) {
+ set_chanestim_carrier(k);
+ advance_spilot(d_mod_symbol_index);
+ advance_chanestim();
+ }
+
+ // Keep data for channel estimation
+ if (k == get_current_cpilot()) {
+ set_chanestim_carrier(k);
+ advance_cpilot();
+ advance_chanestim();
+ }
+ }
+
+ // We use both scattered pilots and continual pilots
+ for (int i = 0, startk = d_chanestim_carriers[0]; i < d_chanestim_index; i++) {
+ // Get a carrier from the list of carriers
+ // used for channel estimation
+ int k = d_chanestim_carriers[i];
+
+ set_channel_gain(k, in[k + d_zeros_on_left]);
+
+ // Calculate tg(alpha) due to linear interpolation
+ gr_complex tg_alpha = (d_channel_gain[k] - d_channel_gain[startk]) / gr_complex(11.0, 0.0);
+
+ // Calculate interpolation for all intermediate values
+ for (int j = 1; j < (k - startk); j++) {
+ gr_complex current = d_channel_gain[startk] + tg_alpha * gr_complex(j, 0.0);
+ d_channel_gain[startk + j] = current;
+ }
+
+ startk = k;
+ }
+
+ // Signal that equalizer is ready
+ d_equalizer_ready = 1;
+
+ int diff_sindex = (d_mod_symbol_index - d_prev_mod_symbol_index + 4) % 4;
+
+ d_prev_mod_symbol_index = d_mod_symbol_index;
+
+ return diff_sindex;
+ }
+
+ /*
+ * Init continual pilot generator
+ */
+ int
+ dvbt_pilot_gen::get_current_cpilot() const
+ {
+ return d_cpilot_carriers[d_cpilot_index];
+ }
+
+ gr_complex
+ dvbt_pilot_gen::get_cpilot_value(int cpilot)
+ {
+ //TODO - can be calculated at the beginning
+ return gr_complex((float)(4 * 2 * (0.5 - d_wk[cpilot])) / 3, 0);
+ }
+
+ void
+ dvbt_pilot_gen::advance_cpilot()
+ {
+ ++d_cpilot_index;
+ d_cpilot_index = d_cpilot_index % d_cpilot_carriers_size;
+ }
+
+ void
+ dvbt_pilot_gen::process_cpilot_data(const gr_complex * in)
+ {
+ // Look for maximum correlation for cpilots
+ // in order to obtain post FFT integer frequency correction
+
+ float max = 0; float sum = 0;
+ int start = 0;
+ float phase;
+
+ for (int i = d_zeros_on_left - d_freq_offset_max; i < d_zeros_on_left + d_freq_offset_max; i++) {
+ sum = 0;
+ for (int j = 0; j < (d_cpilot_carriers_size - 1); j++) {
+ phase = norm(in[i + d_cpilot_carriers[j + 1]] - in[i + d_cpilot_carriers[j]]);
+ sum += d_known_phase_diff[j] * phase;
+ }
+
+ if (sum > max) {
+ max = sum;
+ start = i;
+ }
+ }
+
+ d_freq_offset = start - d_zeros_on_left;
+ }
+
+ void
+ dvbt_pilot_gen::compute_oneshot_csft(const gr_complex * in)
+ {
+ gr_complex left_corr_sum = 0.0; gr_complex right_corr_sum = 0.0;
+ int half_size = (d_cpilot_carriers_size - 1) / 2;
+
+ // TODO init this in constructor
+ float carrier_coeff = 1.0 / (2 * M_PI * (1 + float (d_cp_length) / float (d_fft_length)) * 2);
+ float sampling_coeff = 1.0 / (2 * M_PI * ((1 + float (d_cp_length) / float (d_fft_length)) * ((float)d_cpilot_carriers_size / 2.0)));
+
+ float left_angle, right_angle;
+
+ // Compute cpilots correlation between previous symbol and current symbol
+ // in both halves of the cpilots. The cpilots are distributed evenly
+ // on left and right sides of the center frequency.
+
+ for (int j = 0; j < half_size; j++) {
+ left_corr_sum += in[d_freq_offset + d_zeros_on_left + d_cpilot_carriers[j]] * \
+ std::conj(in[d_freq_offset + d_fft_length + d_zeros_on_left + d_cpilot_carriers[j]]);
+ }
+
+ for (int j = half_size + 1; j < d_cpilot_carriers_size; j++) {
+ right_corr_sum += in[d_freq_offset + d_zeros_on_left + d_cpilot_carriers[j]] * \
+ std::conj(in[d_freq_offset + d_fft_length + d_zeros_on_left + d_cpilot_carriers[j]]);
+ }
+
+ left_angle = std::arg(left_corr_sum);
+ right_angle = std::arg(right_corr_sum);
+
+ d_carrier_freq_correction = (right_angle + left_angle) * carrier_coeff;
+ d_sampling_freq_correction = (right_angle - left_angle) * sampling_coeff;
+ }
+
+ gr_complex *
+ dvbt_pilot_gen::frequency_correction(const gr_complex * in, gr_complex * out)
+ {
+ // TODO - use PI control loop to calculate tracking corrections
+ int symbol_count = 1;
+
+ for (int k = 0; k < d_fft_length; k++) {
+ // TODO - for 2k mode the continuous pilots are not split evenly
+ // between left/right center frequency. Probably the scattered
+ // pilots needs to be added.
+
+ float correction = (float)d_freq_offset + d_carrier_freq_correction;
+
+ gr_complex c = gr_expj(-2 * M_PI * correction * \
+ (d_fft_length + d_cp_length) / d_fft_length * symbol_count);
+
+ // TODO - vectorize this operation
+ out[k] = c * in[k + d_freq_offset];
+ }
+
+ return (out);
+ }
+
+ /*
+ * Init tps sequence, return values for first position
+ * If first symbol then init tps DBPSK data
+ */
+ int
+ dvbt_pilot_gen::get_current_tpilot() const
+ {
+ return d_tps_carriers[d_tpilot_index];
+ }
+
+ gr_complex
+ dvbt_pilot_gen::get_tpilot_value(int tpilot)
+ {
+ //TODO - it can be calculated at the beginnning
+ if (d_symbol_index == 0) {
+ d_tps_carriers_val[d_tpilot_index] = gr_complex(2 * (0.5 - d_wk[tpilot]), 0);
+ }
+ else {
+ if (d_tps_data[d_symbol_index] == 1) {
+ d_tps_carriers_val[d_tpilot_index] = gr_complex(-d_tps_carriers_val[d_tpilot_index].real(), 0);
+ }
+ }
+
+ return d_tps_carriers_val[d_tpilot_index];
+ }
+
+ void
+ dvbt_pilot_gen::advance_tpilot()
+ {
+ ++d_tpilot_index;
+ d_tpilot_index = d_tpilot_index % d_tps_carriers_size;
+ }
+
+ /*
+ * Set a number of bits to a specified value
+ */
+ void
+ dvbt_pilot_gen::set_tps_bits(int start, int stop, unsigned int data)
+ {
+ for (int i = start; i >= stop; i--) {
+ d_tps_data[i] = data & 0x1;
+ data = data >> 1;
+ }
+ }
+
+ /*
+ * Clause 4.6
+ * Format data that will be sent with TPS signals
+ * en 300 744 - section 4.6.2
+ * s0 Initialization
+ * s1-s16 Synchronization word
+ * s17-s22 Length Indicator
+ * s23-s24 Frame Number
+ * S25-s26 Constellation
+ * s27, s28, s29 Hierarchy information
+ * s30, s31, s32 Code rate, HP stream
+ * s33, s34, s35 Code rate, LP stream
+ * s36, s37 Guard interval
+ * s38, s39 Transmission mode
+ * s40, s47 Cell identifier
+ * s48-s53 All set to "0"
+ * s54-s67 Error protection (BCH code)
+ */
+ void
+ dvbt_pilot_gen::format_tps_data()
+ {
+ //Clause 4.6.3
+ set_tps_bits(0, 0, d_wk[0]);
+ //Clause 4.6.2.2
+ if (d_frame_index % 2) {
+ set_tps_bits(16, 1, 0xca11);
+ }
+ else {
+ set_tps_bits(16, 1, 0x35ee);
+ }
+ //Clause 4.6.2.3
+ if (config.d_include_cell_id) {
+ set_tps_bits(22, 17, 0x1f);
+ }
+ else {
+ set_tps_bits(22, 17, 0x17);
+ }
+ //Clause 4.6.2.4
+ set_tps_bits(24, 23, d_frame_index);
+ //Clause 4.6.2.5
+ set_tps_bits(26, 25, config.d_constellation);
+ //Clause 4.6.2.6
+ set_tps_bits(29, 27, config.d_hierarchy);
+ //Clause 4.6.2.7
+ switch (config.d_code_rate_HP) {
+ case C1_2:
+ set_tps_bits(32, 30, 0);
+ break;
+ case C2_3:
+ set_tps_bits(32, 30, 1);
+ break;
+ case C3_4:
+ set_tps_bits(32, 30, 2);
+ break;
+ case C5_6:
+ set_tps_bits(32, 30, 3);
+ break;
+ case C7_8:
+ set_tps_bits(32, 30, 4);
+ break;
+ default:
+ set_tps_bits(32, 30, 0);
+ break;
+ }
+ switch (config.d_code_rate_LP) {
+ case C1_2:
+ set_tps_bits(35, 33, 0);
+ break;
+ case C2_3:
+ set_tps_bits(35, 33, 1);
+ break;
+ case C3_4:
+ set_tps_bits(35, 33, 2);
+ break;
+ case C5_6:
+ set_tps_bits(35, 33, 3);
+ break;
+ case C7_8:
+ set_tps_bits(35, 33, 4);
+ break;
+ default:
+ set_tps_bits(35, 33, 0);
+ break;
+ }
+ //Clause 4.6.2.8
+ set_tps_bits(37, 36, config.d_guard_interval);
+ //Clause 4.6.2.9
+ set_tps_bits(39, 38, config.d_transmission_mode);
+ //Clause 4.6.2.10
+ set_tps_bits(47, 40, config.d_cell_id);
+ //These bits are set to zero
+ set_tps_bits(53, 48, 0);
+ //Clause 4.6.2.11
+ generate_bch_code();
+ }
+
+ int
+ dvbt_pilot_gen::process_tps_data(const gr_complex * in, const int diff_symbol_index)
+ {
+ int end_frame = 0;
+
+ // Look for TPS data only - demodulate DBPSK
+ // Calculate phase difference between previous symbol
+ // and current one to determine the current bit.
+ // Use majority voting for decision
+ int tps_majority_zero = 0;
+
+ for (int k = 0; k < d_tps_carriers_size; k++) {
+ // Use equalizer to correct data and frequency correction
+ gr_complex val = in[d_zeros_on_left + d_tps_carriers[k]] * d_channel_gain[d_tps_carriers[k]];
+
+ if (!d_symbol_index_known || (d_symbol_index != 0)) {
+ gr_complex phdiff = val * conj(d_prev_tps_symbol[k]);
+
+ if (phdiff.real() >= 0.0) {
+ tps_majority_zero++;
+ }
+ else {
+ tps_majority_zero--;
+ }
+ }
+
+ d_prev_tps_symbol[k] = val;
+ }
+
+ // Insert obtained TPS bit into FIFO
+ // Insert the same bit into FIFO in the case
+ // diff_symbol_index is more than one. This will happen
+ // in the case of losing 1 to 3 symbols.
+ // This could be corrected by BCH decoder afterwards.
+
+ for (int i = 0; i < diff_symbol_index; i++) {
+ // Take out the front entry first
+ d_rcv_tps_data.pop_front();
+
+ // Add data at tail
+ if (!d_symbol_index_known || (d_symbol_index != 0)) {
+ if (tps_majority_zero >= 0) {
+ d_rcv_tps_data.push_back(0);
+ }
+ else {
+ d_rcv_tps_data.push_back(1);
+ }
+ }
+ else {
+ d_rcv_tps_data.push_back(0);
+ }
+ }
+
+ // Match synchronization signatures
+ if (std::equal(d_rcv_tps_data.begin() + 1, d_rcv_tps_data.begin() + d_tps_sync_evenv.size(), d_tps_sync_evenv.begin())) {
+ // Verify parity for TPS data
+ if (!verify_bch_code(d_rcv_tps_data)) {
+ d_frame_index = (d_rcv_tps_data[23] << 1) | (d_rcv_tps_data[24]);
+
+ d_symbol_index_known = 1;
+ end_frame = 1;
+ }
+ else {
+ d_symbol_index_known = 0;
+ end_frame = 0;
+ }
+
+ // Clear up FIFO
+ for (int i = 0; i < d_symbols_per_frame; i++) {
+ d_rcv_tps_data[i] = 0;
+ }
+ }
+ else if (std::equal(d_rcv_tps_data.begin() + 1, d_rcv_tps_data.begin() + d_tps_sync_oddv.size(), d_tps_sync_oddv.begin())) {
+ // Verify parity for TPS data
+ if (!verify_bch_code(d_rcv_tps_data)) {
+ d_frame_index = (d_rcv_tps_data[23] << 1) | (d_rcv_tps_data[24]);
+
+ d_symbol_index_known = 1;
+ end_frame = 1;
+ }
+ else {
+ d_symbol_index_known = 0;
+ end_frame = 0;
+ }
+
+ // Clear up FIFO
+ for (int i = 0; i < d_symbols_per_frame; i++) {
+ d_rcv_tps_data[i] = 0;
+ }
+ }
+
+ return end_frame;
+ }
+
+ void
+ dvbt_pilot_gen::set_chanestim_carrier(int k)
+ {
+ d_chanestim_carriers[d_chanestim_index] = k;
+ }
+
+ void
+ dvbt_pilot_gen::advance_chanestim()
+ {
+ d_chanestim_index++;
+ }
+
+ int
+ dvbt_pilot_gen::get_current_payload()
+ {
+ return d_payload_carriers[d_payload_index];
+ }
+
+ void
+ dvbt_pilot_gen::set_payload_carrier(int k)
+ {
+ d_payload_carriers[d_payload_index] = k;
+ }
+
+ void
+ dvbt_pilot_gen::advance_payload()
+ {
+ d_payload_index++;
+ }
+
+ void
+ dvbt_pilot_gen::process_payload_data(const gr_complex *in, gr_complex *out)
+ {
+ //reset indexes
+ d_spilot_index = 0; d_cpilot_index = 0; d_tpilot_index = 0;
+ d_payload_index = 0;d_chanestim_index = 0;
+ int is_payload = 1;
+
+ //process one block - one symbol
+ for (int k = 0; k < (d_Kmax - d_Kmin + 1); k++) {
+ is_payload = 1;
+
+ // Keep data for channel estimation
+ // This depends on the symbol index
+ if (k == get_current_spilot(d_mod_symbol_index)) {
+ advance_spilot(d_mod_symbol_index);
+ is_payload = 0;
+ }
+
+ // Keep data for frequency correction
+ // and channel estimation
+ if (k == get_current_cpilot()) {
+ advance_cpilot();
+ is_payload = 0;
+ }
+
+ if (k == get_current_tpilot()) {
+ advance_tpilot();
+ is_payload = 0;
+ }
+
+ // Keep payload carrier number
+ // This depends on the symbol index
+ if (is_payload) {
+ set_payload_carrier(k);
+ advance_payload();
+ }
+ }
+
+ if (d_equalizer_ready) {
+ // Equalize payload data according to channel estimator
+ for (int i = 0; i < d_payload_index; i++) {
+ out[i] = in[d_zeros_on_left + d_payload_carriers[i]] * d_channel_gain[d_payload_carriers[i]];
+ }
+ }
+ else {
+ // If equ not ready, return 0
+ for (int i = 0; i < d_payload_length; i++) {
+ out[0] = gr_complex(0.0, 0.0);
+ }
+ }
+ }
+
+ void
+ dvbt_pilot_gen::update_output(const gr_complex *in, gr_complex *out)
+ {
+ int is_payload = 1;
+ int payload_count = 0;
+
+ //move to the next symbol
+ //re-genereate TPS data
+ format_tps_data();
+
+ //reset indexes
+ payload_count = 0;
+ d_spilot_index = 0; d_cpilot_index = 0; d_tpilot_index = 0;
+
+ for (int i = 0; i < d_zeros_on_left; i++) {
+ out[i] = gr_complex(0.0, 0.0);
+ }
+
+ //process one block - one symbol
+ for (int k = d_Kmin; k < (d_Kmax - d_Kmin + 1); k++) {
+ is_payload = 1;
+ if (k == get_current_spilot(d_symbol_index)) {
+ out[d_zeros_on_left + k] = get_spilot_value(k);
+ advance_spilot(d_symbol_index);
+ is_payload = 0;
+ }
+
+ if (k == get_current_cpilot()) {
+ out[d_zeros_on_left + k] = get_cpilot_value(k);
+ advance_cpilot();
+ is_payload = 0;
+ }
+
+ if (k == get_current_tpilot()) {
+ out[d_zeros_on_left + k] = get_tpilot_value(k);
+ advance_tpilot();
+ is_payload = 0;
+ }
+
+ if (is_payload == 1) {
+ out[d_zeros_on_left + k] = in[payload_count++];
+ }
+ }
+
+ // update indexes
+ if (++d_symbol_index == d_symbols_per_frame) {
+ d_symbol_index = 0;
+ if (++d_frame_index == d_frames_per_superframe) {
+ d_frame_index = 0;
+ d_superframe_index++;
+ }
+ }
+
+ for (int i = (d_fft_length - d_zeros_on_right); i < d_fft_length; i++) {
+ out[i] = gr_complex(0.0, 0.0);
+ }
+ }
+
+ int
+ dvbt_pilot_gen::parse_input(const gr_complex *in, gr_complex *out, int * symbol_index, int * frame_index)
+ {
+ d_trigger_index++;
+
+ // Obtain frequency correction based on cpilots.
+ // Obtain channel estimation based on both
+ // cpilots and spilots.
+ // We use spilot correlation for finding the symbol index modulo 4
+ // The diff between previous sym index and current index is used
+ // to advance the symbol index inside a frame (0 to 67)
+ // Then based on the TPS data we find out the start of a frame
+
+ // Process cpilot data
+ // This is post FFT integer frequency offset estimation
+ // This is called before all other processing
+ process_cpilot_data(in);
+
+ // Compute one shot Post-FFT Carrier and Sampling Frequency Tracking
+ // Obtain fractional Carrer and Sampling frequency corrections
+ // Before this moment it is assumed to have corrected this:
+ // - symbol timing (pre-FFT)
+ // - symbol frequency correction (pre-FFT)
+ // - integer frequency correction (post-FFT)
+ // TODO - call this just in the aquisition mode
+ compute_oneshot_csft(in);
+
+ // Gather all corrections and obtain a corrected OFDM symbol:
+ // - input symbol shift (post-FFT)
+ // - integer frequency correction (post-FFT)
+ // - fractional frequency (carrier and sampling) corrections (post-FFT)
+ // TODO - use PI to update the corrections
+ frequency_correction(in, d_derot_in);
+
+ // Process spilot data
+ // This is channel estimation function
+ int diff_symbol_index = process_spilot_data(d_derot_in);
+
+ // Correct symbol index so that all subsequent processing
+ // use correct symbol index
+ d_symbol_index = (d_symbol_index + diff_symbol_index) % d_symbols_per_frame;
+
+ // Symbol index is used in other modules too
+ *symbol_index = d_symbol_index;
+ // Frame index is used in other modules too
+ *frame_index = d_frame_index;
+
+ // Process TPS data
+ // If a frame is recognized then signal end of frame
+ int frame_end = process_tps_data(d_derot_in, diff_symbol_index);
+
+ // We are just at the end of a frame
+ if (frame_end) {
+ d_symbol_index = d_symbols_per_frame - 1;
+ }
+
+ // Process payload data with correct symbol index
+ process_payload_data(d_derot_in, out);
+
+ // noutput_items should be 1 in this case
+ return 1;
+ }
+
+ dvbt_reference_signals::sptr
+ dvbt_reference_signals::make(int itemsize, int ninput, int noutput, \
+ dvb_constellation_t constellation, dvbt_hierarchy_t hierarchy, \
+ dvb_code_rate_t code_rate_HP, dvb_code_rate_t code_rate_LP, \
+ dvb_guardinterval_t guard_interval, dvbt_transmission_mode_t transmission_mode, \
+ int include_cell_id, int cell_id)
+ {
+ return gnuradio::get_initial_sptr
+ (new dvbt_reference_signals_impl(itemsize, ninput, \
+ noutput, constellation, hierarchy, code_rate_HP, code_rate_LP, \
+ guard_interval, transmission_mode, include_cell_id, cell_id));
+ }
+
+ /*
+ * The private constructor
+ */
+ dvbt_reference_signals_impl::dvbt_reference_signals_impl(int itemsize, int ninput, int noutput, \
+ dvb_constellation_t constellation, dvbt_hierarchy_t hierarchy, dvb_code_rate_t code_rate_HP,\
+ dvb_code_rate_t code_rate_LP, dvb_guardinterval_t guard_interval,\
+ dvbt_transmission_mode_t transmission_mode, int include_cell_id, int cell_id)
+ : block("dvbt_reference_signals",
+ io_signature::make(1, 1, itemsize * ninput),
+ io_signature::make(1, 1, itemsize * noutput)),
+ config(constellation, hierarchy, code_rate_HP, code_rate_LP, \
+ guard_interval, transmission_mode, include_cell_id, cell_id),
+ d_pg(config)
+ {
+ d_ninput = ninput;
+ d_noutput = noutput;
+ }
+
+ /*
+ * Our virtual destructor.
+ */
+ dvbt_reference_signals_impl::~dvbt_reference_signals_impl()
+ {
+ }
+
+ void
+ dvbt_reference_signals_impl::forecast (int noutput_items, gr_vector_int &ninput_items_required)
+ {
+ ninput_items_required[0] = noutput_items;
+ }
+
+ int
+ dvbt_reference_signals_impl::general_work (int noutput_items,
+ gr_vector_int &ninput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ const gr_complex *in = (const gr_complex *) input_items[0];
+ gr_complex *out = (gr_complex *) output_items[0];
+
+ for (int i = 0; i < noutput_items; i++) {
+ d_pg.update_output(&in[i * d_ninput], &out[i * d_noutput]);
+ }
+
+ // Tell runtime system how many input items we consumed on
+ // each input stream.
+ consume_each (noutput_items);
+
+ // Tell runtime system how many output items we produced.
+ return noutput_items;
+ }
+
+ } /* namespace dtv */
+} /* namespace gr */
+
+