/** * * \section COPYRIGHT * * Copyright 2013-2015 The srsLTE Developers. See the * COPYRIGHT file at the top-level directory of this distribution. * * \section LICENSE * * This file is part of the srsLTE library. * * srsLTE is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of * the License, or (at your option) any later version. * * srsLTE 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 Affero General Public License for more details. * * A copy of the GNU Affero General Public License can be found in * the LICENSE file in the top-level directory of this distribution * and at http://www.gnu.org/licenses/. * */ #include #include "srslte/utils/debug.h" #include "srsapps/ue/phy/phy.h" #include "srsapps/common/log_stdout.h" #include "srsapps/common/tti_sync_cv.h" #include "srsapps/radio/radio_uhd.h" /********************************************************************** * Program arguments processing ***********************************************************************/ typedef struct { float uhd_freq; float uhd_gain; }prog_args_t; void args_default(prog_args_t *args) { args->uhd_freq = -1.0; args->uhd_gain = -1.0; } void usage(prog_args_t *args, char *prog) { printf("Usage: %s [gv] -f rx_frequency (in Hz)\n", prog); printf("\t-g UHD RX gain [Default AGC]\n"); printf("\t-v [increase verbosity, default none]\n"); } void parse_args(prog_args_t *args, int argc, char **argv) { int opt; args_default(args); while ((opt = getopt(argc, argv, "gfv")) != -1) { switch (opt) { case 'g': args->uhd_gain = atof(argv[optind]); break; case 'f': args->uhd_freq = atof(argv[optind]); break; case 'v': srslte_verbose++; break; default: usage(args, argv[0]); exit(-1); } } if (args->uhd_freq < 0) { usage(args, argv[0]); exit(-1); } } srslte::ue::phy phy; prog_args_t prog_args; uint32_t total_pkts=0; uint32_t total_dci=0; uint32_t total_errors=0; uint8_t payload[1024]; // This is the MAC implementation void run_tti(uint32_t tti) { srslte::ue::dl_sched_grant grant(SRSLTE_SIRNTI); INFO("MAC running tti: %d\n", tti); // SIB1 is scheduled in subframe #5 of even frames if ((phy.tti_to_SFN(tti)%2) == 0 && phy.tti_to_subf(tti) == 5) { // Get buffer srslte::ue::dl_buffer *buffer = phy.get_dl_buffer(tti); // Get DL grant if (buffer->get_dl_grant(&grant)) { total_dci++; // MAC sets RV grant.set_rv(((uint32_t) ceilf((float)3*((phy.tti_to_SFN(tti)/2)%4)/2))%4); // Decode packet if (!buffer->decode_data(&grant, payload)) { total_errors++; } } total_pkts++; } float gain = prog_args.uhd_gain; if (gain < 0) { gain = phy.get_agc_gain(); } if (srslte_verbose == SRSLTE_VERBOSE_NONE) { printf("PDCCH BLER %.1f \%% PDSCH BLER %.1f \%% (total pkts: %5u) Gain: %.1f dB\r", 100-(float) 100*total_dci/total_pkts, (float) 100*total_errors/total_pkts, total_pkts, gain); } } int main(int argc, char *argv[]) { srslte_cell_t cell; uint8_t bch_payload[SRSLTE_BCH_PAYLOAD_LEN]; srslte::ue::tti_sync_cv ttisync(10240); srslte::radio_uhd radio_uhd; srslte::log_stdout log("PHY"); parse_args(&prog_args, argc, argv); // Init Radio and PHY if (prog_args.uhd_gain > 0) { radio_uhd.init(); radio_uhd.set_rx_gain(prog_args.uhd_gain); phy.init(&radio_uhd, &ttisync, &log); } else { radio_uhd.init_agc(); phy.init_agc(&radio_uhd, &ttisync, &log); } // Give it time to create thread sleep(1); // Set default parameters phy.set_param(srslte::ue::phy_params::PRACH_CONFIG_INDEX, 0); phy.set_param(srslte::ue::phy_params::PRACH_ROOT_SEQ_IDX, 0); phy.set_param(srslte::ue::phy_params::PRACH_HIGH_SPEED_FLAG, 0); phy.set_param(srslte::ue::phy_params::PRACH_ZC_CONFIG, 1); // Set RX freq and gain phy.get_radio()->set_rx_freq(prog_args.uhd_freq); phy.get_radio()->set_rx_gain(prog_args.uhd_gain); /* Instruct the PHY to decode BCH */ if (!phy.decode_mib_best(&cell, bch_payload)) { exit(-1); } // Print MIB srslte_cell_fprint(stdout, &cell, phy.get_current_tti()/10); // Set the current PHY cell to the detected cell if (!phy.set_cell(cell)) { printf("Error setting cell\n"); exit(-1); } /* Instruct the PHY to start RX streaming and synchronize */ if (!phy.start_rxtx()) { printf("Could not start RX\n"); exit(-1); } /* go to idle and process each tti */ while(1) { uint32_t tti = ttisync.wait(); run_tti(tti); } }