|
|
|
@ -35,61 +35,110 @@
|
|
|
|
|
/* Uncomment next line for avoiding Guru DFT call */
|
|
|
|
|
//#define AVOID_GURU
|
|
|
|
|
|
|
|
|
|
int srslte_ofdm_init_(srslte_ofdm_t* q,
|
|
|
|
|
srslte_cp_t cp,
|
|
|
|
|
cf_t* in_buffer,
|
|
|
|
|
cf_t* out_buffer,
|
|
|
|
|
int symbol_sz,
|
|
|
|
|
int nof_prb,
|
|
|
|
|
srslte_dft_dir_t dir)
|
|
|
|
|
static int ofdm_init_mbsfn_(srslte_ofdm_t* q, srslte_ofdm_cfg_t* cfg, srslte_dft_dir_t dir)
|
|
|
|
|
{
|
|
|
|
|
return srslte_ofdm_init_mbsfn_(q, cp, in_buffer, out_buffer, symbol_sz, nof_prb, dir, SRSLTE_SF_NORM);
|
|
|
|
|
|
|
|
|
|
// If the symbol size is not given, calculate in function of the number of resource blocks
|
|
|
|
|
if (cfg->symbol_sz == 0) {
|
|
|
|
|
int symbol_sz_err = srslte_symbol_sz(cfg->nof_prb);
|
|
|
|
|
if (symbol_sz_err <= SRSLTE_SUCCESS) {
|
|
|
|
|
ERROR("Invalid number of PRB %d\n", cfg->nof_prb);
|
|
|
|
|
return SRSLTE_ERROR;
|
|
|
|
|
}
|
|
|
|
|
cfg->symbol_sz = (uint32_t)symbol_sz_err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int srslte_ofdm_init_mbsfn_(srslte_ofdm_t* q,
|
|
|
|
|
srslte_cp_t cp,
|
|
|
|
|
cf_t* in_buffer,
|
|
|
|
|
cf_t* out_buffer,
|
|
|
|
|
int symbol_sz,
|
|
|
|
|
int nof_prb,
|
|
|
|
|
srslte_dft_dir_t dir,
|
|
|
|
|
srslte_sf_t sf_type)
|
|
|
|
|
{
|
|
|
|
|
if (q->max_prb > 0) {
|
|
|
|
|
// The object was already initialised, update only resizing params
|
|
|
|
|
q->cfg.cp = cfg->cp;
|
|
|
|
|
q->cfg.nof_prb = cfg->nof_prb;
|
|
|
|
|
q->cfg.symbol_sz = cfg->symbol_sz;
|
|
|
|
|
} else {
|
|
|
|
|
// Otherwise copy all parameters
|
|
|
|
|
q->cfg = *cfg;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Set OFDM object attributes */
|
|
|
|
|
q->symbol_sz = (uint32_t)symbol_sz;
|
|
|
|
|
uint32_t symbol_sz = q->cfg.symbol_sz;
|
|
|
|
|
srslte_cp_t cp = q->cfg.cp;
|
|
|
|
|
srslte_sf_t sf_type = q->cfg.sf_type;
|
|
|
|
|
|
|
|
|
|
// Set OFDM object attributes
|
|
|
|
|
q->nof_symbols = SRSLTE_CP_NSYMB(cp);
|
|
|
|
|
q->nof_symbols_mbsfn = SRSLTE_CP_NSYMB(SRSLTE_CP_EXT);
|
|
|
|
|
q->cp = cp;
|
|
|
|
|
q->freq_shift = false;
|
|
|
|
|
q->nof_re = (uint32_t)nof_prb * SRSLTE_NRE;
|
|
|
|
|
q->nof_guards = ((symbol_sz - q->nof_re) / 2);
|
|
|
|
|
q->slot_sz = (uint32_t)SRSLTE_SLOT_LEN(symbol_sz);
|
|
|
|
|
q->sf_sz = (uint32_t)SRSLTE_SF_LEN(symbol_sz);
|
|
|
|
|
q->in_buffer = in_buffer;
|
|
|
|
|
q->out_buffer = out_buffer;
|
|
|
|
|
|
|
|
|
|
q->nof_re = cfg->nof_prb * SRSLTE_NRE;
|
|
|
|
|
q->nof_guards = (q->cfg.symbol_sz - q->nof_re) / 2U;
|
|
|
|
|
q->slot_sz = (uint32_t)SRSLTE_SLOT_LEN(q->cfg.symbol_sz);
|
|
|
|
|
q->sf_sz = (uint32_t)SRSLTE_SF_LEN(q->cfg.symbol_sz);
|
|
|
|
|
|
|
|
|
|
// Plan MBSFN
|
|
|
|
|
if (q->fft_plan.size) {
|
|
|
|
|
// Replan if it was initialised previously
|
|
|
|
|
if (srslte_dft_replan(&q->fft_plan, q->cfg.symbol_sz)) {
|
|
|
|
|
ERROR("Reeplaning DFT plan\n");
|
|
|
|
|
return SRSLTE_ERROR;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Create plan from zero otherwise
|
|
|
|
|
if (srslte_dft_plan_c(&q->fft_plan, symbol_sz, dir)) {
|
|
|
|
|
ERROR("Error: Creating DFT plan\n");
|
|
|
|
|
ERROR("Creating DFT plan\n");
|
|
|
|
|
return SRSLTE_ERROR;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Reallocate temporal buffer only if the new number of resource blocks is bigger than initial
|
|
|
|
|
if (q->cfg.nof_prb > q->max_prb) {
|
|
|
|
|
// Free before reallocating if allocted
|
|
|
|
|
if (q->tmp) {
|
|
|
|
|
free(q->tmp);
|
|
|
|
|
free(q->shift_buffer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef AVOID_GURU
|
|
|
|
|
q->tmp = srslte_vec_cf_malloc(symbol_sz);
|
|
|
|
|
#else
|
|
|
|
|
q->tmp = srslte_vec_cf_malloc(q->sf_sz);
|
|
|
|
|
#endif /* AVOID_GURU */
|
|
|
|
|
if (!q->tmp) {
|
|
|
|
|
perror("malloc");
|
|
|
|
|
return SRSLTE_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
q->shift_buffer = srslte_vec_cf_malloc(q->sf_sz);
|
|
|
|
|
if (!q->shift_buffer) {
|
|
|
|
|
perror("malloc");
|
|
|
|
|
return SRSLTE_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
q->window_offset_buffer = srslte_vec_cf_malloc(q->sf_sz);
|
|
|
|
|
if (!q->window_offset_buffer) {
|
|
|
|
|
perror("malloc");
|
|
|
|
|
return SRSLTE_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
q->max_prb = cfg->nof_prb;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef AVOID_GURU
|
|
|
|
|
srslte_vec_cf_zero(q->tmp, symbol_sz);
|
|
|
|
|
#else
|
|
|
|
|
uint32_t nof_prb = q->cfg.nof_prb;
|
|
|
|
|
cf_t* in_buffer = q->cfg.in_buffer;
|
|
|
|
|
cf_t* out_buffer = q->cfg.out_buffer;
|
|
|
|
|
int cp1 = SRSLTE_CP_ISNORM(cp) ? SRSLTE_CP_LEN_NORM(0, symbol_sz) : SRSLTE_CP_LEN_EXT(symbol_sz);
|
|
|
|
|
int cp2 = SRSLTE_CP_ISNORM(cp) ? SRSLTE_CP_LEN_NORM(1, symbol_sz) : SRSLTE_CP_LEN_EXT(symbol_sz);
|
|
|
|
|
|
|
|
|
|
q->tmp = srslte_vec_cf_malloc(q->sf_sz);
|
|
|
|
|
if (!q->tmp) {
|
|
|
|
|
perror("malloc");
|
|
|
|
|
return SRSLTE_ERROR;
|
|
|
|
|
// Slides DFT window a fraction of cyclic prefix, it does not apply for the inverse-DFT
|
|
|
|
|
if (isnormal(cfg->rx_window_offset)) {
|
|
|
|
|
cfg->rx_window_offset = SRSLTE_MAX(0, cfg->rx_window_offset); // Needs to be positive
|
|
|
|
|
cfg->rx_window_offset = SRSLTE_MIN(100, cfg->rx_window_offset); // Needs to be below 100
|
|
|
|
|
q->window_offset_n = (uint32_t)roundf((float)cp2 * cfg->rx_window_offset);
|
|
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < symbol_sz; i++) {
|
|
|
|
|
q->window_offset_buffer[i] = cexpf(I * M_PI * 2.0f * (float)q->window_offset_n * (float)i / (float)symbol_sz);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Zero temporal and input buffers always
|
|
|
|
|
srslte_vec_cf_zero(q->tmp, q->sf_sz);
|
|
|
|
|
|
|
|
|
|
if (dir == SRSLTE_DFT_BACKWARD) {
|
|
|
|
@ -99,52 +148,52 @@ int srslte_ofdm_init_mbsfn_(srslte_ofdm_t* q,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (int slot = 0; slot < 2; slot++) {
|
|
|
|
|
// If Guru DFT was allocated, free
|
|
|
|
|
if (q->fft_plan_sf[slot].size) {
|
|
|
|
|
srslte_dft_plan_free(&q->fft_plan_sf[slot]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create Tx/Rx plans
|
|
|
|
|
if (dir == SRSLTE_DFT_FORWARD) {
|
|
|
|
|
if (srslte_dft_plan_guru_c(&q->fft_plan_sf[slot],
|
|
|
|
|
symbol_sz,
|
|
|
|
|
dir,
|
|
|
|
|
in_buffer + cp1 + q->slot_sz * slot,
|
|
|
|
|
q->tmp + q->nof_symbols * q->symbol_sz * slot,
|
|
|
|
|
in_buffer + cp1 + q->slot_sz * slot - q->window_offset_n,
|
|
|
|
|
q->tmp,
|
|
|
|
|
1,
|
|
|
|
|
1,
|
|
|
|
|
SRSLTE_CP_NSYMB(cp),
|
|
|
|
|
symbol_sz + cp2,
|
|
|
|
|
symbol_sz)) {
|
|
|
|
|
ERROR("Error: Creating DFT plan (1)\n");
|
|
|
|
|
return -1;
|
|
|
|
|
ERROR("Creating Guru DFT plan (%d)\n", slot);
|
|
|
|
|
return SRSLTE_ERROR;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (srslte_dft_plan_guru_c(&q->fft_plan_sf[slot],
|
|
|
|
|
symbol_sz,
|
|
|
|
|
dir,
|
|
|
|
|
q->tmp + q->nof_symbols * q->symbol_sz * slot,
|
|
|
|
|
q->tmp,
|
|
|
|
|
out_buffer + cp1 + q->slot_sz * slot,
|
|
|
|
|
1,
|
|
|
|
|
1,
|
|
|
|
|
SRSLTE_CP_NSYMB(cp),
|
|
|
|
|
symbol_sz,
|
|
|
|
|
symbol_sz + cp2)) {
|
|
|
|
|
ERROR("Error: Creating DFT plan (1)\n");
|
|
|
|
|
return -1;
|
|
|
|
|
ERROR("Creating Guru inverse-DFT plan (%d)\n", slot);
|
|
|
|
|
return SRSLTE_ERROR;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
q->shift_buffer = srslte_vec_cf_malloc(SRSLTE_SF_LEN(symbol_sz));
|
|
|
|
|
if (!q->shift_buffer) {
|
|
|
|
|
perror("malloc");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
srslte_dft_plan_set_mirror(&q->fft_plan, true);
|
|
|
|
|
srslte_dft_plan_set_dc(&q->fft_plan, true);
|
|
|
|
|
|
|
|
|
|
DEBUG("Init %s symbol_sz=%d, nof_symbols=%d, cp=%s, nof_re=%d, nof_guards=%d\n",
|
|
|
|
|
dir == SRSLTE_DFT_FORWARD ? "FFT" : "iFFT",
|
|
|
|
|
q->symbol_sz,
|
|
|
|
|
q->cfg.symbol_sz,
|
|
|
|
|
q->nof_symbols,
|
|
|
|
|
q->cp == SRSLTE_CP_NORM ? "Normal" : "Extended",
|
|
|
|
|
q->cfg.cp == SRSLTE_CP_NORM ? "Normal" : "Extended",
|
|
|
|
|
q->nof_re,
|
|
|
|
|
q->nof_guards);
|
|
|
|
|
|
|
|
|
@ -156,6 +205,10 @@ int srslte_ofdm_init_mbsfn_(srslte_ofdm_t* q,
|
|
|
|
|
q->mbsfn_subframe = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set other parameters
|
|
|
|
|
srslte_ofdm_set_freq_shift(q, q->cfg.freq_shift_f);
|
|
|
|
|
srslte_dft_plan_set_norm(&q->fft_plan, q->cfg.normalize);
|
|
|
|
|
|
|
|
|
|
return SRSLTE_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -164,98 +217,6 @@ void srslte_ofdm_set_non_mbsfn_region(srslte_ofdm_t* q, uint8_t non_mbsfn_region
|
|
|
|
|
q->non_mbsfn_region = non_mbsfn_region;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int srslte_ofdm_replan_(srslte_ofdm_t* q, srslte_cp_t cp, int symbol_sz, int nof_prb)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
if (srslte_dft_replan_c(&q->fft_plan, symbol_sz)) {
|
|
|
|
|
ERROR("Error: Creating DFT plan\n");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
q->symbol_sz = (uint32_t)symbol_sz;
|
|
|
|
|
q->nof_symbols = SRSLTE_CP_NSYMB(cp);
|
|
|
|
|
q->nof_symbols_mbsfn = SRSLTE_CP_NSYMB(SRSLTE_CP_EXT);
|
|
|
|
|
q->cp = cp;
|
|
|
|
|
q->nof_re = (uint32_t)nof_prb * SRSLTE_NRE;
|
|
|
|
|
q->nof_guards = ((symbol_sz - q->nof_re) / 2);
|
|
|
|
|
q->slot_sz = (uint32_t)SRSLTE_SLOT_LEN(symbol_sz);
|
|
|
|
|
q->sf_sz = (uint32_t)SRSLTE_SF_LEN(symbol_sz);
|
|
|
|
|
|
|
|
|
|
#ifndef AVOID_GURU
|
|
|
|
|
cf_t* in_buffer = q->in_buffer;
|
|
|
|
|
cf_t* out_buffer = q->out_buffer;
|
|
|
|
|
|
|
|
|
|
int cp1 = SRSLTE_CP_ISNORM(cp) ? SRSLTE_CP_LEN_NORM(0, symbol_sz) : SRSLTE_CP_LEN_EXT(symbol_sz);
|
|
|
|
|
int cp2 = SRSLTE_CP_ISNORM(cp) ? SRSLTE_CP_LEN_NORM(1, symbol_sz) : SRSLTE_CP_LEN_EXT(symbol_sz);
|
|
|
|
|
|
|
|
|
|
srslte_dft_dir_t dir = q->fft_plan_sf[0].dir;
|
|
|
|
|
|
|
|
|
|
if (q->tmp) {
|
|
|
|
|
free(q->tmp);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
q->tmp = srslte_vec_cf_malloc(q->sf_sz);
|
|
|
|
|
if (!q->tmp) {
|
|
|
|
|
perror("malloc");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
srslte_vec_cf_zero(q->tmp, q->sf_sz);
|
|
|
|
|
|
|
|
|
|
if (dir == SRSLTE_DFT_BACKWARD) {
|
|
|
|
|
srslte_vec_cf_zero(in_buffer, SRSLTE_SF_LEN_RE(nof_prb, cp));
|
|
|
|
|
} else {
|
|
|
|
|
srslte_vec_cf_zero(in_buffer, q->sf_sz);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (int slot = 0; slot < 2; slot++) {
|
|
|
|
|
srslte_dft_plan_free(&q->fft_plan_sf[slot]);
|
|
|
|
|
|
|
|
|
|
if (dir == SRSLTE_DFT_FORWARD) {
|
|
|
|
|
if (srslte_dft_plan_guru_c(&q->fft_plan_sf[slot],
|
|
|
|
|
symbol_sz,
|
|
|
|
|
dir,
|
|
|
|
|
in_buffer + cp1 + q->slot_sz * slot,
|
|
|
|
|
q->tmp + q->nof_symbols * q->symbol_sz * slot,
|
|
|
|
|
1,
|
|
|
|
|
1,
|
|
|
|
|
SRSLTE_CP_NSYMB(cp),
|
|
|
|
|
symbol_sz + cp2,
|
|
|
|
|
symbol_sz)) {
|
|
|
|
|
ERROR("Error: Creating DFT plan (1)\n");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (srslte_dft_plan_guru_c(&q->fft_plan_sf[slot],
|
|
|
|
|
symbol_sz,
|
|
|
|
|
dir,
|
|
|
|
|
q->tmp + q->nof_symbols * q->symbol_sz * slot,
|
|
|
|
|
out_buffer + cp1 + q->slot_sz * slot,
|
|
|
|
|
1,
|
|
|
|
|
1,
|
|
|
|
|
SRSLTE_CP_NSYMB(cp),
|
|
|
|
|
symbol_sz,
|
|
|
|
|
symbol_sz + cp2)) {
|
|
|
|
|
ERROR("Error: Creating DFT plan (1)\n");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif /* AVOID_GURU */
|
|
|
|
|
|
|
|
|
|
if (q->freq_shift) {
|
|
|
|
|
srslte_ofdm_set_freq_shift(q, q->freq_shift_f);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEBUG("Replan symbol_sz=%d, nof_symbols=%d, cp=%s, nof_re=%d, nof_guards=%d\n",
|
|
|
|
|
q->symbol_sz,
|
|
|
|
|
q->nof_symbols,
|
|
|
|
|
q->cp == SRSLTE_CP_NORM ? "Normal" : "Extended",
|
|
|
|
|
q->nof_re,
|
|
|
|
|
q->nof_guards);
|
|
|
|
|
|
|
|
|
|
return SRSLTE_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void srslte_ofdm_free_(srslte_ofdm_t* q)
|
|
|
|
|
{
|
|
|
|
|
srslte_dft_plan_free(&q->fft_plan);
|
|
|
|
@ -274,154 +235,126 @@ void srslte_ofdm_free_(srslte_ofdm_t* q)
|
|
|
|
|
if (q->shift_buffer) {
|
|
|
|
|
free(q->shift_buffer);
|
|
|
|
|
}
|
|
|
|
|
if (q->window_offset_buffer) {
|
|
|
|
|
free(q->window_offset_buffer);
|
|
|
|
|
}
|
|
|
|
|
bzero(q, sizeof(srslte_ofdm_t));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int srslte_ofdm_rx_init(srslte_ofdm_t* q, srslte_cp_t cp, cf_t* in_buffer, cf_t* out_buffer, uint32_t max_prb)
|
|
|
|
|
{
|
|
|
|
|
int symbol_sz = srslte_symbol_sz(max_prb);
|
|
|
|
|
if (symbol_sz < 0) {
|
|
|
|
|
ERROR("Error: Invalid nof_prb=%d\n", max_prb);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
q->max_prb = max_prb;
|
|
|
|
|
return srslte_ofdm_init_(q, cp, in_buffer, out_buffer, symbol_sz, max_prb, SRSLTE_DFT_FORWARD);
|
|
|
|
|
bzero(q, sizeof(srslte_ofdm_t));
|
|
|
|
|
|
|
|
|
|
srslte_ofdm_cfg_t cfg = {};
|
|
|
|
|
cfg.cp = cp;
|
|
|
|
|
cfg.in_buffer = in_buffer;
|
|
|
|
|
cfg.out_buffer = out_buffer;
|
|
|
|
|
cfg.nof_prb = max_prb;
|
|
|
|
|
cfg.sf_type = SRSLTE_SF_NORM;
|
|
|
|
|
|
|
|
|
|
return ofdm_init_mbsfn_(q, &cfg, SRSLTE_DFT_FORWARD);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int srslte_ofdm_rx_init_mbsfn(srslte_ofdm_t* q, srslte_cp_t cp, cf_t* in_buffer, cf_t* out_buffer, uint32_t max_prb)
|
|
|
|
|
{
|
|
|
|
|
int symbol_sz = srslte_symbol_sz(max_prb);
|
|
|
|
|
if (symbol_sz < 0) {
|
|
|
|
|
ERROR("Error: Invalid nof_prb=%d\n", max_prb);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
q->max_prb = max_prb;
|
|
|
|
|
return srslte_ofdm_init_mbsfn_(q, cp, in_buffer, out_buffer, symbol_sz, max_prb, SRSLTE_DFT_FORWARD, SRSLTE_SF_MBSFN);
|
|
|
|
|
srslte_ofdm_cfg_t cfg = {};
|
|
|
|
|
cfg.cp = cp;
|
|
|
|
|
cfg.in_buffer = in_buffer;
|
|
|
|
|
cfg.out_buffer = out_buffer;
|
|
|
|
|
cfg.nof_prb = max_prb;
|
|
|
|
|
cfg.sf_type = SRSLTE_SF_MBSFN;
|
|
|
|
|
|
|
|
|
|
return ofdm_init_mbsfn_(q, &cfg, SRSLTE_DFT_FORWARD);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int srslte_ofdm_tx_init(srslte_ofdm_t* q, srslte_cp_t cp, cf_t* in_buffer, cf_t* out_buffer, uint32_t max_prb)
|
|
|
|
|
{
|
|
|
|
|
uint32_t i;
|
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
int symbol_sz = srslte_symbol_sz(max_prb);
|
|
|
|
|
if (symbol_sz < 0) {
|
|
|
|
|
ERROR("Error: Invalid nof_prb=%d\n", max_prb);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
q->max_prb = max_prb;
|
|
|
|
|
ret = srslte_ofdm_init_(q, cp, in_buffer, out_buffer, symbol_sz, max_prb, SRSLTE_DFT_BACKWARD);
|
|
|
|
|
bzero(q, sizeof(srslte_ofdm_t));
|
|
|
|
|
|
|
|
|
|
if (ret == SRSLTE_SUCCESS) {
|
|
|
|
|
srslte_dft_plan_set_norm(&q->fft_plan, false);
|
|
|
|
|
srslte_ofdm_cfg_t cfg = {};
|
|
|
|
|
cfg.cp = cp;
|
|
|
|
|
cfg.in_buffer = in_buffer;
|
|
|
|
|
cfg.out_buffer = out_buffer;
|
|
|
|
|
cfg.nof_prb = max_prb;
|
|
|
|
|
cfg.sf_type = SRSLTE_SF_NORM;
|
|
|
|
|
|
|
|
|
|
/* set now zeros at CP */
|
|
|
|
|
for (i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
srslte_vec_cf_zero(q->tmp, q->nof_guards);
|
|
|
|
|
srslte_vec_cf_zero(&q->tmp[q->nof_re + q->nof_guards], q->nof_guards);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return ret;
|
|
|
|
|
return ofdm_init_mbsfn_(q, &cfg, SRSLTE_DFT_BACKWARD);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int srslte_ofdm_tx_init_mbsfn(srslte_ofdm_t* q, srslte_cp_t cp, cf_t* in_buffer, cf_t* out_buffer, uint32_t nof_prb)
|
|
|
|
|
int srslte_ofdm_tx_init_cfg(srslte_ofdm_t* q, srslte_ofdm_cfg_t* cfg)
|
|
|
|
|
{
|
|
|
|
|
uint32_t i;
|
|
|
|
|
int ret;
|
|
|
|
|
return ofdm_init_mbsfn_(q, cfg, SRSLTE_DFT_BACKWARD);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int symbol_sz = srslte_symbol_sz(nof_prb);
|
|
|
|
|
if (symbol_sz < 0) {
|
|
|
|
|
ERROR("Error: Invalid nof_prb=%d\n", nof_prb);
|
|
|
|
|
return -1;
|
|
|
|
|
int srslte_ofdm_rx_init_cfg(srslte_ofdm_t* q, srslte_ofdm_cfg_t* cfg)
|
|
|
|
|
{
|
|
|
|
|
return ofdm_init_mbsfn_(q, cfg, SRSLTE_DFT_FORWARD);
|
|
|
|
|
}
|
|
|
|
|
q->max_prb = nof_prb;
|
|
|
|
|
ret = srslte_ofdm_init_mbsfn_(q, cp, in_buffer, out_buffer, symbol_sz, nof_prb, SRSLTE_DFT_BACKWARD, SRSLTE_SF_MBSFN);
|
|
|
|
|
|
|
|
|
|
if (ret == SRSLTE_SUCCESS) {
|
|
|
|
|
srslte_dft_plan_set_norm(&q->fft_plan, false);
|
|
|
|
|
int srslte_ofdm_tx_init_mbsfn(srslte_ofdm_t* q, srslte_cp_t cp, cf_t* in_buffer, cf_t* out_buffer, uint32_t nof_prb)
|
|
|
|
|
{
|
|
|
|
|
srslte_ofdm_cfg_t cfg = {};
|
|
|
|
|
cfg.cp = cp;
|
|
|
|
|
cfg.in_buffer = in_buffer;
|
|
|
|
|
cfg.out_buffer = out_buffer;
|
|
|
|
|
cfg.nof_prb = nof_prb;
|
|
|
|
|
cfg.sf_type = SRSLTE_SF_MBSFN;
|
|
|
|
|
|
|
|
|
|
/* set now zeros at CP */
|
|
|
|
|
for (i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
srslte_vec_cf_zero(q->tmp, q->nof_guards);
|
|
|
|
|
srslte_vec_cf_zero(&q->tmp[q->nof_re + q->nof_guards], q->nof_guards);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return ret;
|
|
|
|
|
return ofdm_init_mbsfn_(q, &cfg, SRSLTE_DFT_BACKWARD);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int srslte_ofdm_rx_set_prb(srslte_ofdm_t* q, srslte_cp_t cp, uint32_t nof_prb)
|
|
|
|
|
{
|
|
|
|
|
if (nof_prb <= q->max_prb) {
|
|
|
|
|
int symbol_sz = srslte_symbol_sz(nof_prb);
|
|
|
|
|
if (symbol_sz < 0) {
|
|
|
|
|
ERROR("Error: Invalid nof_prb=%d\n", nof_prb);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
return srslte_ofdm_replan_(q, cp, symbol_sz, nof_prb);
|
|
|
|
|
} else {
|
|
|
|
|
ERROR("OFDM (Rx): Error calling set_prb: nof_prb (%d) must be equal or lower initialized max_prb (%d)\n",
|
|
|
|
|
nof_prb,
|
|
|
|
|
q->max_prb);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
srslte_ofdm_cfg_t cfg = {};
|
|
|
|
|
cfg.cp = cp;
|
|
|
|
|
cfg.nof_prb = nof_prb;
|
|
|
|
|
return ofdm_init_mbsfn_(q, &cfg, SRSLTE_DFT_FORWARD);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int srslte_ofdm_tx_set_prb(srslte_ofdm_t* q, srslte_cp_t cp, uint32_t nof_prb)
|
|
|
|
|
{
|
|
|
|
|
uint32_t i;
|
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
if (nof_prb <= q->max_prb) {
|
|
|
|
|
int symbol_sz = srslte_symbol_sz(nof_prb);
|
|
|
|
|
if (symbol_sz < 0) {
|
|
|
|
|
ERROR("Error: Invalid nof_prb=%d\n", nof_prb);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = srslte_ofdm_replan_(q, cp, symbol_sz, nof_prb);
|
|
|
|
|
|
|
|
|
|
if (ret == SRSLTE_SUCCESS) {
|
|
|
|
|
/* set now zeros at CP */
|
|
|
|
|
for (i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
srslte_vec_cf_zero(q->tmp, q->nof_guards);
|
|
|
|
|
srslte_vec_cf_zero(&q->tmp[q->nof_re + q->nof_guards], q->nof_guards);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return ret;
|
|
|
|
|
} else {
|
|
|
|
|
ERROR("OFDM (Tx): Error calling set_prb: nof_prb (%d) must be equal or lower initialized max_prb (%d)\n",
|
|
|
|
|
nof_prb,
|
|
|
|
|
q->max_prb);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
srslte_ofdm_cfg_t cfg = {};
|
|
|
|
|
cfg.cp = cp;
|
|
|
|
|
cfg.nof_prb = nof_prb;
|
|
|
|
|
return ofdm_init_mbsfn_(q, &cfg, SRSLTE_DFT_BACKWARD);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void srslte_ofdm_rx_free(srslte_ofdm_t* q)
|
|
|
|
|
{
|
|
|
|
|
srslte_ofdm_free_(q);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Shifts the signal after the iFFT or before the FFT.
|
|
|
|
|
* Freq_shift is relative to inter-carrier spacing.
|
|
|
|
|
* Caution: This function shall not be called during run-time
|
|
|
|
|
*/
|
|
|
|
|
int srslte_ofdm_set_freq_shift(srslte_ofdm_t* q, float freq_shift)
|
|
|
|
|
{
|
|
|
|
|
q->cfg.freq_shift_f = freq_shift;
|
|
|
|
|
|
|
|
|
|
// Check if fft shift is required
|
|
|
|
|
if (!isnormal(q->cfg.freq_shift_f)) {
|
|
|
|
|
srslte_dft_plan_set_dc(&q->fft_plan, true);
|
|
|
|
|
return SRSLTE_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint32_t symbol_sz = q->cfg.symbol_sz;
|
|
|
|
|
srslte_cp_t cp = q->cfg.cp;
|
|
|
|
|
|
|
|
|
|
cf_t* ptr = q->shift_buffer;
|
|
|
|
|
for (uint32_t n = 0; n < 2; n++) {
|
|
|
|
|
for (uint32_t n = 0; n < SRSLTE_NOF_SLOTS_PER_SF; n++) {
|
|
|
|
|
for (uint32_t i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
uint32_t cplen = SRSLTE_CP_ISNORM(q->cp) ? SRSLTE_CP_LEN_NORM(i, q->symbol_sz) : SRSLTE_CP_LEN_EXT(q->symbol_sz);
|
|
|
|
|
for (uint32_t t = 0; t < q->symbol_sz + cplen; t++) {
|
|
|
|
|
ptr[t] = cexpf(I * 2 * M_PI * ((float)t - (float)cplen) * freq_shift / q->symbol_sz);
|
|
|
|
|
uint32_t cplen = SRSLTE_CP_ISNORM(cp) ? SRSLTE_CP_LEN_NORM(i, symbol_sz) : SRSLTE_CP_LEN_EXT(symbol_sz);
|
|
|
|
|
for (uint32_t t = 0; t < symbol_sz + cplen; t++) {
|
|
|
|
|
ptr[t] = cexpf(I * 2 * M_PI * ((float)t - (float)cplen) * freq_shift / symbol_sz);
|
|
|
|
|
}
|
|
|
|
|
ptr += q->symbol_sz + cplen;
|
|
|
|
|
ptr += symbol_sz + cplen;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Disable DC carrier addition */
|
|
|
|
|
srslte_dft_plan_set_dc(&q->fft_plan, false);
|
|
|
|
|
|
|
|
|
|
q->freq_shift = true;
|
|
|
|
|
q->freq_shift_f = freq_shift;
|
|
|
|
|
return SRSLTE_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -432,12 +365,15 @@ void srslte_ofdm_tx_free(srslte_ofdm_t* q)
|
|
|
|
|
|
|
|
|
|
void srslte_ofdm_rx_slot_ng(srslte_ofdm_t* q, cf_t* input, cf_t* output)
|
|
|
|
|
{
|
|
|
|
|
uint32_t i;
|
|
|
|
|
for (i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
input += SRSLTE_CP_ISNORM(q->cp) ? SRSLTE_CP_LEN_NORM(i, q->symbol_sz) : SRSLTE_CP_LEN_EXT(q->symbol_sz);
|
|
|
|
|
uint32_t symbol_sz = q->cfg.symbol_sz;
|
|
|
|
|
srslte_cp_t cp = q->cfg.cp;
|
|
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
input += SRSLTE_CP_ISNORM(cp) ? SRSLTE_CP_LEN_NORM(i, symbol_sz) : SRSLTE_CP_LEN_EXT(symbol_sz);
|
|
|
|
|
input -= q->window_offset_n;
|
|
|
|
|
srslte_dft_run_c(&q->fft_plan, input, q->tmp);
|
|
|
|
|
memcpy(output, &q->tmp[q->nof_guards], q->nof_re * sizeof(cf_t));
|
|
|
|
|
input += q->symbol_sz;
|
|
|
|
|
input += symbol_sz;
|
|
|
|
|
output += q->nof_re;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -445,45 +381,54 @@ void srslte_ofdm_rx_slot_ng(srslte_ofdm_t* q, cf_t* input, cf_t* output)
|
|
|
|
|
/* Transforms input samples into output OFDM symbols.
|
|
|
|
|
* Performs FFT on a each symbol and removes CP.
|
|
|
|
|
*/
|
|
|
|
|
void srslte_ofdm_rx_slot(srslte_ofdm_t* q, int slot_in_sf)
|
|
|
|
|
static void ofdm_rx_slot(srslte_ofdm_t* q, int slot_in_sf)
|
|
|
|
|
{
|
|
|
|
|
cf_t* output = q->out_buffer + slot_in_sf * q->nof_re * q->nof_symbols;
|
|
|
|
|
|
|
|
|
|
#ifdef AVOID_GURU
|
|
|
|
|
srslte_ofdm_rx_slot_ng(
|
|
|
|
|
q, q->in_buffer + slot_in_sf * q->slot_sz, q->out_buffer + slot_in_sf * q->nof_re * q->nof_symbols);
|
|
|
|
|
q, q->cfg.in_buffer + slot_in_sf * q->slot_sz, q->cfg.out_buffer + slot_in_sf * q->nof_re * q->nof_symbols);
|
|
|
|
|
#else
|
|
|
|
|
uint32_t nof_symbols = q->nof_symbols;
|
|
|
|
|
uint32_t nof_re = q->nof_re;
|
|
|
|
|
cf_t* output = q->cfg.out_buffer + slot_in_sf * nof_re * nof_symbols;
|
|
|
|
|
uint32_t symbol_sz = q->cfg.symbol_sz;
|
|
|
|
|
float norm = 1.0f / sqrtf(q->fft_plan.size);
|
|
|
|
|
cf_t* tmp = q->tmp + slot_in_sf * q->symbol_sz * q->nof_symbols;
|
|
|
|
|
cf_t* tmp = q->tmp;
|
|
|
|
|
uint32_t dc = (q->fft_plan.dc) ? 1 : 0;
|
|
|
|
|
|
|
|
|
|
srslte_dft_run_guru_c(&q->fft_plan_sf[slot_in_sf]);
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
memcpy(output, tmp + q->symbol_sz - q->nof_re / 2, sizeof(cf_t) * q->nof_re / 2);
|
|
|
|
|
memcpy(output + q->nof_re / 2, &tmp[dc], sizeof(cf_t) * q->nof_re / 2);
|
|
|
|
|
// Apply frequency domain window offset
|
|
|
|
|
if (q->window_offset_n) {
|
|
|
|
|
srslte_vec_prod_ccc(tmp, q->window_offset_buffer, tmp, symbol_sz);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Perform FFT shift
|
|
|
|
|
memcpy(output, tmp + symbol_sz - nof_re / 2, sizeof(cf_t) * nof_re / 2);
|
|
|
|
|
memcpy(output + nof_re / 2, &tmp[dc], sizeof(cf_t) * nof_re / 2);
|
|
|
|
|
|
|
|
|
|
// Normalize output
|
|
|
|
|
if (q->fft_plan.norm) {
|
|
|
|
|
srslte_vec_sc_prod_cfc(output, norm, output, q->nof_re);
|
|
|
|
|
srslte_vec_sc_prod_cfc(output, norm, output, nof_re);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tmp += q->symbol_sz;
|
|
|
|
|
output += q->nof_re;
|
|
|
|
|
tmp += symbol_sz;
|
|
|
|
|
output += nof_re;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void srslte_ofdm_rx_slot_mbsfn(srslte_ofdm_t* q, cf_t* input, cf_t* output)
|
|
|
|
|
static void ofdm_rx_slot_mbsfn(srslte_ofdm_t* q, cf_t* input, cf_t* output)
|
|
|
|
|
{
|
|
|
|
|
uint32_t i;
|
|
|
|
|
for (i = 0; i < q->nof_symbols_mbsfn; i++) {
|
|
|
|
|
if (i == q->non_mbsfn_region) {
|
|
|
|
|
input += SRSLTE_NON_MBSFN_REGION_GUARD_LENGTH(q->non_mbsfn_region, q->symbol_sz);
|
|
|
|
|
input += SRSLTE_NON_MBSFN_REGION_GUARD_LENGTH(q->non_mbsfn_region, q->cfg.symbol_sz);
|
|
|
|
|
}
|
|
|
|
|
input += (i >= q->non_mbsfn_region) ? SRSLTE_CP_LEN_EXT(q->symbol_sz) : SRSLTE_CP_LEN_NORM(i, q->symbol_sz);
|
|
|
|
|
input += (i >= q->non_mbsfn_region) ? SRSLTE_CP_LEN_EXT(q->cfg.symbol_sz) : SRSLTE_CP_LEN_NORM(i, q->cfg.symbol_sz);
|
|
|
|
|
srslte_dft_run_c(&q->fft_plan, input, q->tmp);
|
|
|
|
|
memcpy(output, &q->tmp[q->nof_guards], q->nof_re * sizeof(cf_t));
|
|
|
|
|
input += q->symbol_sz;
|
|
|
|
|
input += q->cfg.symbol_sz;
|
|
|
|
|
output += q->nof_re;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -492,11 +437,12 @@ void srslte_ofdm_rx_slot_zerocopy(srslte_ofdm_t* q, cf_t* input, cf_t* output)
|
|
|
|
|
{
|
|
|
|
|
uint32_t i;
|
|
|
|
|
for (i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
input += SRSLTE_CP_ISNORM(q->cp) ? SRSLTE_CP_LEN_NORM(i, q->symbol_sz) : SRSLTE_CP_LEN_EXT(q->symbol_sz);
|
|
|
|
|
input +=
|
|
|
|
|
SRSLTE_CP_ISNORM(q->cfg.cp) ? SRSLTE_CP_LEN_NORM(i, q->cfg.symbol_sz) : SRSLTE_CP_LEN_EXT(q->cfg.symbol_sz);
|
|
|
|
|
srslte_dft_run_c_zerocopy(&q->fft_plan, input, q->tmp);
|
|
|
|
|
memcpy(output, &q->tmp[q->symbol_sz / 2 + q->nof_guards], sizeof(cf_t) * q->nof_re / 2);
|
|
|
|
|
memcpy(output, &q->tmp[q->cfg.symbol_sz / 2 + q->nof_guards], sizeof(cf_t) * q->nof_re / 2);
|
|
|
|
|
memcpy(&output[q->nof_re / 2], &q->tmp[1], sizeof(cf_t) * q->nof_re / 2);
|
|
|
|
|
input += q->symbol_sz;
|
|
|
|
|
input += q->cfg.symbol_sz;
|
|
|
|
|
output += q->nof_re;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -504,99 +450,105 @@ void srslte_ofdm_rx_slot_zerocopy(srslte_ofdm_t* q, cf_t* input, cf_t* output)
|
|
|
|
|
void srslte_ofdm_rx_sf(srslte_ofdm_t* q)
|
|
|
|
|
{
|
|
|
|
|
uint32_t n;
|
|
|
|
|
if (q->freq_shift) {
|
|
|
|
|
srslte_vec_prod_ccc(q->in_buffer, q->shift_buffer, q->in_buffer, 2 * q->slot_sz);
|
|
|
|
|
if (isnormal(q->cfg.freq_shift_f)) {
|
|
|
|
|
srslte_vec_prod_ccc(q->cfg.in_buffer, q->shift_buffer, q->cfg.in_buffer, q->sf_sz);
|
|
|
|
|
}
|
|
|
|
|
if (!q->mbsfn_subframe) {
|
|
|
|
|
for (n = 0; n < 2; n++) {
|
|
|
|
|
srslte_ofdm_rx_slot(q, n);
|
|
|
|
|
for (n = 0; n < SRSLTE_NOF_SLOTS_PER_SF; n++) {
|
|
|
|
|
ofdm_rx_slot(q, n);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
srslte_ofdm_rx_slot_mbsfn(q, &q->in_buffer[0 * q->slot_sz], &q->out_buffer[0 * q->nof_re * q->nof_symbols]);
|
|
|
|
|
srslte_ofdm_rx_slot(q, 1);
|
|
|
|
|
ofdm_rx_slot_mbsfn(q, q->cfg.in_buffer, q->cfg.out_buffer);
|
|
|
|
|
ofdm_rx_slot(q, 1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void srslte_ofdm_rx_sf_ng(srslte_ofdm_t* q, cf_t* input, cf_t* output)
|
|
|
|
|
{
|
|
|
|
|
uint32_t n;
|
|
|
|
|
if (q->freq_shift) {
|
|
|
|
|
srslte_vec_prod_ccc(input, q->shift_buffer, input, 2 * q->slot_sz);
|
|
|
|
|
if (isnormal(q->cfg.freq_shift_f)) {
|
|
|
|
|
srslte_vec_prod_ccc(input, q->shift_buffer, input, q->sf_sz);
|
|
|
|
|
}
|
|
|
|
|
if (!q->mbsfn_subframe) {
|
|
|
|
|
for (n = 0; n < 2; n++) {
|
|
|
|
|
for (n = 0; n < SRSLTE_NOF_SLOTS_PER_SF; n++) {
|
|
|
|
|
srslte_ofdm_rx_slot_ng(q, &input[n * q->slot_sz], &output[n * q->nof_re * q->nof_symbols]);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
srslte_ofdm_rx_slot_mbsfn(q, &q->in_buffer[0 * q->slot_sz], &q->out_buffer[0 * q->nof_re * q->nof_symbols]);
|
|
|
|
|
srslte_ofdm_rx_slot(q, 1);
|
|
|
|
|
ofdm_rx_slot_mbsfn(q, q->cfg.in_buffer, q->cfg.out_buffer);
|
|
|
|
|
ofdm_rx_slot(q, 1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Transforms input OFDM symbols into output samples.
|
|
|
|
|
* Performs FFT on a each symbol and adds CP.
|
|
|
|
|
*/
|
|
|
|
|
void srslte_ofdm_tx_slot(srslte_ofdm_t* q, int slot_in_sf)
|
|
|
|
|
static void ofdm_tx_slot(srslte_ofdm_t* q, int slot_in_sf)
|
|
|
|
|
{
|
|
|
|
|
cf_t* input = q->in_buffer + slot_in_sf * q->nof_re * q->nof_symbols;
|
|
|
|
|
cf_t* output = q->out_buffer + slot_in_sf * q->slot_sz;
|
|
|
|
|
uint32_t symbol_sz = q->cfg.symbol_sz;
|
|
|
|
|
srslte_cp_t cp = q->cfg.cp;
|
|
|
|
|
|
|
|
|
|
cf_t* input = q->cfg.in_buffer + slot_in_sf * q->nof_re * q->nof_symbols;
|
|
|
|
|
cf_t* output = q->cfg.out_buffer + slot_in_sf * q->slot_sz;
|
|
|
|
|
|
|
|
|
|
#ifdef AVOID_GURU
|
|
|
|
|
for (int i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
int cp_len = SRSLTE_CP_ISNORM(q->cp) ? SRSLTE_CP_LEN_NORM(i, q->symbol_sz) : SRSLTE_CP_LEN_EXT(q->symbol_sz);
|
|
|
|
|
int cp_len = SRSLTE_CP_ISNORM(cp) ? SRSLTE_CP_LEN_NORM(i, symbol_sz) : SRSLTE_CP_LEN_EXT(symbol_sz);
|
|
|
|
|
memcpy(&q->tmp[q->nof_guards], input, q->nof_re * sizeof(cf_t));
|
|
|
|
|
srslte_dft_run_c(&q->fft_plan, q->tmp, &output[cp_len]);
|
|
|
|
|
input += q->nof_re;
|
|
|
|
|
/* add CP */
|
|
|
|
|
memcpy(output, &output[q->symbol_sz], cp_len * sizeof(cf_t));
|
|
|
|
|
output += q->symbol_sz + cp_len;
|
|
|
|
|
memcpy(output, &output[symbol_sz], cp_len * sizeof(cf_t));
|
|
|
|
|
output += symbol_sz + cp_len;
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
float norm = 1.0f / sqrtf(q->symbol_sz);
|
|
|
|
|
cf_t* tmp = q->tmp + slot_in_sf * q->symbol_sz * q->nof_symbols;
|
|
|
|
|
uint32_t nof_symbols = q->nof_symbols;
|
|
|
|
|
uint32_t nof_re = q->nof_re;
|
|
|
|
|
float norm = 1.0f / sqrtf(symbol_sz);
|
|
|
|
|
cf_t* tmp = q->tmp;
|
|
|
|
|
|
|
|
|
|
bzero(tmp, q->slot_sz);
|
|
|
|
|
uint32_t dc = (q->fft_plan.dc) ? 1 : 0;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
memcpy(&tmp[dc], &input[q->nof_re / 2], q->nof_re / 2 * sizeof(cf_t));
|
|
|
|
|
memcpy(&tmp[q->symbol_sz - q->nof_re / 2], &input[0], q->nof_re / 2 * sizeof(cf_t));
|
|
|
|
|
for (int i = 0; i < nof_symbols; i++) {
|
|
|
|
|
memcpy(&tmp[dc], &input[nof_re / 2], nof_re / 2 * sizeof(cf_t));
|
|
|
|
|
memcpy(&tmp[symbol_sz - nof_re / 2], &input[0], nof_re / 2 * sizeof(cf_t));
|
|
|
|
|
|
|
|
|
|
input += q->nof_re;
|
|
|
|
|
tmp += q->symbol_sz;
|
|
|
|
|
input += nof_re;
|
|
|
|
|
tmp += symbol_sz;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
srslte_dft_run_guru_c(&q->fft_plan_sf[slot_in_sf]);
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < q->nof_symbols; i++) {
|
|
|
|
|
int cp_len = SRSLTE_CP_ISNORM(q->cp) ? SRSLTE_CP_LEN_NORM(i, q->symbol_sz) : SRSLTE_CP_LEN_EXT(q->symbol_sz);
|
|
|
|
|
for (int i = 0; i < nof_symbols; i++) {
|
|
|
|
|
int cp_len = SRSLTE_CP_ISNORM(cp) ? SRSLTE_CP_LEN_NORM(i, symbol_sz) : SRSLTE_CP_LEN_EXT(symbol_sz);
|
|
|
|
|
|
|
|
|
|
if (q->fft_plan.norm) {
|
|
|
|
|
srslte_vec_sc_prod_cfc(&output[cp_len], norm, &output[cp_len], q->symbol_sz);
|
|
|
|
|
srslte_vec_sc_prod_cfc(&output[cp_len], norm, &output[cp_len], symbol_sz);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* add CP */
|
|
|
|
|
memcpy(output, &output[q->symbol_sz], cp_len * sizeof(cf_t));
|
|
|
|
|
output += q->symbol_sz + cp_len;
|
|
|
|
|
memcpy(output, &output[symbol_sz], cp_len * sizeof(cf_t));
|
|
|
|
|
output += symbol_sz + cp_len;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void srslte_ofdm_tx_slot_mbsfn(srslte_ofdm_t* q, cf_t* input, cf_t* output)
|
|
|
|
|
void ofdm_tx_slot_mbsfn(srslte_ofdm_t* q, cf_t* input, cf_t* output)
|
|
|
|
|
{
|
|
|
|
|
uint32_t i, cp_len;
|
|
|
|
|
for (i = 0; i < q->nof_symbols_mbsfn; i++) {
|
|
|
|
|
cp_len = (i > (q->non_mbsfn_region - 1)) ? SRSLTE_CP_LEN_EXT(q->symbol_sz) : SRSLTE_CP_LEN_NORM(i, q->symbol_sz);
|
|
|
|
|
uint32_t symbol_sz = q->cfg.symbol_sz;
|
|
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < q->nof_symbols_mbsfn; i++) {
|
|
|
|
|
int cp_len = (i > (q->non_mbsfn_region - 1)) ? SRSLTE_CP_LEN_EXT(symbol_sz) : SRSLTE_CP_LEN_NORM(i, symbol_sz);
|
|
|
|
|
memcpy(&q->tmp[q->nof_guards], input, q->nof_re * sizeof(cf_t));
|
|
|
|
|
srslte_dft_run_c(&q->fft_plan, q->tmp, &output[cp_len]);
|
|
|
|
|
input += q->nof_re;
|
|
|
|
|
/* add CP */
|
|
|
|
|
memcpy(output, &output[q->symbol_sz], cp_len * sizeof(cf_t));
|
|
|
|
|
output += q->symbol_sz + cp_len;
|
|
|
|
|
memcpy(output, &output[symbol_sz], cp_len * sizeof(cf_t));
|
|
|
|
|
output += symbol_sz + cp_len;
|
|
|
|
|
|
|
|
|
|
/*skip the small section between the non mbms region and the mbms region*/
|
|
|
|
|
if (i == (q->non_mbsfn_region - 1))
|
|
|
|
|
output += SRSLTE_NON_MBSFN_REGION_GUARD_LENGTH(q->non_mbsfn_region, q->symbol_sz);
|
|
|
|
|
output += SRSLTE_NON_MBSFN_REGION_GUARD_LENGTH(q->non_mbsfn_region, symbol_sz);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -609,14 +561,14 @@ void srslte_ofdm_tx_sf(srslte_ofdm_t* q)
|
|
|
|
|
{
|
|
|
|
|
uint32_t n;
|
|
|
|
|
if (!q->mbsfn_subframe) {
|
|
|
|
|
for (n = 0; n < 2; n++) {
|
|
|
|
|
srslte_ofdm_tx_slot(q, n);
|
|
|
|
|
for (n = 0; n < SRSLTE_NOF_SLOTS_PER_SF; n++) {
|
|
|
|
|
ofdm_tx_slot(q, n);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
srslte_ofdm_tx_slot_mbsfn(q, &q->in_buffer[0 * q->nof_re * q->nof_symbols], &q->out_buffer[0 * q->slot_sz]);
|
|
|
|
|
srslte_ofdm_tx_slot(q, 1);
|
|
|
|
|
ofdm_tx_slot_mbsfn(q, q->cfg.in_buffer, q->cfg.out_buffer);
|
|
|
|
|
ofdm_tx_slot(q, 1);
|
|
|
|
|
}
|
|
|
|
|
if (q->freq_shift) {
|
|
|
|
|
srslte_vec_prod_ccc(q->out_buffer, q->shift_buffer, q->out_buffer, 2 * q->slot_sz);
|
|
|
|
|
if (isnormal(q->cfg.freq_shift_f)) {
|
|
|
|
|
srslte_vec_prod_ccc(q->cfg.out_buffer, q->shift_buffer, q->cfg.out_buffer, q->sf_sz);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|