/** * Copyright 2013-2022 Software Radio Systems Limited * * This file is part of srsRAN. * * srsRAN 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. * * srsRAN 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/. * */ #ifndef SRSRAN_TPC_H #define SRSRAN_TPC_H #include "srsran/adt/accumulators.h" #include "srsran/common/common.h" #include "srsran/srslog/srslog.h" namespace srsenb { /** * Class to handle TPC Commands sent to the UE. * The TPC value sent to the UE in each DCI is a result of: * - the difference between the target SINR and the windowed average of past UE UL SNR estimates * - subtracted by the sum of TPC values sent since the last PHR */ class tpc { static constexpr int undefined_phr = std::numeric_limits::max(); static constexpr float null_snr = std::numeric_limits::max(); static constexpr uint32_t nof_ul_ch_code = 2; public: static constexpr uint32_t PUSCH_CODE = 0, PUCCH_CODE = 1; static constexpr int PHR_NEG_NOF_PRB = 1; explicit tpc(uint16_t rnti_, uint32_t cell_nof_prb, float target_pucch_snr_dB_ = -1.0, float target_pusch_sn_dB_ = -1.0, bool phr_handling_flag_ = false, int min_phr_thres_ = 0, uint32_t min_tpc_tti_interval_ = 1, float ul_snr_avg_alpha = 0.05, int init_ul_snr_value = 5) : rnti(rnti_), nof_prb(cell_nof_prb), target_pucch_snr_dB(target_pucch_snr_dB_), target_pusch_snr_dB(target_pusch_sn_dB_), min_phr_thres(min_phr_thres_), snr_estim_list( {ul_ch_snr_estim{ul_snr_avg_alpha, init_ul_snr_value}, ul_ch_snr_estim{ul_snr_avg_alpha, init_ul_snr_value}}), phr_handling_flag(phr_handling_flag_), max_prbs_cached(nof_prb), min_tpc_tti_interval(min_tpc_tti_interval_), logger(srslog::fetch_basic_logger("MAC")) {} void set_cfg(float target_pusch_snr_dB_, float target_pucch_snr_dB_) { target_pucch_snr_dB = target_pucch_snr_dB_; target_pusch_snr_dB = target_pusch_snr_dB_; } void set_snr(float snr, uint32_t ul_ch_code) { static const float MIN_UL_SNR = -4.0f; if (snr < MIN_UL_SNR) { // Assume signal was not sent return; } if (ul_ch_code < nof_ul_ch_code) { snr_estim_list[ul_ch_code].pending_snr = snr; } } void set_phr(int phr_, uint32_t grant_nof_prbs) { last_phr = phr_; for (auto& ch_snr : snr_estim_list) { ch_snr.acc_tpc_phr_values = 0; } // compute and cache the max nof UL PRBs that avoids overflowing PHR if (phr_handling_flag) { max_prbs_cached = PHR_NEG_NOF_PRB; int phr_x_prb = std::roundf(last_phr + 10.0F * log10f(grant_nof_prbs)); // get what the PHR would be if Nprb=1 for (int n = nof_prb; n > PHR_NEG_NOF_PRB; --n) { if (phr_x_prb >= 10 * log10f(n) + min_phr_thres) { max_prbs_cached = n; break; } } logger.info("SCHED: rnti=0x%x received PHR=%d for UL Nprb=%d. Max UL Nprb is now=%d", rnti, phr_, grant_nof_prbs, max_prbs_cached); } } void new_tti() { tti_count++; for (size_t chidx = 0; chidx < nof_ul_ch_code; ++chidx) { float target_snr_dB = chidx == PUSCH_CODE ? target_pusch_snr_dB : target_pucch_snr_dB; auto& ch_snr = snr_estim_list[chidx]; // Enqueue pending UL Channel SNR measurement if (ch_snr.pending_snr == null_snr) { ch_snr.last_snr_sample_count++; ch_snr.acc_tpc_values += ch_snr.win_tpc_values.oldest(); ch_snr.acc_tpc_phr_values += ch_snr.win_tpc_values.oldest(); } else { ch_snr.acc_tpc_values = 0; ch_snr.snr_avg.push(ch_snr.pending_snr, ch_snr.last_snr_sample_count); ch_snr.last_snr_sample = ch_snr.pending_snr; ch_snr.last_snr_sample_count = 1; } ch_snr.pending_snr = null_snr; // Enqueue PUSCH/PUCCH TPC sent in last TTI (zero for both Delta_PUSCH/Delta_PUCCH=0 and TPC not sent) if (target_snr_dB >= 0) { ch_snr.win_tpc_values.push(ch_snr.pending_delta); } ch_snr.pending_delta = 0; } } /** * Called during DCI format0 encoding to set PUSCH TPC command * @remark See TS 36.213 Section 5.1.1 * @return accumulated TPC value {-1, 0, 1, 3} */ uint8_t encode_pusch_tpc() { return encode_tpc(PUSCH_CODE); } /** * Called during DCI format1/2A/A encoding to set PUCCH TPC command * @remark See TS 36.213 Section 5.1.2 * @return accumulated TPC value {-1, 0, 1, 3} */ uint8_t encode_pucch_tpc() { return encode_tpc(PUCCH_CODE); } uint32_t max_ul_prbs() const { return max_prbs_cached; } float get_ul_snr_estim(uint32_t ul_ch_code = PUSCH_CODE) const { return snr_estim_list[ul_ch_code].snr_avg.value(); } private: uint8_t encode_tpc_delta(int8_t delta) { switch (delta) { case -1: return 0; case 0: return 1; case 1: return 2; case 3: return 3; default: logger.warning("Invalid TPC delta value=%d", delta); return 1; } } uint8_t encode_tpc(uint32_t cc) { auto& ch_snr = snr_estim_list[cc]; assert(ch_snr.pending_delta == 0); // ensure called once per {cc,tti} float target_snr_dB = cc == PUSCH_CODE ? target_pusch_snr_dB : target_pucch_snr_dB; if (target_snr_dB < 0) { // undefined target SINR case return encode_tpc_delta(0); } // limitation of TPC based on PHR int max_delta = 3; int eff_phr = last_phr; if (cc == PUSCH_CODE and last_phr != undefined_phr) { eff_phr -= ch_snr.win_tpc_values.value() + ch_snr.acc_tpc_phr_values; max_delta = std::min(max_delta, eff_phr); } float snr = ch_snr.last_snr_sample; // In case of periodicity of TPCs is > 1 tti, use average SNR to compute SNR diff if (min_tpc_tti_interval > 1) { ch_snr.tpc_snr_avg.push(snr); if ((tti_count - ch_snr.last_tpc_tti_count) < min_tpc_tti_interval) { // more time required before sending next TPC return encode_tpc_delta(0); } snr = ch_snr.tpc_snr_avg.value(); } float diff = target_snr_dB - snr; diff -= ch_snr.win_tpc_values.value() + ch_snr.acc_tpc_values; ch_snr.pending_delta = std::max(std::min((int)floorf(diff), max_delta), -1); ch_snr.pending_delta = (ch_snr.pending_delta == 2) ? 1 : ch_snr.pending_delta; if (ch_snr.pending_delta != 0) { if (min_tpc_tti_interval > 1) { ch_snr.last_tpc_tti_count = tti_count; ch_snr.tpc_snr_avg.reset(); } logger.debug("TPC: rnti=0x%x, %s command=%d, last SNR=%d, SNR average=%f, diff_acc=%f, eff_phr=%d", rnti, cc == PUSCH_CODE ? "PUSCH" : "PUCCH", encode_tpc_delta(ch_snr.pending_delta), ch_snr.last_snr_sample, ch_snr.snr_avg.value(), diff, eff_phr); } return encode_tpc_delta(ch_snr.pending_delta); } uint16_t rnti; uint32_t nof_prb; uint32_t min_tpc_tti_interval = 1; float target_pucch_snr_dB, target_pusch_snr_dB; int min_phr_thres; bool phr_handling_flag; srslog::basic_logger& logger; // state uint32_t tti_count = 0; // PHR-related variables int last_phr = undefined_phr; uint32_t max_prbs_cached = 100; // SNR estimation struct ul_ch_snr_estim { // pending new snr sample float pending_snr = srsran::null_sliding_average::null_value(); // SNR average estimation with irregular sample spacing uint32_t last_snr_sample_count = 1; // jump in spacing srsran::exp_average_irreg_sampling snr_avg; int last_snr_sample; // Accumulation of past TPC commands srsran::sliding_sum win_tpc_values; int acc_tpc_values = 0; int acc_tpc_phr_values = 0; int8_t pending_delta = 0; uint32_t last_tpc_tti_count = 0; srsran::rolling_average tpc_snr_avg; // average of SNRs since last TPC != 1 explicit ul_ch_snr_estim(float exp_avg_alpha, int initial_snr) : snr_avg(exp_avg_alpha, initial_snr), win_tpc_values(FDD_HARQ_DELAY_UL_MS + FDD_HARQ_DELAY_DL_MS), last_snr_sample(initial_snr) {} }; std::array snr_estim_list; }; } // namespace srsenb #endif // SRSRAN_TPC_H