/* * Copyright 2013-2019 Software Radio Systems Limited * * This file is part of srsLTE. * * 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 "srsue/hdr/phy/scell/intra_measure.h" #define Error(fmt, ...) \ if (SRSLTE_DEBUG_ENABLED) \ log_h->error(fmt, ##__VA_ARGS__) #define Warning(fmt, ...) \ if (SRSLTE_DEBUG_ENABLED) \ log_h->warning(fmt, ##__VA_ARGS__) #define Info(fmt, ...) \ if (SRSLTE_DEBUG_ENABLED) \ log_h->info(fmt, ##__VA_ARGS__) #define Debug(fmt, ...) \ if (SRSLTE_DEBUG_ENABLED) \ log_h->debug(fmt, ##__VA_ARGS__) namespace srsue { namespace scell { intra_measure::intra_measure() : scell(), thread("SYNC_INTRA_MEASURE") { } intra_measure::~intra_measure() { srslte_ringbuffer_free(&ring_buffer); scell.deinit(); free(search_buffer); } void intra_measure::init(phy_common* common, rrc_interface_phy_lte* rrc, srslte::log* log_h) { this->rrc = rrc; this->log_h = log_h; this->common = common; receive_enabled = false; // Initialise Reference signal measurement srslte_refsignal_dl_sync_init(&refsignal_dl_sync); // Start scell scell.init(log_h, common->args->sic_pss_enabled, common->args->intra_freq_meas_len_ms, common); search_buffer = (cf_t*)srslte_vec_malloc(common->args->intra_freq_meas_len_ms * SRSLTE_SF_LEN_PRB(SRSLTE_MAX_PRB) * sizeof(cf_t)); if (srslte_ringbuffer_init( &ring_buffer, sizeof(cf_t) * common->args->intra_freq_meas_len_ms * 2 * SRSLTE_SF_LEN_PRB(SRSLTE_MAX_PRB))) { return; } running = true; start(INTRA_FREQ_MEAS_PRIO); } void intra_measure::stop() { running = false; srslte_ringbuffer_stop(&ring_buffer); tti_sync.increase(); wait_thread_finish(); srslte_refsignal_dl_sync_free(&refsignal_dl_sync); } void intra_measure::set_primay_cell(uint32_t earfcn, srslte_cell_t cell) { this->current_earfcn = earfcn; current_sflen = (uint32_t)SRSLTE_SF_LEN_PRB(cell.nof_prb); this->primary_cell = cell; } void intra_measure::clear_cells() { active_pci.clear(); receive_enabled = false; receiving = false; receive_cnt = 0; srslte_ringbuffer_reset(&ring_buffer); } void intra_measure::add_cell(int pci) { if (std::find(active_pci.begin(), active_pci.end(), pci) == active_pci.end()) { active_pci.push_back(pci); receive_enabled = true; Info("INTRA: Starting intra-frequency measurement for pci=%d\n", pci); } else { Debug("INTRA: Requested to start already existing intra-frequency measurement for PCI=%d\n", pci); } } int intra_measure::get_offset(uint32_t pci) { for (auto& i : info) { if (i.pci == pci) { return i.offset; } } return -1; } void intra_measure::rem_cell(int pci) { auto newEnd = std::remove(active_pci.begin(), active_pci.end(), pci); if (newEnd != active_pci.end()) { active_pci.erase(newEnd, active_pci.end()); if (active_pci.empty()) { receive_enabled = false; } Info("INTRA: Stopping intra-frequency measurement for pci=%d. Number of cells: %zu\n", pci, active_pci.size()); } else { Warning("INTRA: Requested to stop non-existing intra-frequency measurement for PCI=%d\n", pci); } } void intra_measure::write(uint32_t tti, cf_t* data, uint32_t nsamples) { if (receive_enabled) { if ((tti % common->args->intra_freq_meas_period_ms) == 0) { receiving = true; receive_cnt = 0; measure_tti = tti; srslte_ringbuffer_reset(&ring_buffer); } if (receiving) { if (srslte_ringbuffer_write(&ring_buffer, data, nsamples * sizeof(cf_t)) < (int)(nsamples * sizeof(cf_t))) { Warning("Error writting to ringbuffer\n"); receiving = false; } else { receive_cnt++; if (receive_cnt == common->args->intra_freq_meas_len_ms) { tti_sync.increase(); receiving = false; } } } } } void intra_measure::run_thread() { while (running) { if (running) { tti_sync.wait(); } if (running) { // Read data from buffer and find cells in it srslte_ringbuffer_read( &ring_buffer, search_buffer, common->args->intra_freq_meas_len_ms * current_sflen * sizeof(cf_t)); int found_cells = scell.find_cells( search_buffer, common->rx_gain_offset, primary_cell, common->args->intra_freq_meas_len_ms, info); receiving = false; // Look for other cells not found automatically // Using Cell Reference signal synchronization for all known active PCI for (auto q : active_pci) { srslte_cell_t cell = primary_cell; cell.id = q; srslte_refsignal_dl_sync_set_cell(&refsignal_dl_sync, cell); srslte_refsignal_dl_sync_run( &refsignal_dl_sync, search_buffer, common->args->intra_freq_meas_len_ms * current_sflen); if (refsignal_dl_sync.found) { Info("INTRA: Found neighbour cell: PCI=%03d, RSRP=%5.1f dBm, RSRQ=%5.1f, peak_idx=%5d, CFO=%+.1fHz\n", cell.id, refsignal_dl_sync.rsrp_dBfs, refsignal_dl_sync.rsrq_dB, refsignal_dl_sync.peak_index, refsignal_dl_sync.cfo_Hz); bool found = false; float weakest_rsrp_value = +INFINITY; uint32_t weakest_rsrp_index = 0; // Try to find PCI in info list for (int i = 0; i < found_cells && !found; i++) { // Finds cell, update if (info[i].pci == cell.id) { info[i].rsrp = refsignal_dl_sync.rsrp_dBfs; info[i].rsrq = refsignal_dl_sync.rsrq_dB; info[i].offset = refsignal_dl_sync.peak_index; found = true; } else if (weakest_rsrp_value > info[i].rsrp) { // Update weakest weakest_rsrp_value = info[i].rsrp; weakest_rsrp_index = i; } } if (!found) { // If number of cells exceeds if (found_cells >= scell_recv::MAX_CELLS) { // overwrite weakest cell if stronger if (refsignal_dl_sync.rsrp_dBfs > weakest_rsrp_value) { info[weakest_rsrp_index].pci = cell.id; info[weakest_rsrp_index].rsrp = refsignal_dl_sync.rsrp_dBfs; info[weakest_rsrp_index].rsrq = refsignal_dl_sync.rsrq_dB; info[weakest_rsrp_index].offset = refsignal_dl_sync.peak_index; } else { // Ignore measurement } } else { // Otherwise append cell info[found_cells].pci = cell.id; info[found_cells].rsrp = refsignal_dl_sync.rsrp_dBfs; info[found_cells].rsrq = refsignal_dl_sync.rsrq_dB; info[found_cells].offset = refsignal_dl_sync.peak_index; found_cells++; } } } } // Send measurements to RRC for (int i = 0; i < found_cells; i++) { rrc->new_phy_meas(info[i].rsrp, info[i].rsrq, measure_tti, current_earfcn, info[i].pci); } } } } } // namespace scell } // namespace srsue