|
|
|
/**
|
|
|
|
* 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_EVM_H_
|
|
|
|
#define SRSRAN_EVM_H_
|
|
|
|
|
|
|
|
#include "srsran/phy/modem/mod.h"
|
|
|
|
#include "srsran/phy/modem/modem_table.h"
|
|
|
|
#include "srsran/phy/phch/ra.h"
|
|
|
|
#include "srsran/phy/utils/debug.h"
|
|
|
|
#include "srsran/phy/utils/vector.h"
|
|
|
|
|
|
|
|
/** @struct srsran_evm_buffer_t
|
|
|
|
* This structure carries the necessary temporary data required for calculating the EVM.
|
|
|
|
*
|
|
|
|
* @var max_bits maximum of bits which can support
|
|
|
|
* @var hard_bits vector that stores hard bits after decision
|
|
|
|
* @var symbols vector that stores the modulated symbols from the hard bits
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
uint32_t max_bits;
|
|
|
|
uint8_t* hard_bits;
|
|
|
|
cf_t* symbols;
|
|
|
|
} srsran_evm_buffer_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Allocates the EVM calculation buffer
|
|
|
|
* @param nof_prb that provides the maximum number of bits
|
|
|
|
* @return EVM buffer pointer
|
|
|
|
*/
|
|
|
|
static inline srsran_evm_buffer_t* srsran_evm_buffer_alloc(uint32_t nof_bits)
|
|
|
|
{
|
|
|
|
srsran_evm_buffer_t* q = SRSRAN_MEM_ALLOC(srsran_evm_buffer_t, 1);
|
|
|
|
|
|
|
|
// Check allocation result and number of PRB
|
|
|
|
if (!q || !nof_bits) {
|
|
|
|
ERROR("Malloc");
|
|
|
|
return q;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Zero memory the buffer fields
|
|
|
|
SRSRAN_MEM_ZERO(q, srsran_evm_buffer_t, 1);
|
|
|
|
|
|
|
|
// Set max number of bits
|
|
|
|
q->max_bits = nof_bits;
|
|
|
|
|
|
|
|
// Allocate hard bits
|
|
|
|
q->hard_bits = srsran_vec_u8_malloc(q->max_bits);
|
|
|
|
if (!q->hard_bits) {
|
|
|
|
ERROR("Malloc");
|
|
|
|
return q;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate symbols assuming BPSK
|
|
|
|
q->symbols = srsran_vec_cf_malloc(q->max_bits);
|
|
|
|
if (!q->symbols) {
|
|
|
|
ERROR("Malloc");
|
|
|
|
return q;
|
|
|
|
}
|
|
|
|
|
|
|
|
return q;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Allocates the EVM calculation buffer
|
|
|
|
* @param nof_prb that provides the maximum number of bits
|
|
|
|
* @return EVM buffer pointer
|
|
|
|
*/
|
|
|
|
static inline void srsran_evm_buffer_resize(srsran_evm_buffer_t* q, uint32_t new_max_bits)
|
|
|
|
{
|
|
|
|
// Assert pointer and number of PRB
|
|
|
|
if (!q || !new_max_bits) {
|
|
|
|
ERROR("Invalid inputs");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return if no resize is required
|
|
|
|
if (q->max_bits >= new_max_bits) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update with greater value
|
|
|
|
q->max_bits = new_max_bits;
|
|
|
|
|
|
|
|
// Free hard bits if it was allocated
|
|
|
|
if (q->hard_bits) {
|
|
|
|
free(q->hard_bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate hard bits again
|
|
|
|
q->hard_bits = srsran_vec_u8_malloc(q->max_bits);
|
|
|
|
if (!q->hard_bits) {
|
|
|
|
ERROR("Malloc");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Free symbols if it was allocated
|
|
|
|
if (q->symbols) {
|
|
|
|
free(q->symbols);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate symbols again
|
|
|
|
q->symbols = srsran_vec_cf_malloc(q->max_bits);
|
|
|
|
if (!q->symbols) {
|
|
|
|
ERROR("Malloc");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Template for hard decision taking
|
|
|
|
*/
|
|
|
|
#define HARD_DECISION(LLR_T, SOFTBITS, HARDBITS, NOF_SOFTBITS) \
|
|
|
|
do { \
|
|
|
|
/* Typecasts pointer type to minimum width */ \
|
|
|
|
uint8_t* ptr = (uint8_t*)SOFTBITS; \
|
|
|
|
\
|
|
|
|
/* Big endian compensation, ptr needs to point at the MSB */ \
|
|
|
|
ptr += sizeof(LLR_T) - 1UL; \
|
|
|
|
\
|
|
|
|
for (uint32_t i = 0; i < NOF_SOFTBITS / 8; i++) { \
|
|
|
|
/* Default mask */ \
|
|
|
|
uint8_t w = 0xff; \
|
|
|
|
\
|
|
|
|
/* For each soft bit, take MSB ad collocate in right position */ \
|
|
|
|
w ^= (*ptr & 0x80); \
|
|
|
|
ptr += sizeof(LLR_T); \
|
|
|
|
w ^= (*ptr & 0x80) >> 1; \
|
|
|
|
ptr += sizeof(LLR_T); \
|
|
|
|
w ^= (*ptr & 0x80) >> 2; \
|
|
|
|
ptr += sizeof(LLR_T); \
|
|
|
|
w ^= (*ptr & 0x80) >> 3; \
|
|
|
|
ptr += sizeof(LLR_T); \
|
|
|
|
w ^= (*ptr & 0x80) >> 4; \
|
|
|
|
ptr += sizeof(LLR_T); \
|
|
|
|
w ^= (*ptr & 0x80) >> 5; \
|
|
|
|
ptr += sizeof(LLR_T); \
|
|
|
|
w ^= (*ptr & 0x80) >> 6; \
|
|
|
|
ptr += sizeof(LLR_T); \
|
|
|
|
w ^= (*ptr & 0x80) >> 7; \
|
|
|
|
ptr += sizeof(LLR_T); \
|
|
|
|
HARDBITS[i] = w; \
|
|
|
|
} \
|
|
|
|
if (NOF_SOFTBITS % 8) { \
|
|
|
|
uint8_t w = 0xff; \
|
|
|
|
for (uint32_t i = 0; i < NOF_SOFTBITS % 8; i++) { \
|
|
|
|
w ^= (*ptr & 0x80) >> i; \
|
|
|
|
ptr += sizeof(LLR_T); \
|
|
|
|
} \
|
|
|
|
HARDBITS[NOF_SOFTBITS / 8] = w; \
|
|
|
|
} \
|
|
|
|
} while (false)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Template RMS EVM calculation for different LLR
|
|
|
|
*/
|
|
|
|
#define EVM_RUN_TEMPLATE(LLR_T, SUFFIX) \
|
|
|
|
static inline float srsran_evm_run_##SUFFIX(srsran_evm_buffer_t* q, \
|
|
|
|
const srsran_modem_table_t* modem_table, \
|
|
|
|
const cf_t* symbols, \
|
|
|
|
const LLR_T* llr, \
|
|
|
|
uint32_t nof_bits) \
|
|
|
|
{ \
|
|
|
|
float evm_rms = NAN; \
|
|
|
|
\
|
|
|
|
/* Return NAN if EVM buffers, modem table, LLR, symbols or bits missing*/ \
|
|
|
|
if (!q || !modem_table || !modem_table->nbits_x_symbol || !llr || !symbols || !nof_bits) { \
|
|
|
|
ERROR("Invalid inputs %p %p %p %p %d", q, modem_table, llr, symbols, nof_bits); \
|
|
|
|
return evm_rms; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* Limit number of bits to the one supported by the buffer */ \
|
|
|
|
nof_bits = SRSRAN_MIN(q->max_bits, nof_bits); \
|
|
|
|
\
|
|
|
|
/* Calculate number of symbols */ \
|
|
|
|
uint32_t nsymbols = nof_bits / modem_table->nbits_x_symbol; \
|
|
|
|
\
|
|
|
|
/* Hard decision */ \
|
|
|
|
HARD_DECISION(LLR_T, llr, q->hard_bits, nof_bits); \
|
|
|
|
\
|
|
|
|
/* Modulate */ \
|
|
|
|
srsran_mod_modulate_bytes(modem_table, q->hard_bits, q->symbols, nof_bits); \
|
|
|
|
\
|
|
|
|
/* Compute symbol difference */ \
|
|
|
|
srsran_vec_sub_ccc(symbols, q->symbols, q->symbols, nsymbols); \
|
|
|
|
\
|
|
|
|
/* Average squares */ \
|
|
|
|
float evm_pow = srsran_vec_avg_power_cf(q->symbols, nsymbols); \
|
|
|
|
\
|
|
|
|
/* Convert measure to RMS */ \
|
|
|
|
evm_rms = sqrtf(evm_pow); \
|
|
|
|
\
|
|
|
|
return evm_rms; \
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @function srsran_evm_run_f
|
|
|
|
* Calculates the Root Mean Squared EVM given a modulation table, complex symbols, floating point LLR (soft bits) and
|
|
|
|
* the number of bits.
|
|
|
|
*
|
|
|
|
* @param q is the EVM buffers, need to be preallocated
|
|
|
|
* @param modem_table points at the modulator table
|
|
|
|
* @param symbols vector carrying the modulated complex symbols
|
|
|
|
* @param llr softbits
|
|
|
|
* @param nof_bits number of bits
|
|
|
|
* @return the measured RMS EVM if no error occurs, otherwise it returns NAN
|
|
|
|
*/
|
|
|
|
EVM_RUN_TEMPLATE(float, f)
|
|
|
|
|
|
|
|
/** @function srsran_evm_run_s
|
|
|
|
* Calculates the Root Mean Squared EVM given a modulation table, complex symbols, fixed integer 16 bit LLR (soft bits)
|
|
|
|
* and the number of bits.
|
|
|
|
*
|
|
|
|
* @param q is the EVM buffers, need to be preallocated
|
|
|
|
* @param modem_table points at the modulator table
|
|
|
|
* @param symbols vector carrying the modulated complex symbols
|
|
|
|
* @param llr softbits
|
|
|
|
* @param nof_bits number of bits
|
|
|
|
* @return the measured RMS EVM if no error occurs, otherwise it returns NAN
|
|
|
|
*/
|
|
|
|
EVM_RUN_TEMPLATE(int16_t, s)
|
|
|
|
|
|
|
|
/** @function srsran_evm_run_b
|
|
|
|
* Calculates the Root Mean Squared EVM given a modulation table, complex symbols, fixed integer 8 bit LLR (soft bits)
|
|
|
|
* and the number of bits.
|
|
|
|
*
|
|
|
|
* @param q is the EVM buffers, need to be preallocated
|
|
|
|
* @param modem_table points at the modulator table
|
|
|
|
* @param symbols vector carrying the modulated complex symbols
|
|
|
|
* @param llr softbits
|
|
|
|
* @param nof_bits number of bits
|
|
|
|
* @return the measured RMS EVM if no error occurs, otherwise it returns NAN
|
|
|
|
*/
|
|
|
|
EVM_RUN_TEMPLATE(int8_t, b)
|
|
|
|
|
|
|
|
#undef EVM_RUN_TEMPLATE
|
|
|
|
#undef HARD_DECISION
|
|
|
|
|
|
|
|
static inline void srsran_evm_free(srsran_evm_buffer_t* q)
|
|
|
|
{
|
|
|
|
// Check EVM buffer object exist
|
|
|
|
if (q) {
|
|
|
|
// Check hard bits were allocated
|
|
|
|
if (q->hard_bits) {
|
|
|
|
free(q->hard_bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check symbols were allocated
|
|
|
|
if (q->symbols) {
|
|
|
|
free(q->symbols);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Free buffer object
|
|
|
|
free(q);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // SRSRAN_EVM_H_
|