|
|
@ -15,28 +15,36 @@
|
|
|
|
#include <string.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include "srsran/common/test_common.h"
|
|
|
|
#include "srsran/srsran.h"
|
|
|
|
#include "srsran/srsran.h"
|
|
|
|
|
|
|
|
|
|
|
|
srsran_cell_t cell = {.nof_prb = 6,
|
|
|
|
// Test parameters
|
|
|
|
.nof_ports = 1,
|
|
|
|
static uint32_t pci = 1;
|
|
|
|
.id = 1,
|
|
|
|
static uint16_t rnti = 0x46;
|
|
|
|
.cp = SRSRAN_CP_NORM,
|
|
|
|
static uint32_t cfi = 1;
|
|
|
|
.phich_resources = SRSRAN_PHICH_R_1,
|
|
|
|
static uint32_t nof_ports = 1;
|
|
|
|
.phich_length = SRSRAN_PHICH_NORM};
|
|
|
|
static bool print_dci_table;
|
|
|
|
|
|
|
|
static srsran_dci_cfg_t dci_cfg = {};
|
|
|
|
uint32_t cfi = 1;
|
|
|
|
static uint32_t nof_prb = 100;
|
|
|
|
uint32_t nof_rx_ant = 1;
|
|
|
|
static float snr_dB = 20.0f;
|
|
|
|
bool print_dci_table;
|
|
|
|
static uint32_t repetitions = 1;
|
|
|
|
srsran_dci_cfg_t dci_cfg = {};
|
|
|
|
static bool false_check = false;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Test objects
|
|
|
|
|
|
|
|
static srsran_random_t random_gen = NULL;
|
|
|
|
|
|
|
|
static srsran_pdcch_t pdcch_tx = {};
|
|
|
|
|
|
|
|
static srsran_pdcch_t pdcch_rx = {};
|
|
|
|
|
|
|
|
static srsran_chest_dl_res_t chest_dl_res = {};
|
|
|
|
|
|
|
|
static srsran_channel_awgn_t awgn = {};
|
|
|
|
|
|
|
|
static cf_t* slot_symbols[SRSRAN_MAX_PORTS];
|
|
|
|
|
|
|
|
|
|
|
|
void usage(char* prog)
|
|
|
|
void usage(char* prog)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
printf("Usage: %s [cfpndxv]\n", prog);
|
|
|
|
printf("Usage: %s [cfpndxv]\n", prog);
|
|
|
|
printf("\t-c cell id [Default %d]\n", cell.id);
|
|
|
|
printf("\t-c cell id [Default %d]\n", pci);
|
|
|
|
printf("\t-f cfi [Default %d]\n", cfi);
|
|
|
|
printf("\t-f cfi [Default %d]\n", cfi);
|
|
|
|
printf("\t-p cell.nof_ports [Default %d]\n", cell.nof_ports);
|
|
|
|
printf("\t-p cell.nof_ports [Default %d]\n", nof_ports);
|
|
|
|
printf("\t-n cell.nof_prb [Default %d]\n", cell.nof_prb);
|
|
|
|
printf("\t-n cell.nof_prb [Default %d]\n", nof_prb);
|
|
|
|
printf("\t-A nof_rx_ant [Default %d]\n", nof_rx_ant);
|
|
|
|
|
|
|
|
printf("\t-d Print DCI table [Default %s]\n", print_dci_table ? "yes" : "no");
|
|
|
|
printf("\t-d Print DCI table [Default %s]\n", print_dci_table ? "yes" : "no");
|
|
|
|
printf("\t-x Enable/Disable Cross-scheduling [Default %s]\n", dci_cfg.cif_enabled ? "enabled" : "disabled");
|
|
|
|
printf("\t-x Enable/Disable Cross-scheduling [Default %s]\n", dci_cfg.cif_enabled ? "enabled" : "disabled");
|
|
|
|
printf("\t-v [set srsran_verbose to debug, default none]\n");
|
|
|
|
printf("\t-v [set srsran_verbose to debug, default none]\n");
|
|
|
@ -48,19 +56,16 @@ void parse_args(int argc, char** argv)
|
|
|
|
while ((opt = getopt(argc, argv, "cfpndvAx")) != -1) {
|
|
|
|
while ((opt = getopt(argc, argv, "cfpndvAx")) != -1) {
|
|
|
|
switch (opt) {
|
|
|
|
switch (opt) {
|
|
|
|
case 'p':
|
|
|
|
case 'p':
|
|
|
|
cell.nof_ports = (uint32_t)strtol(argv[optind], NULL, 10);
|
|
|
|
nof_ports = (uint32_t)strtol(argv[optind], NULL, 10);
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
case 'f':
|
|
|
|
cfi = (uint32_t)strtol(argv[optind], NULL, 10);
|
|
|
|
cfi = (uint32_t)strtol(argv[optind], NULL, 10);
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
case 'n':
|
|
|
|
cell.nof_prb = (uint32_t)strtol(argv[optind], NULL, 10);
|
|
|
|
nof_prb = (uint32_t)strtol(argv[optind], NULL, 10);
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
case 'c':
|
|
|
|
case 'c':
|
|
|
|
cell.id = (uint32_t)strtol(argv[optind], NULL, 10);
|
|
|
|
pci = (uint32_t)strtol(argv[optind], NULL, 10);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'A':
|
|
|
|
|
|
|
|
nof_rx_ant = (uint32_t)strtol(argv[optind], NULL, 10);
|
|
|
|
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
case 'd':
|
|
|
|
case 'd':
|
|
|
|
print_dci_table = true;
|
|
|
|
print_dci_table = true;
|
|
|
@ -78,10 +83,19 @@ void parse_args(int argc, char** argv)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int test_dci_payload_size()
|
|
|
|
static void print_dci_msg(const char* desc, const srsran_dci_msg_t* dci_msg)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
printf("%srnti=0x%04x; format=%s; L=%d; ncce=%d; payload=",
|
|
|
|
|
|
|
|
desc,
|
|
|
|
|
|
|
|
rnti,
|
|
|
|
|
|
|
|
srsran_dci_format_string(dci_msg->format),
|
|
|
|
|
|
|
|
dci_msg->location.L,
|
|
|
|
|
|
|
|
dci_msg->location.ncce);
|
|
|
|
|
|
|
|
srsran_vec_fprint_byte(stdout, dci_msg->payload, dci_msg->nof_bits);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int assert_payload_size(const srsran_cell_t* cell, srsran_dl_sf_cfg_t* dl_sf)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
int x[5];
|
|
|
|
|
|
|
|
const srsran_dci_format_t formats[] = {
|
|
|
|
const srsran_dci_format_t formats[] = {
|
|
|
|
SRSRAN_DCI_FORMAT0, SRSRAN_DCI_FORMAT1, SRSRAN_DCI_FORMAT1A, SRSRAN_DCI_FORMAT1C, SRSRAN_DCI_FORMAT2A};
|
|
|
|
SRSRAN_DCI_FORMAT0, SRSRAN_DCI_FORMAT1, SRSRAN_DCI_FORMAT1A, SRSRAN_DCI_FORMAT1C, SRSRAN_DCI_FORMAT2A};
|
|
|
|
const int prb[6] = {6, 15, 25, 50, 75, 100};
|
|
|
|
const int prb[6] = {6, 15, 25, 50, 75, 100};
|
|
|
@ -92,297 +106,236 @@ int test_dci_payload_size()
|
|
|
|
{27, 33, 27, 14, 42},
|
|
|
|
{27, 33, 27, 14, 42},
|
|
|
|
{28, 39, 28, 15, 48}};
|
|
|
|
{28, 39, 28, 15, 48}};
|
|
|
|
|
|
|
|
|
|
|
|
srsran_dl_sf_cfg_t dl_sf;
|
|
|
|
// Skip if special options are requested
|
|
|
|
ZERO_OBJECT(dl_sf);
|
|
|
|
if (dci_cfg.cif_enabled || dci_cfg.multiple_csi_request_enabled) {
|
|
|
|
|
|
|
|
return SRSRAN_SUCCESS;
|
|
|
|
srsran_cell_t cell_test;
|
|
|
|
}
|
|
|
|
ZERO_OBJECT(cell_test);
|
|
|
|
|
|
|
|
cell_test.nof_ports = 1;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
ZERO_OBJECT(dci_cfg);
|
|
|
|
// Skip if MIMO is enabled
|
|
|
|
|
|
|
|
if (cell->nof_ports > 1) {
|
|
|
|
|
|
|
|
return SRSRAN_SUCCESS;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
printf("Testing DCI payload sizes...\n");
|
|
|
|
for (uint32_t i = 0; i < 6; i++) {
|
|
|
|
printf(" PRB\t0\t1\t1A\t1C\t2A\n");
|
|
|
|
if (prb[i] != cell->nof_prb) {
|
|
|
|
for (i = 0; i < 6; i++) {
|
|
|
|
continue;
|
|
|
|
|
|
|
|
}
|
|
|
|
int n = prb[i];
|
|
|
|
int n = prb[i];
|
|
|
|
cell_test.nof_prb = n;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for (j = 0; j < 5; j++) {
|
|
|
|
uint32_t x[5];
|
|
|
|
x[j] = srsran_dci_format_sizeof(&cell_test, &dl_sf, &dci_cfg, formats[j]);
|
|
|
|
for (uint32_t j = 0; j < 5; j++) {
|
|
|
|
|
|
|
|
x[j] = srsran_dci_format_sizeof(cell, dl_sf, &dci_cfg, formats[j]);
|
|
|
|
if (x[j] != dci_sz[i][j]) {
|
|
|
|
if (x[j] != dci_sz[i][j]) {
|
|
|
|
ERROR("Invalid DCI payload size for %s and %d PRB. Is %d and should be %d",
|
|
|
|
ERROR("Invalid DCI payload size for %s and %d PRB. Is %d and should be %d",
|
|
|
|
srsran_dci_format_string(formats[j]),
|
|
|
|
srsran_dci_format_string(formats[j]),
|
|
|
|
n,
|
|
|
|
n,
|
|
|
|
x[j],
|
|
|
|
x[j],
|
|
|
|
dci_sz[i][j]);
|
|
|
|
dci_sz[i][j]);
|
|
|
|
return -1;
|
|
|
|
return SRSRAN_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf(" %2d:\t%2d\t%2d\t%2d\t%2d\t%2d\n", n, x[0], x[1], x[2], x[3], x[4]);
|
|
|
|
return SRSRAN_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("Ok\n");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (print_dci_table) {
|
|
|
|
|
|
|
|
printf("dci_sz_table[101][4] = {\n");
|
|
|
|
|
|
|
|
for (i = 0; i <= 100; i++) {
|
|
|
|
|
|
|
|
printf(" {");
|
|
|
|
|
|
|
|
for (j = 0; j < 4; j++) {
|
|
|
|
|
|
|
|
cell_test.nof_prb = i;
|
|
|
|
|
|
|
|
printf("%d", srsran_dci_format_sizeof(&cell, &dl_sf, &dci_cfg, formats[j]));
|
|
|
|
|
|
|
|
if (j < 3) {
|
|
|
|
|
|
|
|
printf(", ");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i < 100) {
|
|
|
|
|
|
|
|
printf("},\n");
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
printf("}\n");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("};\n");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
static const srsran_dci_format_t formats[] = {SRSRAN_DCI_FORMAT0,
|
|
|
|
srsran_dci_msg_t dci_tx, dci_rx;
|
|
|
|
SRSRAN_DCI_FORMAT1A,
|
|
|
|
srsran_dci_location_t dci_location;
|
|
|
|
SRSRAN_DCI_FORMAT1,
|
|
|
|
srsran_dci_format_t dci_format;
|
|
|
|
SRSRAN_DCI_FORMAT2A,
|
|
|
|
srsran_dci_dl_t ra_dl_tx;
|
|
|
|
SRSRAN_DCI_FORMAT2,
|
|
|
|
srsran_dci_dl_t ra_dl_rx;
|
|
|
|
SRSRAN_DCI_NOF_FORMATS};
|
|
|
|
} testcase_dci_t;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int main(int argc, char** argv)
|
|
|
|
static int test_case1()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
srsran_chest_dl_res_t chest_dl_res;
|
|
|
|
uint32_t nof_re = SRSRAN_NOF_RE(pdcch_tx.cell);
|
|
|
|
srsran_pdcch_t pdcch_tx, pdcch_rx;
|
|
|
|
struct timeval t[3] = {};
|
|
|
|
testcase_dci_t testcases[10];
|
|
|
|
uint64_t t_encode_us = 0;
|
|
|
|
srsran_regs_t regs;
|
|
|
|
uint64_t t_encode_count = 0;
|
|
|
|
int i;
|
|
|
|
uint64_t t_llr_us = 0;
|
|
|
|
int nof_re;
|
|
|
|
uint64_t t_decode_us = 0;
|
|
|
|
cf_t* slot_symbols[SRSRAN_MAX_PORTS];
|
|
|
|
uint64_t t_decode_count = 0;
|
|
|
|
int nof_dcis;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bzero(&testcases, sizeof(testcase_dci_t) * 10);
|
|
|
|
// Iterate all possible subframes
|
|
|
|
srsran_random_t random_gen = srsran_random_init(0x1234);
|
|
|
|
for (uint32_t f_idx = 0; formats[f_idx] != SRSRAN_DCI_NOF_FORMATS; f_idx++) {
|
|
|
|
|
|
|
|
srsran_dci_format_t format = formats[f_idx];
|
|
|
|
|
|
|
|
|
|
|
|
int ret = -1;
|
|
|
|
for (uint32_t sf_idx = 0; sf_idx < repetitions * SRSRAN_NOF_SF_X_FRAME; sf_idx++) {
|
|
|
|
|
|
|
|
srsran_dl_sf_cfg_t dl_sf_cfg = {};
|
|
|
|
|
|
|
|
dl_sf_cfg.cfi = cfi;
|
|
|
|
|
|
|
|
dl_sf_cfg.tti = sf_idx % 10240;
|
|
|
|
|
|
|
|
|
|
|
|
parse_args(argc, argv);
|
|
|
|
// Generate PDCCH locations
|
|
|
|
|
|
|
|
srsran_dci_location_t locations[SRSRAN_MAX_CANDIDATES] = {};
|
|
|
|
|
|
|
|
uint32_t locations_count = 0;
|
|
|
|
|
|
|
|
locations_count +=
|
|
|
|
|
|
|
|
srsran_pdcch_common_locations(&pdcch_tx, &locations[locations_count], SRSRAN_MAX_CANDIDATES_COM, cfi);
|
|
|
|
|
|
|
|
locations_count +=
|
|
|
|
|
|
|
|
srsran_pdcch_ue_locations(&pdcch_tx, &dl_sf_cfg, &locations[locations_count], SRSRAN_MAX_CANDIDATES_UE, rnti);
|
|
|
|
|
|
|
|
|
|
|
|
nof_re = SRSRAN_CP_NORM_NSYMB * cell.nof_prb * SRSRAN_NRE;
|
|
|
|
// Iterate all possible locations
|
|
|
|
|
|
|
|
for (uint32_t loc = 0; loc < locations_count; loc++) {
|
|
|
|
|
|
|
|
srsran_dci_msg_t dci_tx = {};
|
|
|
|
|
|
|
|
dci_tx.nof_bits = srsran_dci_format_sizeof(&pdcch_tx.cell, &dl_sf_cfg, &dci_cfg, format);
|
|
|
|
|
|
|
|
dci_tx.location = locations[loc];
|
|
|
|
|
|
|
|
dci_tx.format = format;
|
|
|
|
|
|
|
|
dci_tx.rnti = rnti;
|
|
|
|
|
|
|
|
|
|
|
|
if (test_dci_payload_size()) {
|
|
|
|
// Assert DCI size
|
|
|
|
exit(-1);
|
|
|
|
TESTASSERT(assert_payload_size(&pdcch_tx.cell, &dl_sf_cfg) == SRSRAN_SUCCESS);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Initialise resource grid for each Tx port
|
|
|
|
|
|
|
|
for (uint32_t p = 0; p < nof_ports; p++) {
|
|
|
|
|
|
|
|
srsran_vec_cf_zero(slot_symbols[p], nof_re);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* init memory */
|
|
|
|
// Generate Tx DCI
|
|
|
|
|
|
|
|
srsran_random_bit_vector(random_gen, dci_tx.payload, dci_tx.nof_bits);
|
|
|
|
|
|
|
|
|
|
|
|
srsran_chest_dl_res_init(&chest_dl_res, cell.nof_prb);
|
|
|
|
// Encode
|
|
|
|
srsran_chest_dl_res_set_identity(&chest_dl_res);
|
|
|
|
gettimeofday(&t[1], NULL);
|
|
|
|
|
|
|
|
TESTASSERT(srsran_pdcch_encode(&pdcch_tx, &dl_sf_cfg, &dci_tx, slot_symbols) == SRSRAN_SUCCESS);
|
|
|
|
|
|
|
|
gettimeofday(&t[2], NULL);
|
|
|
|
|
|
|
|
get_time_interval(t);
|
|
|
|
|
|
|
|
t_encode_us += (size_t)(t[0].tv_sec * 1e6 + t[0].tv_usec);
|
|
|
|
|
|
|
|
t_encode_count++;
|
|
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < SRSRAN_MAX_PORTS; i++) {
|
|
|
|
// Apply AWGN
|
|
|
|
slot_symbols[i] = srsran_vec_cf_malloc(nof_re);
|
|
|
|
for (uint32_t p = 0; p < nof_ports; p++) {
|
|
|
|
if (!slot_symbols[i]) {
|
|
|
|
srsran_channel_awgn_run_c(&awgn, slot_symbols[p], slot_symbols[p], nof_re);
|
|
|
|
perror("malloc");
|
|
|
|
|
|
|
|
exit(-1);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
srsran_vec_cf_zero(slot_symbols[i], nof_re);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (srsran_regs_init(®s, cell)) {
|
|
|
|
// Extract LLR
|
|
|
|
ERROR("Error initiating regs");
|
|
|
|
gettimeofday(&t[1], NULL);
|
|
|
|
exit(-1);
|
|
|
|
TESTASSERT(srsran_pdcch_extract_llr(&pdcch_rx, &dl_sf_cfg, &chest_dl_res, slot_symbols) == SRSRAN_SUCCESS);
|
|
|
|
}
|
|
|
|
gettimeofday(&t[2], NULL);
|
|
|
|
|
|
|
|
get_time_interval(t);
|
|
|
|
|
|
|
|
t_llr_us += (size_t)(t[0].tv_sec * 1e6 + t[0].tv_usec);
|
|
|
|
|
|
|
|
|
|
|
|
if (srsran_pdcch_init_enb(&pdcch_tx, cell.nof_prb)) {
|
|
|
|
// Try decoding the PDCCH in all possible locations
|
|
|
|
ERROR("Error creating PDCCH object");
|
|
|
|
for (uint32_t loc_rx = 0; loc_rx < locations_count; loc_rx++) {
|
|
|
|
exit(-1);
|
|
|
|
if (!false_check && loc_rx != loc) {
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
if (srsran_pdcch_set_cell(&pdcch_tx, ®s, cell)) {
|
|
|
|
|
|
|
|
ERROR("Error setting cell in PDCCH object");
|
|
|
|
|
|
|
|
exit(-1);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (srsran_pdcch_init_ue(&pdcch_rx, cell.nof_prb, nof_rx_ant)) {
|
|
|
|
// Prepare DCI message context
|
|
|
|
ERROR("Error creating PDCCH object");
|
|
|
|
srsran_dci_msg_t dci_rx = {};
|
|
|
|
exit(-1);
|
|
|
|
dci_rx.location = locations[loc_rx];
|
|
|
|
}
|
|
|
|
dci_rx.format = format;
|
|
|
|
if (srsran_pdcch_set_cell(&pdcch_rx, ®s, cell)) {
|
|
|
|
|
|
|
|
ERROR("Error setting cell in PDCCH object");
|
|
|
|
|
|
|
|
exit(-1);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Resource allocate init */
|
|
|
|
// Try to decode PDCCH message
|
|
|
|
nof_dcis = 0;
|
|
|
|
gettimeofday(&t[1], NULL);
|
|
|
|
srsran_dci_dl_t dci;
|
|
|
|
TESTASSERT(srsran_pdcch_decode_msg(&pdcch_rx, &dl_sf_cfg, &dci_cfg, &dci_rx) == SRSRAN_SUCCESS);
|
|
|
|
ZERO_OBJECT(dci);
|
|
|
|
gettimeofday(&t[2], NULL);
|
|
|
|
dci.pid = 5;
|
|
|
|
get_time_interval(t);
|
|
|
|
dci.tb[0].mcs_idx = 5;
|
|
|
|
t_decode_us += (size_t)(t[0].tv_sec * 1e6 + t[0].tv_usec);
|
|
|
|
dci.tb[0].ndi = 0;
|
|
|
|
t_decode_count++;
|
|
|
|
dci.tb[0].rv = 1;
|
|
|
|
|
|
|
|
dci.alloc_type = SRSRAN_RA_ALLOC_TYPE0;
|
|
|
|
|
|
|
|
dci.type0_alloc.rbg_bitmask = 0x5;
|
|
|
|
|
|
|
|
dci.cif_present = dci_cfg.cif_enabled;
|
|
|
|
|
|
|
|
if (dci_cfg.cif_enabled) {
|
|
|
|
|
|
|
|
dci.cif = (uint32_t)srsran_random_uniform_int_dist(random_gen, 0, 7);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Format 1 Test case */
|
|
|
|
bool rnti_match = (dci_tx.rnti == dci_rx.rnti);
|
|
|
|
if (cell.nof_ports == 1) {
|
|
|
|
bool location_match = (loc == loc_rx);
|
|
|
|
testcases[nof_dcis].dci_format = SRSRAN_DCI_FORMAT1;
|
|
|
|
bool payload_match = (memcmp(dci_tx.payload, dci_rx.payload, dci_tx.nof_bits) == 0);
|
|
|
|
if (dci_cfg.cif_enabled) {
|
|
|
|
|
|
|
|
dci.cif = (uint32_t)srsran_random_uniform_int_dist(random_gen, 0, 7);
|
|
|
|
// Skip location if the decoding is not successful in a different location than transmitted
|
|
|
|
}
|
|
|
|
if (!location_match && !rnti_match) {
|
|
|
|
testcases[nof_dcis].ra_dl_tx = dci;
|
|
|
|
continue;
|
|
|
|
nof_dcis++;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Format 1 Test case */
|
|
|
|
|
|
|
|
dci.tb[0].mcs_idx = 15;
|
|
|
|
|
|
|
|
testcases[nof_dcis].dci_format = SRSRAN_DCI_FORMAT1;
|
|
|
|
|
|
|
|
if (dci_cfg.cif_enabled) {
|
|
|
|
|
|
|
|
dci.cif = (uint32_t)srsran_random_uniform_int_dist(random_gen, 0, 7);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
testcases[nof_dcis].ra_dl_tx = dci;
|
|
|
|
|
|
|
|
nof_dcis++;
|
|
|
|
if (srsran_verbose >= SRSRAN_VERBOSE_INFO || !payload_match) {
|
|
|
|
|
|
|
|
print_dci_msg("Tx: ", &dci_tx);
|
|
|
|
|
|
|
|
print_dci_msg("Rx: ", &dci_rx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Tx Diversity Test case */
|
|
|
|
// Assert received message
|
|
|
|
if (cell.nof_ports > 1) {
|
|
|
|
TESTASSERT(payload_match);
|
|
|
|
dci.tb[1].mcs_idx = 13;
|
|
|
|
|
|
|
|
dci.tb[1].rv = 3;
|
|
|
|
|
|
|
|
dci.tb[1].ndi = true;
|
|
|
|
|
|
|
|
testcases[nof_dcis].dci_format = SRSRAN_DCI_FORMAT2A;
|
|
|
|
|
|
|
|
if (dci_cfg.cif_enabled) {
|
|
|
|
|
|
|
|
dci.cif = (uint32_t)srsran_random_uniform_int_dist(random_gen, 0, 7);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
testcases[nof_dcis].ra_dl_tx = dci;
|
|
|
|
|
|
|
|
nof_dcis++;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* CDD Spatial Multiplexing Test case */
|
|
|
|
|
|
|
|
if (cell.nof_ports > 1) {
|
|
|
|
|
|
|
|
dci.tb[1].mcs_idx = 28;
|
|
|
|
|
|
|
|
dci.tb[1].rv = 1;
|
|
|
|
|
|
|
|
dci.tb[1].ndi = false;
|
|
|
|
|
|
|
|
testcases[nof_dcis].dci_format = SRSRAN_DCI_FORMAT2;
|
|
|
|
|
|
|
|
if (dci_cfg.cif_enabled) {
|
|
|
|
|
|
|
|
dci.cif = (uint32_t)srsran_random_uniform_int_dist(random_gen, 0, 7);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
testcases[nof_dcis].ra_dl_tx = dci;
|
|
|
|
|
|
|
|
nof_dcis++;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
srsran_dl_sf_cfg_t dl_sf;
|
|
|
|
printf("test_case_1 - %.1f usec/encode; %.1f usec/llr; %.1f usec/decode;\n",
|
|
|
|
ZERO_OBJECT(dl_sf);
|
|
|
|
(double)t_encode_us / (double)(t_encode_count),
|
|
|
|
dl_sf.cfi = cfi;
|
|
|
|
(double)t_llr_us / (double)(t_encode_count),
|
|
|
|
|
|
|
|
(double)t_decode_us / (double)(t_decode_count));
|
|
|
|
for (int s = 0; s < 10; s++) {
|
|
|
|
|
|
|
|
dl_sf.tti = s;
|
|
|
|
|
|
|
|
printf("Encoding %d DCIs for sf_idx=%d\n", nof_dcis, s);
|
|
|
|
|
|
|
|
/* Execute Rx */
|
|
|
|
|
|
|
|
for (i = 0; i < nof_dcis; i++) {
|
|
|
|
|
|
|
|
testcases[i].ra_dl_tx.rnti = (uint16_t)(1234 + i);
|
|
|
|
|
|
|
|
testcases[i].ra_dl_tx.format = testcases[i].dci_format;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
srsran_dci_msg_pack_pdsch(&cell, &dl_sf, &dci_cfg, &testcases[i].ra_dl_tx, &testcases[i].dci_tx);
|
|
|
|
return SRSRAN_SUCCESS;
|
|
|
|
srsran_dci_location_set(&testcases[i].dci_location, 0, (uint32_t)i);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
testcases[i].dci_tx.format = testcases[i].dci_format;
|
|
|
|
int main(int argc, char** argv)
|
|
|
|
testcases[i].dci_tx.location = testcases[i].dci_location;
|
|
|
|
{
|
|
|
|
|
|
|
|
srsran_regs_t regs;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
int ret = SRSRAN_ERROR;
|
|
|
|
|
|
|
|
|
|
|
|
// Enable just 1 TB per default
|
|
|
|
parse_args(argc, argv);
|
|
|
|
if (testcases[i].dci_format < SRSRAN_DCI_FORMAT2) {
|
|
|
|
random_gen = srsran_random_init(0x1234);
|
|
|
|
for (int j = 1; j < SRSRAN_MAX_CODEWORDS; j++) {
|
|
|
|
|
|
|
|
SRSRAN_DCI_TB_DISABLE(testcases[i].ra_dl_tx.tb[j]);
|
|
|
|
// Create cell
|
|
|
|
}
|
|
|
|
srsran_cell_t cell = {};
|
|
|
|
|
|
|
|
cell.nof_prb = nof_prb;
|
|
|
|
|
|
|
|
cell.nof_ports = nof_ports;
|
|
|
|
|
|
|
|
cell.cp = SRSRAN_CP_NORM;
|
|
|
|
|
|
|
|
cell.phich_resources = SRSRAN_PHICH_R_1;
|
|
|
|
|
|
|
|
cell.phich_length = SRSRAN_PHICH_NORM;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Initialise channel estimates with identity matrix
|
|
|
|
|
|
|
|
if (srsran_chest_dl_res_init(&chest_dl_res, cell.nof_prb) < SRSRAN_SUCCESS) {
|
|
|
|
|
|
|
|
ERROR("Error channel estimates");
|
|
|
|
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
srsran_chest_dl_res_set_identity(&chest_dl_res);
|
|
|
|
|
|
|
|
|
|
|
|
if (srsran_pdcch_encode(&pdcch_tx, &dl_sf, &testcases[i].dci_tx, slot_symbols)) {
|
|
|
|
// Allocate grid
|
|
|
|
ERROR("Error encoding DCI message");
|
|
|
|
uint32_t nof_re = SRSRAN_NOF_RE(cell);
|
|
|
|
|
|
|
|
for (i = 0; i < SRSRAN_MAX_PORTS; i++) {
|
|
|
|
|
|
|
|
slot_symbols[i] = srsran_vec_cf_malloc(nof_re);
|
|
|
|
|
|
|
|
if (slot_symbols[i] == NULL) {
|
|
|
|
|
|
|
|
ERROR("malloc");
|
|
|
|
goto quit;
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Execute 'Rx' */
|
|
|
|
if (srsran_regs_init(®s, cell)) {
|
|
|
|
if (srsran_pdcch_extract_llr(&pdcch_rx, &dl_sf, &chest_dl_res, slot_symbols)) {
|
|
|
|
ERROR("Error initiating regs");
|
|
|
|
ERROR("Error extracting LLRs");
|
|
|
|
|
|
|
|
goto quit;
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Decode DCIs */
|
|
|
|
if (srsran_pdcch_init_enb(&pdcch_tx, cell.nof_prb)) {
|
|
|
|
for (i = 0; i < nof_dcis; i++) {
|
|
|
|
ERROR("Error creating PDCCH object");
|
|
|
|
testcases[i].dci_rx.format = testcases[i].dci_format;
|
|
|
|
|
|
|
|
testcases[i].dci_rx.location = testcases[i].dci_location;
|
|
|
|
|
|
|
|
if (srsran_pdcch_decode_msg(&pdcch_rx, &dl_sf, &dci_cfg, &testcases[i].dci_rx)) {
|
|
|
|
|
|
|
|
ERROR("Error decoding DCI message");
|
|
|
|
|
|
|
|
goto quit;
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (srsran_dci_msg_unpack_pdsch(&cell, &dl_sf, &dci_cfg, &testcases[i].dci_rx, &testcases[i].ra_dl_rx)) {
|
|
|
|
if (srsran_pdcch_set_cell(&pdcch_tx, ®s, cell)) {
|
|
|
|
ERROR("Error unpacking DCI message");
|
|
|
|
ERROR("Error setting cell in PDCCH object");
|
|
|
|
goto quit;
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (testcases[i].dci_rx.rnti >= 1234 && testcases[i].dci_rx.rnti < 1234 + nof_dcis) {
|
|
|
|
|
|
|
|
testcases[i].dci_rx.rnti -= 1234;
|
|
|
|
if (srsran_pdcch_init_ue(&pdcch_rx, cell.nof_prb, nof_ports)) {
|
|
|
|
} else {
|
|
|
|
ERROR("Error creating PDCCH object");
|
|
|
|
printf("Received invalid DCI CRC %d\n", testcases[i].dci_rx.rnti);
|
|
|
|
|
|
|
|
goto quit;
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Compare Tx and Rx */
|
|
|
|
if (srsran_pdcch_set_cell(&pdcch_rx, ®s, cell)) {
|
|
|
|
for (i = 0; i < nof_dcis; i++) {
|
|
|
|
ERROR("Error setting cell in PDCCH object");
|
|
|
|
if (memcmp(testcases[i].dci_tx.payload, testcases[i].dci_rx.payload, testcases[i].dci_tx.nof_bits)) {
|
|
|
|
|
|
|
|
printf("Error in DCI %d: Received data does not match\n", i);
|
|
|
|
|
|
|
|
goto quit;
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#if SRSRAN_DCI_HEXDEBUG
|
|
|
|
|
|
|
|
// Ignore Hex str
|
|
|
|
if (srsran_channel_awgn_init(&awgn, 0x1234) < SRSRAN_SUCCESS) {
|
|
|
|
bzero(testcases[i].ra_dl_rx.hex_str, sizeof(testcases[i].ra_dl_rx.hex_str));
|
|
|
|
ERROR("Error init AWGN");
|
|
|
|
testcases[i].ra_dl_rx.nof_bits = 0;
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Ignore DCI location
|
|
|
|
|
|
|
|
testcases[i].ra_dl_rx.location = testcases[i].ra_dl_tx.location;
|
|
|
|
|
|
|
|
// Ignore cw_idx
|
|
|
|
|
|
|
|
for (int j = 0; j < SRSRAN_MAX_CODEWORDS; j++) {
|
|
|
|
|
|
|
|
testcases[i].ra_dl_rx.tb[j].cw_idx = testcases[i].ra_dl_tx.tb[j].cw_idx;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memcmp(&testcases[i].ra_dl_tx, &testcases[i].ra_dl_rx, sizeof(srsran_dci_dl_t))) {
|
|
|
|
|
|
|
|
uint8_t* x = (uint8_t*)&testcases[i].ra_dl_rx;
|
|
|
|
|
|
|
|
uint8_t* y = (uint8_t*)&testcases[i].ra_dl_tx;
|
|
|
|
|
|
|
|
for (int j = 0; j < sizeof(srsran_dci_dl_t); j++) {
|
|
|
|
|
|
|
|
if (x[j] != y[j]) {
|
|
|
|
|
|
|
|
printf("error in byte %d, rx=%d, tx=%d\n", j, x[j], y[j]);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("tx: ");
|
|
|
|
|
|
|
|
srsran_vec_fprint_byte(stdout, (uint8_t*)&testcases[i].ra_dl_tx, sizeof(srsran_dci_dl_t));
|
|
|
|
|
|
|
|
printf("rx: ");
|
|
|
|
|
|
|
|
srsran_vec_fprint_byte(stdout, (uint8_t*)&testcases[i].ra_dl_rx, sizeof(srsran_dci_dl_t));
|
|
|
|
|
|
|
|
printf("Error in RA %d: Received data does not match\n", i);
|
|
|
|
|
|
|
|
printf(" Field | Tx | Rx \n");
|
|
|
|
|
|
|
|
printf("--------------+----------+----------\n");
|
|
|
|
|
|
|
|
if (testcases[i].ra_dl_tx.cif) {
|
|
|
|
|
|
|
|
printf(" cif | %8d | %8d\n", testcases[i].ra_dl_tx.cif, testcases[i].ra_dl_rx.cif);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
printf(" harq_process | %8d | %8d\n", testcases[i].ra_dl_tx.pid, testcases[i].ra_dl_rx.pid);
|
|
|
|
|
|
|
|
printf(" mcs_idx | %8d | %8d\n", testcases[i].ra_dl_tx.tb[0].mcs_idx, testcases[i].ra_dl_rx.tb[0].mcs_idx);
|
|
|
|
|
|
|
|
printf(" rv_idx | %8d | %8d\n", testcases[i].ra_dl_tx.tb[0].rv, testcases[i].ra_dl_rx.tb[0].rv);
|
|
|
|
|
|
|
|
printf(" ndi | %8d | %8d\n", testcases[i].ra_dl_tx.tb[0].ndi, testcases[i].ra_dl_rx.tb[0].ndi);
|
|
|
|
|
|
|
|
printf(" mcs_idx_1 | %8d | %8d\n", testcases[i].ra_dl_tx.tb[1].mcs_idx, testcases[i].ra_dl_rx.tb[1].mcs_idx);
|
|
|
|
|
|
|
|
printf(" rv_idx_1 | %8d | %8d\n", testcases[i].ra_dl_tx.tb[1].rv, testcases[i].ra_dl_rx.tb[1].rv);
|
|
|
|
|
|
|
|
printf(" ndi_1 | %8d | %8d\n", testcases[i].ra_dl_tx.tb[1].ndi, testcases[i].ra_dl_rx.tb[1].ndi);
|
|
|
|
|
|
|
|
printf(" tb_cw_swap | %8d | %8d\n", testcases[i].ra_dl_tx.tb_cw_swap, testcases[i].ra_dl_rx.tb_cw_swap);
|
|
|
|
|
|
|
|
printf(" sram_id | %8d | %8d\n", testcases[i].ra_dl_tx.sram_id, testcases[i].ra_dl_rx.sram_id);
|
|
|
|
|
|
|
|
printf(" pinfo | %8d | %8d\n", testcases[i].ra_dl_tx.pinfo, testcases[i].ra_dl_rx.pinfo);
|
|
|
|
|
|
|
|
printf(" pconf | %8d | %8d\n", testcases[i].ra_dl_tx.pconf, testcases[i].ra_dl_rx.pconf);
|
|
|
|
|
|
|
|
printf(" power_offset | %8d | %8d\n", testcases[i].ra_dl_tx.power_offset, testcases[i].ra_dl_rx.power_offset);
|
|
|
|
|
|
|
|
printf(" tpc_pucch | %8d | %8d\n", testcases[i].ra_dl_tx.tpc_pucch, testcases[i].ra_dl_rx.tpc_pucch);
|
|
|
|
|
|
|
|
goto quit;
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (srsran_channel_awgn_set_n0(&awgn, -snr_dB) < SRSRAN_SUCCESS) {
|
|
|
|
|
|
|
|
ERROR("Error setting n0");
|
|
|
|
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Execute actual test cases
|
|
|
|
|
|
|
|
if (test_case1() < SRSRAN_SUCCESS) {
|
|
|
|
|
|
|
|
ERROR("Test case 1 failed");
|
|
|
|
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
|
|
|
|
ret = SRSRAN_SUCCESS;
|
|
|
|
|
|
|
|
|
|
|
|
quit:
|
|
|
|
quit:
|
|
|
|
srsran_pdcch_free(&pdcch_tx);
|
|
|
|
srsran_pdcch_free(&pdcch_tx);
|
|
|
@ -390,10 +343,13 @@ quit:
|
|
|
|
srsran_chest_dl_res_free(&chest_dl_res);
|
|
|
|
srsran_chest_dl_res_free(&chest_dl_res);
|
|
|
|
srsran_regs_free(®s);
|
|
|
|
srsran_regs_free(®s);
|
|
|
|
srsran_random_free(random_gen);
|
|
|
|
srsran_random_free(random_gen);
|
|
|
|
|
|
|
|
srsran_channel_awgn_free(&awgn);
|
|
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < SRSRAN_MAX_PORTS; i++) {
|
|
|
|
for (i = 0; i < SRSRAN_MAX_PORTS; i++) {
|
|
|
|
|
|
|
|
if (slot_symbols[i]) {
|
|
|
|
free(slot_symbols[i]);
|
|
|
|
free(slot_symbols[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
if (ret) {
|
|
|
|
if (ret) {
|
|
|
|
printf("Error\n");
|
|
|
|
printf("Error\n");
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|