mirror of https://github.com/PentHertz/srsLTE.git
779 lines
23 KiB
C
779 lines
23 KiB
C
/**
|
|
* 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/.
|
|
*
|
|
*/
|
|
|
|
#include "srsran/srsran.h"
|
|
#include <assert.h>
|
|
#include <math.h>
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <strings.h>
|
|
|
|
#include "srsran/phy/common/phy_common.h"
|
|
#include "srsran/phy/phch/cqi.h"
|
|
#include "srsran/phy/utils/bit.h"
|
|
#include "srsran/phy/utils/debug.h"
|
|
#include "srsran/phy/utils/vector.h"
|
|
|
|
/*******************************************************
|
|
* PACKING FUNCTIONS *
|
|
*******************************************************/
|
|
static int cqi_hl_subband_pack(srsran_cqi_cfg_t* cfg, srsran_cqi_hl_subband_t* msg, uint8_t* buff)
|
|
{
|
|
uint8_t* body_ptr = buff;
|
|
uint32_t bit_count = 0;
|
|
|
|
/* Unpack codeword 0, common for 3GPP 36.212 Tables 5.2.2.6.2-1 and 5.2.2.6.2-2 */
|
|
srsran_bit_unpack(msg->wideband_cqi_cw0, &body_ptr, 4);
|
|
srsran_bit_unpack(msg->subband_diff_cqi_cw0, &body_ptr, 2 * cfg->N);
|
|
bit_count += 4 + 2 * cfg->N;
|
|
|
|
/* Unpack codeword 1, 3GPP 36.212 Table 5.2.2.6.2-2 */
|
|
if (cfg->rank_is_not_one) {
|
|
srsran_bit_unpack(msg->wideband_cqi_cw1, &body_ptr, 4);
|
|
srsran_bit_unpack(msg->subband_diff_cqi_cw1, &body_ptr, 2 * cfg->N);
|
|
bit_count += 4 + 2 * cfg->N;
|
|
}
|
|
|
|
/* If PMI is present, unpack it */
|
|
if (cfg->pmi_present) {
|
|
if (cfg->four_antenna_ports) {
|
|
srsran_bit_unpack(msg->pmi, &body_ptr, 4);
|
|
bit_count += 4;
|
|
} else {
|
|
if (cfg->rank_is_not_one) {
|
|
srsran_bit_unpack(msg->pmi, &body_ptr, 1);
|
|
bit_count += 1;
|
|
} else {
|
|
srsran_bit_unpack(msg->pmi, &body_ptr, 2);
|
|
bit_count += 2;
|
|
}
|
|
}
|
|
}
|
|
|
|
return bit_count;
|
|
}
|
|
|
|
static int cqi_ue_subband_pack(srsran_cqi_cfg_t* cfg, srsran_cqi_ue_diff_subband_t* msg, uint8_t* buff)
|
|
{
|
|
uint8_t* body_ptr = buff;
|
|
srsran_bit_unpack(msg->wideband_cqi, &body_ptr, 4);
|
|
srsran_bit_unpack(msg->subband_diff_cqi, &body_ptr, 2);
|
|
srsran_bit_unpack(msg->subband_diff_cqi, &body_ptr, cfg->L);
|
|
|
|
return 4 + 2 + cfg->L;
|
|
}
|
|
|
|
/* Pack CQI following 3GPP TS 36.212 Tables 5.2.3.3.1-1 and 5.2.3.3.1-2 */
|
|
static int cqi_format2_wideband_pack(srsran_cqi_cfg_t* cfg, srsran_cqi_format2_wideband_t* msg, uint8_t* buff)
|
|
{
|
|
uint8_t* body_ptr = buff;
|
|
|
|
srsran_bit_unpack(msg->wideband_cqi, &body_ptr, 4);
|
|
|
|
if (cfg->pmi_present) {
|
|
if (cfg->four_antenna_ports) {
|
|
if (cfg->rank_is_not_one) {
|
|
srsran_bit_unpack(msg->spatial_diff_cqi, &body_ptr, 3);
|
|
}
|
|
srsran_bit_unpack(msg->pmi, &body_ptr, 4);
|
|
} else {
|
|
if (cfg->rank_is_not_one) {
|
|
srsran_bit_unpack(msg->spatial_diff_cqi, &body_ptr, 3);
|
|
srsran_bit_unpack(msg->pmi, &body_ptr, 1);
|
|
} else {
|
|
srsran_bit_unpack(msg->pmi, &body_ptr, 2);
|
|
}
|
|
}
|
|
}
|
|
|
|
return (int)(body_ptr - buff);
|
|
}
|
|
|
|
static int cqi_format2_subband_pack(srsran_cqi_cfg_t* cfg, srsran_cqi_ue_subband_t* msg, uint8_t* buff)
|
|
{
|
|
uint8_t* body_ptr = buff;
|
|
srsran_bit_unpack(msg->subband_cqi, &body_ptr, 4);
|
|
srsran_bit_unpack(msg->subband_label, &body_ptr, cfg->subband_label_2_bits ? 2 : 1);
|
|
return 4 + ((cfg->subband_label_2_bits) ? 2 : 1);
|
|
}
|
|
|
|
int srsran_cqi_value_pack(srsran_cqi_cfg_t* cfg, srsran_cqi_value_t* value, uint8_t buff[SRSRAN_CQI_MAX_BITS])
|
|
{
|
|
switch (cfg->type) {
|
|
case SRSRAN_CQI_TYPE_WIDEBAND:
|
|
return cqi_format2_wideband_pack(cfg, &value->wideband, buff);
|
|
case SRSRAN_CQI_TYPE_SUBBAND_UE:
|
|
return cqi_format2_subband_pack(cfg, &value->subband_ue, buff);
|
|
case SRSRAN_CQI_TYPE_SUBBAND_UE_DIFF:
|
|
return cqi_ue_subband_pack(cfg, &value->subband_ue_diff, buff);
|
|
case SRSRAN_CQI_TYPE_SUBBAND_HL:
|
|
return cqi_hl_subband_pack(cfg, &value->subband_hl, buff);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/*******************************************************
|
|
* UNPACKING FUNCTIONS *
|
|
*******************************************************/
|
|
|
|
int cqi_hl_subband_unpack(srsran_cqi_cfg_t* cfg, uint8_t* buff, srsran_cqi_hl_subband_t* msg)
|
|
{
|
|
uint8_t* body_ptr = buff;
|
|
uint32_t bit_count = 0;
|
|
|
|
msg->wideband_cqi_cw0 = (uint8_t)srsran_bit_pack(&body_ptr, 4);
|
|
msg->subband_diff_cqi_cw0 = srsran_bit_pack(&body_ptr, 2 * cfg->N);
|
|
bit_count += 4 + 2 * cfg->N;
|
|
|
|
/* Unpack codeword 1, 3GPP 36.212 Table 5.2.2.6.2-2 */
|
|
if (cfg->rank_is_not_one) {
|
|
msg->wideband_cqi_cw1 = (uint8_t)srsran_bit_pack(&body_ptr, 4);
|
|
msg->subband_diff_cqi_cw1 = srsran_bit_pack(&body_ptr, 2 * cfg->N);
|
|
bit_count += 4 + 2 * cfg->N;
|
|
}
|
|
|
|
/* If PMI is present, unpack it */
|
|
if (cfg->pmi_present) {
|
|
if (cfg->four_antenna_ports) {
|
|
msg->pmi = (uint8_t)srsran_bit_pack(&body_ptr, 4);
|
|
bit_count += 4;
|
|
} else {
|
|
if (cfg->rank_is_not_one) {
|
|
msg->pmi = (uint8_t)srsran_bit_pack(&body_ptr, 1);
|
|
bit_count += 1;
|
|
} else {
|
|
msg->pmi = (uint8_t)srsran_bit_pack(&body_ptr, 2);
|
|
bit_count += 2;
|
|
}
|
|
}
|
|
}
|
|
|
|
return bit_count;
|
|
}
|
|
|
|
int cqi_ue_subband_unpack(srsran_cqi_cfg_t* cfg, uint8_t* buff, srsran_cqi_ue_diff_subband_t* msg)
|
|
{
|
|
uint8_t* body_ptr = buff;
|
|
msg->wideband_cqi = srsran_bit_pack(&body_ptr, 4);
|
|
msg->subband_diff_cqi = srsran_bit_pack(&body_ptr, 2);
|
|
msg->subband_diff_cqi = srsran_bit_pack(&body_ptr, cfg->L);
|
|
|
|
return 4 + 2 + cfg->L;
|
|
}
|
|
|
|
/* Unpack CQI following 3GPP TS 36.212 Tables 5.2.3.3.1-1 and 5.2.3.3.1-2 */
|
|
static int cqi_format2_wideband_unpack(srsran_cqi_cfg_t* cfg, uint8_t* buff, srsran_cqi_format2_wideband_t* msg)
|
|
{
|
|
uint8_t* body_ptr = buff;
|
|
msg->wideband_cqi = (uint8_t)srsran_bit_pack(&body_ptr, 4);
|
|
|
|
if (cfg->pmi_present) {
|
|
if (cfg->four_antenna_ports) {
|
|
if (cfg->rank_is_not_one) {
|
|
msg->spatial_diff_cqi = (uint8_t)srsran_bit_pack(&body_ptr, 3);
|
|
}
|
|
msg->pmi = (uint8_t)srsran_bit_pack(&body_ptr, 4);
|
|
} else {
|
|
if (cfg->rank_is_not_one) {
|
|
msg->spatial_diff_cqi = (uint8_t)srsran_bit_pack(&body_ptr, 3);
|
|
msg->pmi = (uint8_t)srsran_bit_pack(&body_ptr, 1);
|
|
} else {
|
|
msg->pmi = (uint8_t)srsran_bit_pack(&body_ptr, 2);
|
|
}
|
|
}
|
|
}
|
|
return 4;
|
|
}
|
|
|
|
static int cqi_format2_subband_unpack(srsran_cqi_cfg_t* cfg, uint8_t* buff, srsran_cqi_ue_subband_t* msg)
|
|
{
|
|
uint8_t* body_ptr = buff;
|
|
msg->subband_cqi = srsran_bit_pack(&body_ptr, 4);
|
|
msg->subband_label = srsran_bit_pack(&body_ptr, cfg->subband_label_2_bits ? 2 : 1);
|
|
return 4 + ((cfg->subband_label_2_bits) ? 2 : 1);
|
|
}
|
|
|
|
int srsran_cqi_value_unpack(srsran_cqi_cfg_t* cfg, uint8_t buff[SRSRAN_CQI_MAX_BITS], srsran_cqi_value_t* value)
|
|
{
|
|
switch (cfg->type) {
|
|
case SRSRAN_CQI_TYPE_WIDEBAND:
|
|
return cqi_format2_wideband_unpack(cfg, buff, &value->wideband);
|
|
case SRSRAN_CQI_TYPE_SUBBAND_UE:
|
|
return cqi_format2_subband_unpack(cfg, buff, &value->subband_ue);
|
|
case SRSRAN_CQI_TYPE_SUBBAND_UE_DIFF:
|
|
return cqi_ue_subband_unpack(cfg, buff, &value->subband_ue_diff);
|
|
case SRSRAN_CQI_TYPE_SUBBAND_HL:
|
|
return cqi_hl_subband_unpack(cfg, buff, &value->subband_hl);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/*******************************************************
|
|
* TO STRING FUNCTIONS *
|
|
*******************************************************/
|
|
|
|
static int
|
|
cqi_format2_wideband_tostring(srsran_cqi_cfg_t* cfg, srsran_cqi_format2_wideband_t* msg, char* buff, uint32_t buff_len)
|
|
{
|
|
int n = 0;
|
|
|
|
n += snprintf(buff + n, buff_len - n, ", cqi=%d", msg->wideband_cqi);
|
|
|
|
if (cfg->pmi_present) {
|
|
if (cfg->rank_is_not_one) {
|
|
n += snprintf(buff + n, buff_len - n, ", diff_cqi=%d", msg->spatial_diff_cqi);
|
|
}
|
|
n += snprintf(buff + n, buff_len - n, ", pmi=%d", msg->pmi);
|
|
}
|
|
|
|
return n;
|
|
}
|
|
|
|
static int
|
|
cqi_format2_subband_tostring(srsran_cqi_cfg_t* cfg, srsran_cqi_ue_subband_t* msg, char* buff, uint32_t buff_len)
|
|
{
|
|
int n = 0;
|
|
|
|
n += snprintf(buff + n, buff_len - n, ", cqi=%d", msg->subband_cqi);
|
|
n += snprintf(buff + n, buff_len - n, ", label=%d", msg->subband_label);
|
|
n += snprintf(buff + n, buff_len - n, ", sb_idx=%d", cfg->sb_idx);
|
|
return n;
|
|
}
|
|
|
|
static int
|
|
cqi_ue_subband_tostring(srsran_cqi_cfg_t* cfg, srsran_cqi_ue_diff_subband_t* msg, char* buff, uint32_t buff_len)
|
|
{
|
|
int n = 0;
|
|
|
|
n += snprintf(buff + n, buff_len - n, ", cqi=%d", msg->wideband_cqi);
|
|
n += snprintf(buff + n, buff_len - n, ", diff_cqi=%d", msg->subband_diff_cqi);
|
|
n += snprintf(buff + n, buff_len - n, ", L=%d", cfg->L);
|
|
|
|
return n;
|
|
}
|
|
|
|
static int cqi_hl_subband_tostring(srsran_cqi_cfg_t* cfg, srsran_cqi_hl_subband_t* msg, char* buff, uint32_t buff_len)
|
|
{
|
|
int n = 0;
|
|
|
|
n += snprintf(buff + n, buff_len - n, ", cqi=%d", msg->wideband_cqi_cw0);
|
|
n += snprintf(buff + n, buff_len - n, ", diff=%d", msg->subband_diff_cqi_cw0);
|
|
|
|
if (cfg->rank_is_not_one) {
|
|
n += snprintf(buff + n, buff_len - n, ", cqi1=%d", msg->wideband_cqi_cw1);
|
|
n += snprintf(buff + n, buff_len - n, ", diff1=%d", msg->subband_diff_cqi_cw1);
|
|
}
|
|
|
|
if (cfg->pmi_present) {
|
|
n += snprintf(buff + n, buff_len - n, ", pmi=%d", msg->pmi);
|
|
}
|
|
|
|
n += snprintf(buff + n, buff_len - n, ", N=%d", cfg->N);
|
|
|
|
return n;
|
|
}
|
|
|
|
int srsran_cqi_value_tostring(srsran_cqi_cfg_t* cfg, srsran_cqi_value_t* value, char* buff, uint32_t buff_len)
|
|
{
|
|
int ret = -1;
|
|
|
|
switch (cfg->type) {
|
|
case SRSRAN_CQI_TYPE_WIDEBAND:
|
|
ret = cqi_format2_wideband_tostring(cfg, &value->wideband, buff, buff_len);
|
|
break;
|
|
case SRSRAN_CQI_TYPE_SUBBAND_UE:
|
|
ret = cqi_format2_subband_tostring(cfg, &value->subband_ue, buff, buff_len);
|
|
break;
|
|
case SRSRAN_CQI_TYPE_SUBBAND_UE_DIFF:
|
|
ret = cqi_ue_subband_tostring(cfg, &value->subband_ue_diff, buff, buff_len);
|
|
break;
|
|
case SRSRAN_CQI_TYPE_SUBBAND_HL:
|
|
ret = cqi_hl_subband_tostring(cfg, &value->subband_hl, buff, buff_len);
|
|
break;
|
|
default:
|
|
/* Do nothing */;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int srsran_cqi_size(srsran_cqi_cfg_t* cfg)
|
|
{
|
|
int size = 0;
|
|
|
|
if (!cfg->data_enable) {
|
|
return cfg->ri_len;
|
|
}
|
|
|
|
switch (cfg->type) {
|
|
case SRSRAN_CQI_TYPE_WIDEBAND:
|
|
/* Compute size according to 3GPP TS 36.212 Tables 5.2.3.3.1-1 and 5.2.3.3.1-2 */
|
|
size = 4;
|
|
if (cfg->pmi_present) {
|
|
if (cfg->four_antenna_ports) {
|
|
if (cfg->rank_is_not_one) {
|
|
size += 3; // Differential
|
|
} else {
|
|
size += 0; // Differential
|
|
}
|
|
size += 4; // PMI
|
|
} else {
|
|
if (cfg->rank_is_not_one) {
|
|
size += 3; // Differential
|
|
size += 1; // PMI
|
|
} else {
|
|
size += 0; // Differential
|
|
size += 2; // PMI
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case SRSRAN_CQI_TYPE_SUBBAND_UE:
|
|
size = 4 + ((cfg->subband_label_2_bits) ? 2 : 1);
|
|
break;
|
|
case SRSRAN_CQI_TYPE_SUBBAND_UE_DIFF:
|
|
size = 4 + 2 + cfg->L;
|
|
break;
|
|
case SRSRAN_CQI_TYPE_SUBBAND_HL:
|
|
/* First codeword */
|
|
size += 4 + 2 * cfg->N;
|
|
|
|
/* Add Second codeword if required */
|
|
if (cfg->rank_is_not_one && cfg->pmi_present) {
|
|
size += 4 + 2 * cfg->N;
|
|
}
|
|
|
|
/* Add PMI if required*/
|
|
if (cfg->pmi_present) {
|
|
if (cfg->four_antenna_ports) {
|
|
size += 4;
|
|
} else {
|
|
if (cfg->rank_is_not_one) {
|
|
size += 1;
|
|
} else {
|
|
size += 2;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
size = SRSRAN_ERROR;
|
|
}
|
|
return size;
|
|
}
|
|
|
|
static bool cqi_get_N_fdd(uint32_t I_cqi_pmi, uint32_t* N_p, uint32_t* N_offset)
|
|
{
|
|
// Acccording to 3GPP 36.213 R10 Table 7.2.2-1A: Mapping of I CQI / PMI to N pd and N OFFSET , CQI for FDD
|
|
if (I_cqi_pmi <= 1) {
|
|
*N_p = 2;
|
|
*N_offset = I_cqi_pmi;
|
|
} else if (I_cqi_pmi <= 6) {
|
|
*N_p = 5;
|
|
*N_offset = I_cqi_pmi - 2;
|
|
} else if (I_cqi_pmi <= 16) {
|
|
*N_p = 10;
|
|
*N_offset = I_cqi_pmi - 7;
|
|
} else if (I_cqi_pmi <= 36) {
|
|
*N_p = 20;
|
|
*N_offset = I_cqi_pmi - 17;
|
|
} else if (I_cqi_pmi <= 76) {
|
|
*N_p = 40;
|
|
*N_offset = I_cqi_pmi - 37;
|
|
} else if (I_cqi_pmi <= 156) {
|
|
*N_p = 80;
|
|
*N_offset = I_cqi_pmi - 77;
|
|
} else if (I_cqi_pmi <= 316) {
|
|
*N_p = 160;
|
|
*N_offset = I_cqi_pmi - 157;
|
|
} else if (I_cqi_pmi == 317) {
|
|
return false;
|
|
} else if (I_cqi_pmi <= 349) {
|
|
*N_p = 32;
|
|
*N_offset = I_cqi_pmi - 318;
|
|
} else if (I_cqi_pmi <= 413) {
|
|
*N_p = 64;
|
|
*N_offset = I_cqi_pmi - 350;
|
|
} else if (I_cqi_pmi <= 541) {
|
|
*N_p = 128;
|
|
*N_offset = I_cqi_pmi - 414;
|
|
} else if (I_cqi_pmi <= 1023) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool cqi_get_N_tdd(uint32_t I_cqi_pmi, uint32_t* N_p, uint32_t* N_offset)
|
|
{
|
|
// Acccording to 3GPP 36.213 R10 Table 7.2.2-1A: Mapping of I CQI / PMI to N pd and N OFFSET , CQI for TDD
|
|
if (I_cqi_pmi == 0) {
|
|
*N_p = 1;
|
|
*N_offset = I_cqi_pmi;
|
|
} else if (I_cqi_pmi <= 5) {
|
|
*N_p = 5;
|
|
*N_offset = I_cqi_pmi - 1;
|
|
} else if (I_cqi_pmi <= 15) {
|
|
*N_p = 10;
|
|
*N_offset = I_cqi_pmi - 5;
|
|
} else if (I_cqi_pmi <= 35) {
|
|
*N_p = 20;
|
|
*N_offset = I_cqi_pmi - 16;
|
|
} else if (I_cqi_pmi <= 75) {
|
|
*N_p = 40;
|
|
*N_offset = I_cqi_pmi - 36;
|
|
} else if (I_cqi_pmi <= 155) {
|
|
*N_p = 80;
|
|
*N_offset = I_cqi_pmi - 76;
|
|
} else if (I_cqi_pmi <= 315) {
|
|
*N_p = 160;
|
|
*N_offset = I_cqi_pmi - 156;
|
|
} else if (I_cqi_pmi == 1023) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool cqi_send(uint32_t I_cqi_pmi, uint32_t tti, bool is_fdd, uint32_t H)
|
|
{
|
|
uint32_t N_p = 0;
|
|
uint32_t N_offset = 0;
|
|
|
|
if (is_fdd) {
|
|
if (!cqi_get_N_fdd(I_cqi_pmi, &N_p, &N_offset)) {
|
|
return false;
|
|
}
|
|
} else {
|
|
if (!cqi_get_N_tdd(I_cqi_pmi, &N_p, &N_offset)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (N_p) {
|
|
if ((tti - N_offset) % (H * N_p) == 0) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static bool ri_send(uint32_t I_cqi_pmi, uint32_t I_ri, uint32_t tti, bool is_fdd)
|
|
{
|
|
uint32_t M_ri = 0;
|
|
int N_offset_ri = 0;
|
|
uint32_t N_p = 0;
|
|
uint32_t N_offset_p = 0;
|
|
|
|
if (is_fdd) {
|
|
if (!cqi_get_N_fdd(I_cqi_pmi, &N_p, &N_offset_p)) {
|
|
return false;
|
|
}
|
|
} else {
|
|
if (!cqi_get_N_tdd(I_cqi_pmi, &N_p, &N_offset_p)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// According to 3GPP 36.213 R10 Table 7.2.2-1B Mapping I_RI to M_RI and N_OFFSET_RI
|
|
if (I_ri <= 160) {
|
|
M_ri = 1;
|
|
N_offset_ri = -I_ri;
|
|
} else if (I_ri <= 321) {
|
|
M_ri = 2;
|
|
N_offset_ri = -(I_ri - 161);
|
|
} else if (I_ri <= 482) {
|
|
M_ri = 4;
|
|
N_offset_ri = -(I_ri - 322);
|
|
} else if (I_ri <= 643) {
|
|
M_ri = 8;
|
|
N_offset_ri = -(I_ri - 483);
|
|
} else if (I_ri <= 804) {
|
|
M_ri = 16;
|
|
N_offset_ri = -(I_ri - 644);
|
|
} else if (I_ri <= 965) {
|
|
M_ri = 32;
|
|
N_offset_ri = -(I_ri - 805);
|
|
} else {
|
|
return false;
|
|
}
|
|
|
|
if (M_ri && N_p) {
|
|
if ((tti - N_offset_p - N_offset_ri) % (N_p * M_ri) == 0) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/* Returns the subband size for higher layer-configured subband feedback,
|
|
* i.e., the number of RBs per subband as a function of the cell bandwidth
|
|
* (Table 7.2.1-3 in TS 36.213)
|
|
*/
|
|
static int cqi_hl_get_subband_size(int nof_prb)
|
|
{
|
|
if (nof_prb < 7) {
|
|
ERROR("Error: nof_prb is invalid (< 7)");
|
|
return SRSRAN_ERROR;
|
|
} else if (nof_prb <= 26) {
|
|
return 4;
|
|
} else if (nof_prb <= 63) {
|
|
return 6;
|
|
} else if (nof_prb <= 110) {
|
|
return 8;
|
|
} else {
|
|
ERROR("Error: nof_prb is invalid (> 110)");
|
|
return SRSRAN_ERROR;
|
|
}
|
|
}
|
|
|
|
/* Returns the bandwidth parts (J)
|
|
* (Table 7.2.2-2 in TS 36.213)
|
|
*/
|
|
static int cqi_hl_get_bwp_J(int nof_prb)
|
|
{
|
|
if (nof_prb < 7) {
|
|
ERROR("Error: nof_prb is not valid (< 7)");
|
|
return SRSRAN_ERROR;
|
|
} else if (nof_prb <= 10) {
|
|
return 1;
|
|
} else if (nof_prb <= 26) {
|
|
return 2;
|
|
} else if (nof_prb <= 63) {
|
|
return 3;
|
|
} else if (nof_prb <= 110) {
|
|
return 4;
|
|
} else {
|
|
ERROR("Error: nof_prb is not valid (> 110)");
|
|
return SRSRAN_ERROR;
|
|
}
|
|
}
|
|
|
|
/* Returns the number of subbands in the j-th bandwidth part
|
|
*/
|
|
static int cqi_sb_get_Nj(uint32_t j, uint32_t nof_prb)
|
|
{
|
|
// from Table 7.2.2-2 in TS 36.213
|
|
int J = cqi_hl_get_bwp_J(nof_prb);
|
|
int K = cqi_hl_get_subband_size(nof_prb);
|
|
|
|
// Catch the J and k errors, and prevent undefined modulo operations
|
|
if (J <= 0 || K <= 0) {
|
|
return 0;
|
|
}
|
|
|
|
if (J == 1) {
|
|
return (uint32_t)ceil((float)nof_prb / K);
|
|
} else {
|
|
// all bw parts have the same number of subbands except the last one
|
|
uint32_t Nj = (uint32_t)ceil((float)nof_prb / K / J);
|
|
if (j < J - 1) {
|
|
return Nj;
|
|
} else {
|
|
return Nj - 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
uint32_t srsran_cqi_get_sb_idx(uint32_t tti,
|
|
uint32_t subband_label,
|
|
const srsran_cqi_report_cfg_t* cqi_report_cfg,
|
|
const srsran_cell_t* cell)
|
|
{
|
|
uint32_t bw_part_idx = srsran_cqi_periodic_sb_bw_part_idx(cqi_report_cfg, tti, cell->nof_prb, cell->frame_type);
|
|
return subband_label + bw_part_idx * cqi_sb_get_Nj(bw_part_idx, cell->nof_prb);
|
|
}
|
|
|
|
int srsran_cqi_hl_get_L(int nof_prb)
|
|
{
|
|
int subband_size = cqi_hl_get_subband_size(nof_prb);
|
|
int bwp_J = cqi_hl_get_bwp_J(nof_prb);
|
|
if (subband_size <= 0 || bwp_J <= 0) {
|
|
ERROR("Invalid parameters");
|
|
return SRSRAN_ERROR;
|
|
}
|
|
return (int)SRSRAN_CEIL_LOG2((double)nof_prb / (subband_size * bwp_J));
|
|
}
|
|
|
|
bool srsran_cqi_periodic_ri_send(const srsran_cqi_report_cfg_t* cfg, uint32_t tti, srsran_frame_type_t frame_type)
|
|
{
|
|
return cfg->periodic_configured && cfg->ri_idx_present &&
|
|
ri_send(cfg->pmi_idx, cfg->ri_idx, tti, frame_type == SRSRAN_FDD);
|
|
}
|
|
|
|
bool srsran_cqi_periodic_send(const srsran_cqi_report_cfg_t* cfg, uint32_t tti, srsran_frame_type_t frame_type)
|
|
{
|
|
return cfg->periodic_configured && cqi_send(cfg->pmi_idx, tti, frame_type == SRSRAN_FDD, 1);
|
|
}
|
|
|
|
uint32_t cqi_sb_get_H(const srsran_cqi_report_cfg_t* cfg, uint32_t nof_prb)
|
|
{
|
|
uint32_t K = cfg->subband_wideband_ratio;
|
|
uint32_t J = cqi_hl_get_bwp_J(nof_prb);
|
|
|
|
// Catch the J errors
|
|
if (J <= 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
uint32_t H = J * K + 1;
|
|
return H;
|
|
}
|
|
|
|
bool srsran_cqi_periodic_is_subband(const srsran_cqi_report_cfg_t* cfg,
|
|
uint32_t tti,
|
|
uint32_t nof_prb,
|
|
srsran_frame_type_t frame_type)
|
|
{
|
|
uint32_t H = cqi_sb_get_H(cfg, nof_prb);
|
|
|
|
// A periodic report is subband if it's a CQI opportunity and is not wideband
|
|
return srsran_cqi_periodic_send(cfg, tti, frame_type) && !cqi_send(cfg->pmi_idx, tti, frame_type == SRSRAN_FDD, H);
|
|
}
|
|
|
|
uint32_t srsran_cqi_periodic_sb_bw_part_idx(const srsran_cqi_report_cfg_t* cfg,
|
|
uint32_t tti,
|
|
uint32_t nof_prb,
|
|
srsran_frame_type_t frame_type)
|
|
{
|
|
uint32_t H = cqi_sb_get_H(cfg, nof_prb);
|
|
uint32_t N_p = 0, N_offset = 0;
|
|
|
|
if (frame_type == SRSRAN_FDD) {
|
|
if (!cqi_get_N_fdd(cfg->pmi_idx, &N_p, &N_offset)) {
|
|
return false;
|
|
}
|
|
} else {
|
|
if (!cqi_get_N_tdd(cfg->pmi_idx, &N_p, &N_offset)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
assert(N_p != 0);
|
|
uint32_t x = ((tti - N_offset) / N_p) % H;
|
|
if (x > 0) {
|
|
int J = cqi_hl_get_bwp_J(nof_prb);
|
|
// Catch the J errors and prevent undefined modulo operation
|
|
if (J <= 0) {
|
|
return 0;
|
|
}
|
|
return (x - 1) % J;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// CQI-to-Spectral Efficiency: 36.213 Table 7.2.3-1
|
|
static float cqi_to_coderate[16] = {0,
|
|
0.1523,
|
|
0.2344,
|
|
0.3770,
|
|
0.6016,
|
|
0.8770,
|
|
1.1758,
|
|
1.4766,
|
|
1.9141,
|
|
2.4063,
|
|
2.7305,
|
|
3.3223,
|
|
3.9023,
|
|
4.5234,
|
|
5.1152,
|
|
5.5547};
|
|
|
|
// CQI-to-Spectral Efficiency: 36.213 Table 7.2.3-2
|
|
static float cqi_to_coderate_table2[16] = {0,
|
|
0.1523,
|
|
0.3770,
|
|
0.8770,
|
|
1.4766,
|
|
1.9141,
|
|
2.4063,
|
|
2.7305,
|
|
3.3223,
|
|
3.9023,
|
|
4.5234,
|
|
5.1152,
|
|
5.5547,
|
|
6.2266,
|
|
6.9141,
|
|
7.4063};
|
|
|
|
float srsran_cqi_to_coderate(uint32_t cqi, bool use_alt_table)
|
|
{
|
|
if (cqi < 16) {
|
|
return use_alt_table ? cqi_to_coderate_table2[cqi] : cqi_to_coderate[cqi];
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* SNR-to-CQI conversion, got from "Downlink SNR to CQI Mapping for Different Multiple Antenna Techniques in LTE"
|
|
* Table III.
|
|
*/
|
|
// From paper
|
|
static float cqi_to_snr_table[15] = {1.95, 4, 6, 8, 10, 11.95, 14.05, 16, 17.9, 20.9, 22.5, 24.75, 25.5, 27.30, 29};
|
|
|
|
// From experimental measurements @ 5 MHz
|
|
// static float cqi_to_snr_table[15] = { 1, 1.75, 3, 4, 5, 6, 7.5, 9, 11.5, 13.0, 15.0, 18, 20, 22.5, 26.5};
|
|
|
|
uint8_t srsran_cqi_from_snr(float snr)
|
|
{
|
|
for (int cqi = 14; cqi >= 0; cqi--) {
|
|
if (snr >= cqi_to_snr_table[cqi]) {
|
|
return (uint8_t)cqi + 1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* Returns the number of subbands to be reported in CQI measurements as
|
|
* defined in clause 7.2 in TS 36.213, i.e., the N parameter
|
|
*/
|
|
int srsran_cqi_hl_get_no_subbands(int nof_prb)
|
|
{
|
|
int hl_size = cqi_hl_get_subband_size(nof_prb);
|
|
if (hl_size > 0) {
|
|
return (int)ceil((float)nof_prb / hl_size);
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
void srsran_cqi_to_str(const uint8_t* cqi_value, int cqi_len, char* str, int str_len)
|
|
{
|
|
int i = 0;
|
|
|
|
for (i = 0; i < cqi_len && i < (str_len - 5); i++) {
|
|
str[i] = (cqi_value[i] == 0) ? (char)'0' : (char)'1';
|
|
}
|
|
|
|
if (i == (str_len - 5)) {
|
|
str[i++] = '.';
|
|
str[i++] = '.';
|
|
str[i++] = '.';
|
|
str[i++] = (cqi_value[cqi_len - 1] == 0) ? (char)'0' : (char)'1';
|
|
}
|
|
str[i] = '\0';
|
|
}
|