More clear PDSCH CP function

master
Xavier Arteaga 5 years ago committed by Xavier Arteaga
parent 1cabe2d55f
commit 93771126ed

@ -80,110 +80,126 @@ typedef struct {
static void* srslte_pdsch_decode_thread(void* arg); static void* srslte_pdsch_decode_thread(void* arg);
int srslte_pdsch_cp(srslte_pdsch_t* q, static inline bool pdsch_cp_skip_symbol(const srslte_cell_t* cell,
cf_t* input, const srslte_pdsch_grant_t* grant,
cf_t* output, uint32_t sf_idx,
srslte_pdsch_grant_t* grant, uint32_t s,
uint32_t lstart_grant, uint32_t l,
uint32_t sf_idx, uint32_t n)
bool put)
{ {
uint32_t s, n, l, lp, lstart, nof_refs; // Skip center block signals
bool skip_symbol; if ((n >= cell->nof_prb / 2 - 3 && n < cell->nof_prb / 2 + 3 + (cell->nof_prb % 2))) {
cf_t * in_ptr = input, *out_ptr = output; if (cell->frame_type == SRSLTE_FDD) {
uint32_t offset = 0; // FDD PSS/SSS
if (s == 0 && (sf_idx == 0 || sf_idx == 5) && (l >= grant->nof_symb_slot[s] - 2)) {
#ifdef DEBUG_IDX return true;
indices_ptr = 0; }
if (put) { } else {
offset_original = output; // TDD SSS
} else { if (s == 1 && (sf_idx == 0 || sf_idx == 5) && (l >= grant->nof_symb_slot[s] - 1)) {
offset_original = input; return true;
}
// TDD PSS
if (s == 0 && (sf_idx == 1 || sf_idx == 6) && (l == 2)) {
return true;
}
}
// PBCH same in FDD and TDD
if (s == 1 && sf_idx == 0 && l < 4) {
return true;
}
} }
#endif
if (q->cell.nof_ports == 1) { return false;
nof_refs = 2; }
} else {
nof_refs = 4; static inline uint32_t pdsch_cp_crs_offset(const srslte_cell_t* cell, uint32_t l, bool has_crs)
{
// No CRS, return 0
if (!has_crs) {
return 0;
} }
for (s = 0; s < 2; s++) { // For 1 port cell
if (s == 0) { if (cell->nof_ports == 1) {
lstart = lstart_grant; if (l == 0) {
return cell->id % 6;
} else { } else {
lstart = 0; return (cell->id + 3) % 6;
} }
for (l = lstart; l < grant->nof_symb_slot[s]; l++) { }
for (n = 0; n < q->cell.nof_prb; n++) {
// For more 2 ports or more
return cell->id % 3;
}
static int srslte_pdsch_cp(const srslte_pdsch_t* q,
cf_t* input,
cf_t* output,
const srslte_pdsch_grant_t* grant,
uint32_t lstart_grant,
uint32_t sf_idx,
bool put)
{
cf_t* in_ptr = input;
cf_t* out_ptr = output;
uint32_t nof_refs = (q->cell.nof_ports == 1) ? 2 : 4;
// Iterate over slots
for (uint32_t s = 0; s < SRSLTE_NOF_SLOTS_PER_SF; s++) {
// Skip PDCCH symbols
uint32_t lstart = (s == 0) ? lstart_grant : 0;
// Iterate over symbols
for (uint32_t l = lstart; l < grant->nof_symb_slot[s]; l++) {
bool has_crs = SRSLTE_SYMBOL_HAS_REF(l, q->cell.cp, q->cell.nof_ports);
uint32_t crs_offset = pdsch_cp_crs_offset(&q->cell, l, has_crs);
// Grid symbol
uint32_t lp = l + s * grant->nof_symb_slot[0];
// Iterate over PRB
for (uint32_t n = 0; n < q->cell.nof_prb; n++) {
// If this PRB is assigned // If this PRB is assigned
if (grant->prb_idx[s][n]) { if (grant->prb_idx[s][n]) {
bool skip = pdsch_cp_skip_symbol(&q->cell, grant, sf_idx, s, l, n);
skip_symbol = false; // Get grid pointer
// Skip center block signals
if ((n >= q->cell.nof_prb / 2 - 3 && n < q->cell.nof_prb / 2 + 3 + (q->cell.nof_prb % 2))) {
if (q->cell.frame_type == SRSLTE_FDD) {
// FDD PSS/SSS
if (s == 0 && (sf_idx == 0 || sf_idx == 5) && (l >= grant->nof_symb_slot[s] - 2)) {
skip_symbol = true;
}
} else {
// TDD SSS
if (s == 1 && (sf_idx == 0 || sf_idx == 5) && (l >= grant->nof_symb_slot[s] - 1)) {
skip_symbol = true;
}
// TDD PSS
if (s == 0 && (sf_idx == 1 || sf_idx == 6) && (l == 2)) {
skip_symbol = true;
}
}
// PBCH same in FDD and TDD
if (s == 1 && sf_idx == 0 && l < 4) {
skip_symbol = true;
}
}
lp = l + s * grant->nof_symb_slot[0];
if (put) { if (put) {
out_ptr = &output[(lp * q->cell.nof_prb + n) * SRSLTE_NRE]; out_ptr = &output[(lp * q->cell.nof_prb + n) * SRSLTE_NRE];
} else { } else {
in_ptr = &input[(lp * q->cell.nof_prb + n) * SRSLTE_NRE]; in_ptr = &input[(lp * q->cell.nof_prb + n) * SRSLTE_NRE];
} }
// This is a symbol in a normal PRB with or without references // This is a symbol in a normal PRB with or without references
if (!skip_symbol) { if (!skip) {
if (SRSLTE_SYMBOL_HAS_REF(l, q->cell.cp, q->cell.nof_ports)) { if (has_crs) {
if (nof_refs == 2) { prb_cp_ref(&in_ptr, &out_ptr, crs_offset, nof_refs, nof_refs, put);
if (l == 0) {
offset = q->cell.id % 6;
} else {
offset = (q->cell.id + 3) % 6;
}
} else {
offset = q->cell.id % 3;
}
prb_cp_ref(&in_ptr, &out_ptr, offset, nof_refs, nof_refs, put);
} else { } else {
prb_cp(&in_ptr, &out_ptr, 1); prb_cp(&in_ptr, &out_ptr, 1);
} }
} } else if (q->cell.nof_prb % 2 != 0) {
// This is a symbol in a PRB with PBCH or Synch signals (SS). // This is a symbol in a PRB with PBCH or Synch signals (SS).
// If the number or total PRB is odd, half of the the PBCH or SS will fall into the symbol // If the number or total PRB is odd, half of the the PBCH or SS will fall into the symbol
if ((q->cell.nof_prb % 2) && skip_symbol) {
if (n == q->cell.nof_prb / 2 - 3) { if (n == q->cell.nof_prb / 2 - 3) {
if (SRSLTE_SYMBOL_HAS_REF(l, q->cell.cp, q->cell.nof_ports)) { // Lower sync block half RB
prb_cp_ref(&in_ptr, &out_ptr, offset, nof_refs, nof_refs / 2, put); if (has_crs) {
prb_cp_ref(&in_ptr, &out_ptr, crs_offset, nof_refs, nof_refs / 2, put);
} else { } else {
prb_cp_half(&in_ptr, &out_ptr, 1); prb_cp_half(&in_ptr, &out_ptr, 1);
} }
} else if (n == q->cell.nof_prb / 2 + 3) { } else if (n == q->cell.nof_prb / 2 + 3) {
// Upper sync block half RB
// Skip half RB on the grid
if (put) { if (put) {
out_ptr += 6; out_ptr += SRSLTE_NRE / 2;
} else { } else {
in_ptr += 6; in_ptr += SRSLTE_NRE / 2;
} }
if (SRSLTE_SYMBOL_HAS_REF(l, q->cell.cp, q->cell.nof_ports)) {
prb_cp_ref(&in_ptr, &out_ptr, offset, nof_refs, nof_refs / 2, put); if (has_crs) {
prb_cp_ref(&in_ptr, &out_ptr, crs_offset, nof_refs, nof_refs / 2, put);
} else { } else {
prb_cp_half(&in_ptr, &out_ptr, 1); prb_cp_half(&in_ptr, &out_ptr, 1);
} }
@ -977,6 +993,10 @@ int srslte_pdsch_decode(srslte_pdsch_t* q,
ERROR("Error predecoding\n"); ERROR("Error predecoding\n");
return SRSLTE_ERROR; return SRSLTE_ERROR;
} }
// printf("h%d=", sf->tti); srslte_vec_fprint_c(stdout, channel->ce[0][0], SRSLTE_SF_LEN_RE(q->cell.nof_prb,
// q->cell.cp)); printf("h%d=", sf->tti); srslte_vec_fprint_c(stdout, q->ce[0][0], cfg->grant.nof_re);
// printf("xx%d=", sf->tti); srslte_vec_fprint_c(stdout, x[0], cfg->grant.nof_re);
// printf("y%d=", sf->tti); srslte_vec_fprint_c(stdout, q->symbols[0], cfg->grant.nof_re);
// Layer demapping only if necessary // Layer demapping only if necessary
if (cfg->grant.nof_layers != nof_tb) { if (cfg->grant.nof_layers != nof_tb) {
@ -1221,6 +1241,7 @@ int srslte_pdsch_encode(srslte_pdsch_t* q,
for (i = 0; i < q->cell.nof_ports; i++) { for (i = 0; i < q->cell.nof_ports; i++) {
srslte_pdsch_put(q, q->symbols[i], sf_symbols[i], &cfg->grant, lstart, sf->tti % 10); srslte_pdsch_put(q, q->symbols[i], sf_symbols[i], &cfg->grant, lstart, sf->tti % 10);
} }
// printf("x%d=", sf->tti); srslte_vec_fprint_c(stdout, q->symbols[0], cfg->grant.nof_re);
if (cfg->meas_time_en) { if (cfg->meas_time_en) {
gettimeofday(&t[2], NULL); gettimeofday(&t[2], NULL);

Loading…
Cancel
Save