srsLTE/srslte/lib/ch_estimation/src/chest_dl.c

357 lines
14 KiB
C
Raw Normal View History

2014-11-05 05:19:35 -08:00
/**
*
* \section COPYRIGHT
*
2015-11-13 04:22:33 -08:00
* Copyright 2013-2015 Software Radio Systems Limited
2014-11-05 05:19:35 -08:00
*
* \section LICENSE
*
* This file is part of the srsLTE library.
2014-11-05 05:19:35 -08:00
*
* srsLTE is free software: you can redistribute it and/or modify
2015-05-08 08:05:40 -07:00
* it under the terms of the GNU Affero General Public License as
2014-11-05 05:19:35 -08:00
* 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,
2014-11-05 05:19:35 -08:00
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
2015-05-08 08:05:40 -07:00
* GNU Affero General Public License for more details.
2014-11-05 05:19:35 -08:00
*
2015-05-08 08:05:40 -07:00
* A copy of the GNU Affero General Public License can be found in
2014-11-05 05:19:35 -08:00
* the LICENSE file in the top-level directory of this distribution
* and at http://www.gnu.org/licenses/.
*
*/
#include <stdio.h>
#include <stdlib.h>
#include <strings.h>
#include <string.h>
#include <complex.h>
2014-11-11 10:20:09 -08:00
#include <math.h>
2014-11-05 05:19:35 -08:00
#include "srslte/config.h"
2014-11-05 05:19:35 -08:00
#include "srslte/ch_estimation/chest_dl.h"
#include "srslte/utils/vector.h"
#include "srslte/utils/convolution.h"
2014-11-11 10:20:09 -08:00
#define ESTIMATE_NOISE_LS_PSS
2014-11-05 05:19:35 -08:00
/** 3GPP LTE Downlink channel estimator and equalizer.
* Estimates the channel in the resource elements transmitting references and interpolates for the rest
* of the resource grid.
*
* The equalizer uses the channel estimates to produce an estimation of the transmitted symbol.
*
2015-03-18 05:41:50 -07:00
* This object depends on the srslte_refsignal_t object for creating the LTE CSR signal.
2014-11-05 05:19:35 -08:00
*/
2015-03-18 05:41:50 -07:00
int srslte_chest_dl_init(srslte_chest_dl_t *q, srslte_cell_t cell)
2014-11-05 05:19:35 -08:00
{
int ret = SRSLTE_ERROR_INVALID_INPUTS;
2014-11-05 05:19:35 -08:00
if (q != NULL &&
2015-03-18 05:59:29 -07:00
srslte_cell_isvalid(&cell))
2014-11-05 05:19:35 -08:00
{
2015-03-18 05:41:50 -07:00
bzero(q, sizeof(srslte_chest_dl_t));
2014-11-05 05:19:35 -08:00
2015-03-18 05:41:50 -07:00
ret = srslte_refsignal_cs_init(&q->csr_signal, cell);
if (ret != SRSLTE_SUCCESS) {
2014-11-05 05:19:35 -08:00
fprintf(stderr, "Error initializing CSR signal (%d)\n",ret);
goto clean_exit;
}
q->tmp_noise = srslte_vec_malloc(sizeof(cf_t) * SRSLTE_REFSIGNAL_MAX_NUM_SF(cell.nof_prb));
if (!q->tmp_noise) {
2014-11-05 05:19:35 -08:00
perror("malloc");
goto clean_exit;
}
q->pilot_estimates = srslte_vec_malloc(sizeof(cf_t) * SRSLTE_REFSIGNAL_MAX_NUM_SF(cell.nof_prb));
if (!q->pilot_estimates) {
2014-12-01 13:15:32 -08:00
perror("malloc");
goto clean_exit;
}
q->pilot_recv_signal = srslte_vec_malloc(sizeof(cf_t) * SRSLTE_REFSIGNAL_MAX_NUM_SF(cell.nof_prb));
if (!q->pilot_recv_signal) {
perror("malloc");
goto clean_exit;
}
2014-11-05 05:19:35 -08:00
2015-03-18 05:59:29 -07:00
if (srslte_interp_linear_vector_init(&q->srslte_interp_linvec, SRSLTE_NRE*cell.nof_prb)) {
2014-11-11 10:20:09 -08:00
fprintf(stderr, "Error initializing vector interpolator\n");
goto clean_exit;
}
2014-11-05 05:19:35 -08:00
2015-03-18 05:59:29 -07:00
if (srslte_interp_linear_init(&q->srslte_interp_lin, 2*cell.nof_prb, SRSLTE_NRE/2)) {
2014-11-11 10:20:09 -08:00
fprintf(stderr, "Error initializing interpolator\n");
goto clean_exit;
2014-11-05 05:19:35 -08:00
}
if (srslte_pss_generate(q->pss_signal, cell.id%3)) {
fprintf(stderr, "Error initializing PSS signal for noise estimation\n");
goto clean_exit;
}
q->w_filter = NULL;
2014-11-05 05:19:35 -08:00
q->cell = cell;
}
ret = SRSLTE_SUCCESS;
2014-11-05 05:19:35 -08:00
clean_exit:
if (ret != SRSLTE_SUCCESS) {
2015-03-18 05:41:50 -07:00
srslte_chest_dl_free(q);
2014-11-05 05:19:35 -08:00
}
return ret;
}
2015-03-18 05:41:50 -07:00
void srslte_chest_dl_free(srslte_chest_dl_t *q)
2014-11-05 05:19:35 -08:00
{
2015-03-18 05:41:50 -07:00
srslte_refsignal_cs_free(&q->csr_signal);
2014-11-05 05:19:35 -08:00
2014-12-01 13:15:32 -08:00
if (q->tmp_noise) {
free(q->tmp_noise);
}
2015-03-18 05:41:50 -07:00
srslte_interp_linear_vector_free(&q->srslte_interp_linvec);
srslte_interp_linear_free(&q->srslte_interp_lin);
2014-11-11 10:20:09 -08:00
if (q->pilot_estimates) {
free(q->pilot_estimates);
}
if (q->pilot_recv_signal) {
free(q->pilot_recv_signal);
2014-11-05 05:19:35 -08:00
}
2015-03-18 05:41:50 -07:00
bzero(q, sizeof(srslte_chest_dl_t));
2014-11-05 05:19:35 -08:00
}
void srslte_chest_dl_set_filter_w(srslte_chest_dl_t *q, cf_t *w) {
q->w_filter = w;
2014-11-11 10:20:09 -08:00
}
/* Uses the difference between the averaged and non-averaged pilot estimates */
static float estimate_noise_pilots(srslte_chest_dl_t *q, cf_t *ce, uint32_t port_id)
{
int nref=SRSLTE_REFSIGNAL_NUM_SF(q->cell.nof_prb, port_id);
/* Get averaged pilots from channel estimates */
srslte_refsignal_cs_get_sf(q->cell, port_id, ce, q->tmp_noise);
/* Substract noisy pilot estimates */
srslte_vec_sub_ccc(q->tmp_noise, q->pilot_estimates, q->tmp_noise, nref);
/* Compute average power */
float power = sqrt(2.0)*q->cell.nof_ports*srslte_vec_avg_power_cf(q->tmp_noise, nref);
return power;
2014-11-11 10:20:09 -08:00
}
2014-11-05 05:19:35 -08:00
#ifdef ESTIMATE_NOISE_LS_PSS
static float estimate_noise_pss(srslte_chest_dl_t *q, cf_t *input, cf_t *ce)
{
/* Get PSS from received signal */
srslte_pss_get_slot(input, q->tmp_pss, q->cell.nof_prb, q->cell.cp);
/* Get channel estimates for PSS position */
srslte_pss_get_slot(ce, q->tmp_pss_noisy, q->cell.nof_prb, q->cell.cp);
2014-12-01 13:15:32 -08:00
/* Multiply known PSS by channel estimates */
srslte_vec_prod_ccc(q->tmp_pss_noisy, q->pss_signal, q->tmp_pss_noisy, SRSLTE_PSS_LEN);
2015-01-14 18:16:35 -08:00
/* Substract received signal */
srslte_vec_sub_ccc(q->tmp_pss_noisy, q->tmp_pss, q->tmp_pss_noisy, SRSLTE_PSS_LEN);
/* Compute average power */
float power = q->cell.nof_ports*srslte_vec_avg_power_cf(q->tmp_pss_noisy, SRSLTE_PSS_LEN)/sqrt(2);
return power;
2014-12-01 13:15:32 -08:00
}
#else
2015-01-14 18:16:35 -08:00
/* Uses the 5 empty transmitted SC before and after the SSS and PSS sequences for noise estimation */
static float estimate_noise_empty_sc(srslte_chest_dl_t *q, cf_t *input) {
2015-03-18 05:59:29 -07:00
int k_sss = (SRSLTE_CP_NSYMB(q->cell.cp) - 2) * q->cell.nof_prb * SRSLTE_NRE + q->cell.nof_prb * SRSLTE_NRE / 2 - 31;
2015-01-14 18:16:35 -08:00
float noise_power = 0;
2015-03-18 11:14:24 -07:00
noise_power += srslte_vec_avg_power_cf(&input[k_sss-5], 5); // 5 empty SC before SSS
noise_power += srslte_vec_avg_power_cf(&input[k_sss+62], 5); // 5 empty SC after SSS
2015-03-18 05:59:29 -07:00
int k_pss = (SRSLTE_CP_NSYMB(q->cell.cp) - 1) * q->cell.nof_prb * SRSLTE_NRE + q->cell.nof_prb * SRSLTE_NRE / 2 - 31;
2015-03-18 11:14:24 -07:00
noise_power += srslte_vec_avg_power_cf(&input[k_pss-5], 5); // 5 empty SC before PSS
noise_power += srslte_vec_avg_power_cf(&input[k_pss+62], 5); // 5 empty SC after PSS
2015-01-14 18:16:35 -08:00
return noise_power;
}
#endif
2014-11-05 05:19:35 -08:00
#define cesymb(i) ce[SRSLTE_RE_IDX(q->cell.nof_prb,i,0)]
static void interpolate_filter_pilots(srslte_chest_dl_t *q, cf_t *pilot_estimates, cf_t *w, cf_t *ce, uint32_t port_id)
2014-11-05 05:19:35 -08:00
{
int nsymbols = srslte_refsignal_cs_nof_symbols(port_id);
int nsc=SRSLTE_NRE*q->cell.nof_prb;
int nref=2*q->cell.nof_prb;
// Interpolation filter in frequency domain
for (uint32_t s=0;s<nsymbols;s++) {
for (int i=0;i<nsc;i++) {
uint32_t sym_idx=srslte_refsignal_cs_nsymbol(s,q->cell.cp, port_id);
ce[nsc*sym_idx+i] = srslte_vec_dot_prod_ccc(&pilot_estimates[s*nref], &w[i*nref], nref);
2014-11-11 10:20:09 -08:00
}
2014-12-01 13:15:32 -08:00
}
/* Now interpolate in the time domain between symbols */
if (SRSLTE_CP_ISNORM(q->cell.cp)) {
if (nsymbols == 4) {
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(0), &cesymb(4), &cesymb(1), 3);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(4), &cesymb(7), &cesymb(5), 2);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(7), &cesymb(11), &cesymb(8), 3);
srslte_interp_linear_vector2(&q->srslte_interp_linvec, &cesymb(7), &cesymb(11), &cesymb(11), &cesymb(12), 2);
} else {
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(8), &cesymb(1), &cesymb(0), 1);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(1), &cesymb(8), &cesymb(2), 6);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(1), &cesymb(8), &cesymb(9), 5);
}
} else {
if (nsymbols == 4) {
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(0), &cesymb(3), &cesymb(1), 2);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(3), &cesymb(6), &cesymb(4), 2);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(6), &cesymb(9), &cesymb(7), 2);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(6), &cesymb(9), &cesymb(9), 2);
} else {
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(7), &cesymb(1), &cesymb(0), 1);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(1), &cesymb(7), &cesymb(2), 5);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(1), &cesymb(7), &cesymb(8), 4);
}
2014-11-05 05:19:35 -08:00
}
}
static void interpolate_pilots(srslte_chest_dl_t *q, cf_t *pilot_estimates, cf_t *ce, uint32_t port_id)
2014-11-05 05:19:35 -08:00
{
/* interpolate the symbols with references in the freq domain */
2014-11-11 10:20:09 -08:00
uint32_t l;
2015-03-18 05:41:50 -07:00
uint32_t nsymbols = srslte_refsignal_cs_nof_symbols(port_id);
2014-11-11 10:20:09 -08:00
/* Interpolate in the frequency domain */
for (l=0;l<nsymbols;l++) {
2015-03-18 05:41:50 -07:00
uint32_t fidx_offset = srslte_refsignal_cs_fidx(q->cell, l, port_id, 0);
srslte_interp_linear_offset(&q->srslte_interp_lin, &pilot_estimates[2*q->cell.nof_prb*l],
2015-03-18 05:59:29 -07:00
&ce[srslte_refsignal_cs_nsymbol(l,q->cell.cp, port_id) * q->cell.nof_prb * SRSLTE_NRE],
fidx_offset, SRSLTE_NRE/2-fidx_offset);
2014-11-05 05:19:35 -08:00
}
2014-11-11 10:20:09 -08:00
/* Now interpolate in the time domain between symbols */
2015-03-18 05:59:29 -07:00
if (SRSLTE_CP_ISNORM(q->cell.cp)) {
2014-11-25 08:23:40 -08:00
if (nsymbols == 4) {
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(0), &cesymb(4), &cesymb(1), 3);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(4), &cesymb(7), &cesymb(5), 2);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(7), &cesymb(11), &cesymb(8), 3);
srslte_interp_linear_vector2(&q->srslte_interp_linvec, &cesymb(7), &cesymb(11), &cesymb(11), &cesymb(12), 2);
2014-11-25 08:23:40 -08:00
} else {
2015-03-18 05:41:50 -07:00
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(8), &cesymb(1), &cesymb(0), 1);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(1), &cesymb(8), &cesymb(2), 6);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(1), &cesymb(8), &cesymb(9), 5);
2014-11-25 08:23:40 -08:00
}
2014-11-11 10:20:09 -08:00
} else {
2014-11-25 08:23:40 -08:00
if (nsymbols == 4) {
2015-03-18 05:41:50 -07:00
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(0), &cesymb(3), &cesymb(1), 2);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(3), &cesymb(6), &cesymb(4), 2);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(6), &cesymb(9), &cesymb(7), 2);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(6), &cesymb(9), &cesymb(9), 2);
2014-11-25 08:23:40 -08:00
} else {
2015-03-18 05:41:50 -07:00
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(7), &cesymb(1), &cesymb(0), 1);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(1), &cesymb(7), &cesymb(2), 5);
srslte_interp_linear_vector(&q->srslte_interp_linvec, &cesymb(1), &cesymb(7), &cesymb(8), 4);
2014-11-25 08:23:40 -08:00
}
2014-11-11 10:20:09 -08:00
}
}
2015-03-18 05:41:50 -07:00
float srslte_chest_dl_rssi(srslte_chest_dl_t *q, cf_t *input, uint32_t port_id) {
uint32_t l;
2014-11-11 10:20:09 -08:00
float rssi = 0;
2015-03-18 05:41:50 -07:00
uint32_t nsymbols = srslte_refsignal_cs_nof_symbols(port_id);
for (l=0;l<nsymbols;l++) {
2015-03-18 05:59:29 -07:00
cf_t *tmp = &input[srslte_refsignal_cs_nsymbol(l, q->cell.cp, port_id) * q->cell.nof_prb * SRSLTE_NRE];
2015-03-18 11:14:24 -07:00
rssi += srslte_vec_dot_prod_conj_ccc(tmp, tmp, q->cell.nof_prb * SRSLTE_NRE);
}
return rssi/nsymbols;
2014-11-11 10:20:09 -08:00
}
2015-03-18 05:41:50 -07:00
int srslte_chest_dl_estimate_port(srslte_chest_dl_t *q, cf_t *input, cf_t *ce, uint32_t sf_idx, uint32_t port_id)
2014-11-05 05:19:35 -08:00
{
/* Get references from the input signal */
srslte_refsignal_cs_get_sf(q->cell, port_id, input, q->pilot_recv_signal);
2014-11-05 05:19:35 -08:00
/* Use the known CSR signal to compute Least-squares estimates */
srslte_vec_prod_conj_ccc(q->pilot_recv_signal, q->csr_signal.pilots[port_id/2][sf_idx],
q->pilot_estimates, SRSLTE_REFSIGNAL_NUM_SF(q->cell.nof_prb, port_id));
if (ce != NULL) {
/* TESTING: Use robust MMSE interpolation filter */
if (q->w_filter) {
interpolate_filter_pilots(q, q->pilot_estimates, q->w_filter, ce, port_id);
/* Estimate noise from difference from averaged and estimated pilots */
if (sf_idx == 0 || sf_idx == 5) {
q->noise_estimate[port_id] = estimate_noise_pilots(q, ce, port_id);
}
/* If w filter not defined, resort to LS estimate + linear interpolation */
} else {
interpolate_pilots(q, q->pilot_estimates, ce, port_id);
/* If not averaging, compute noise from empty subcarriers */
if (sf_idx == 0 || sf_idx == 5) {
#ifdef ESTIMATE_NOISE_LS_PSS
q->noise_estimate[port_id] = estimate_noise_pss(q, input, ce);
#else
q->noise_estimate[port_id] = estimate_noise_empty_sc(q, input);
#endif
}
}
}
2015-01-17 21:33:28 -08:00
/* Compute RSRP for the channel estimates in this port */
2016-02-17 09:08:16 -08:00
q->rsrp[port_id] = srslte_vec_avg_power_cf(q->pilot_recv_signal, SRSLTE_REFSIGNAL_NUM_SF(q->cell.nof_prb, port_id));
2015-01-17 21:33:28 -08:00
if (port_id == 0) {
/* compute rssi only for port 0 */
2015-03-18 05:41:50 -07:00
q->rssi[port_id] = srslte_chest_dl_rssi(q, input, port_id);
2015-01-17 21:33:28 -08:00
}
2014-11-05 05:19:35 -08:00
return 0;
}
2015-03-18 05:41:50 -07:00
int srslte_chest_dl_estimate(srslte_chest_dl_t *q, cf_t *input, cf_t *ce[SRSLTE_MAX_PORTS], uint32_t sf_idx)
2014-11-05 05:19:35 -08:00
{
uint32_t port_id;
for (port_id=0;port_id<q->cell.nof_ports;port_id++) {
2015-03-18 05:41:50 -07:00
srslte_chest_dl_estimate_port(q, input, ce[port_id], sf_idx, port_id);
2014-11-05 05:19:35 -08:00
}
return SRSLTE_SUCCESS;
2014-11-05 05:19:35 -08:00
}
2015-03-18 05:41:50 -07:00
float srslte_chest_dl_get_noise_estimate(srslte_chest_dl_t *q) {
2015-05-19 11:22:58 -07:00
return srslte_vec_acc_ff(q->noise_estimate, q->cell.nof_ports)/q->cell.nof_ports;
}
float srslte_chest_dl_get_snr(srslte_chest_dl_t *q) {
2016-02-17 09:08:16 -08:00
return srslte_chest_dl_get_rsrp(q)/srslte_chest_dl_get_noise_estimate(q);
}
2015-03-18 05:41:50 -07:00
float srslte_chest_dl_get_rssi(srslte_chest_dl_t *q) {
2015-03-18 05:59:29 -07:00
return 4*q->rssi[0]/q->cell.nof_prb/SRSLTE_NRE;
2014-11-05 05:19:35 -08:00
}
/* q->rssi[0] is the average power in all RE in all symbol containing references for port 0 . q->rssi[0]/q->cell.nof_prb is the average power per PRB
* q->rsrp[0] is the average power of RE containing references only (for port 0).
*/
2015-03-18 05:41:50 -07:00
float srslte_chest_dl_get_rsrq(srslte_chest_dl_t *q) {
return q->cell.nof_prb*q->rsrp[0] / q->rssi[0];
2014-11-05 05:19:35 -08:00
}
2015-03-18 05:41:50 -07:00
float srslte_chest_dl_get_rsrp(srslte_chest_dl_t *q) {
2015-01-17 21:33:28 -08:00
// return sum of power received from all tx ports
2015-03-18 11:14:24 -07:00
return srslte_vec_acc_ff(q->rsrp, q->cell.nof_ports);
2014-11-05 05:19:35 -08:00
}