|
|
@ -40,21 +40,6 @@ static void* reader_thread(void* arg)
|
|
|
|
|
|
|
|
|
|
|
|
switch (rx_status) {
|
|
|
|
switch (rx_status) {
|
|
|
|
case skiq_rx_status_success:
|
|
|
|
case skiq_rx_status_success:
|
|
|
|
// Check Rx index boundary
|
|
|
|
|
|
|
|
if (p_rx_block->rfic_control >= q->param.rx_param->gain_index_min &&
|
|
|
|
|
|
|
|
p_rx_block->rfic_control <= q->param.rx_param->gain_index_max) {
|
|
|
|
|
|
|
|
double new_rx_gain = q->rx_gain_table_db[p_rx_block->rfic_control];
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// If the Rx index has changed, update gain
|
|
|
|
|
|
|
|
if (new_rx_gain != q->cur_rx_gain_db) {
|
|
|
|
|
|
|
|
SKIQ_RF_DEBUG("card %d index=%d; gain=%.2f/%.2f dB;\n",
|
|
|
|
|
|
|
|
q->card,
|
|
|
|
|
|
|
|
p_rx_block->rfic_control,
|
|
|
|
|
|
|
|
q->rx_gain_table_db[p_rx_block->rfic_control],
|
|
|
|
|
|
|
|
q->cur_rx_gain_db);
|
|
|
|
|
|
|
|
q->cur_rx_gain_db = new_rx_gain;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (curr_rx_hdl < q->nof_ports && p_rx_block != NULL && len > SKIQ_RX_HEADER_SIZE_IN_BYTES) {
|
|
|
|
if (curr_rx_hdl < q->nof_ports && p_rx_block != NULL && len > SKIQ_RX_HEADER_SIZE_IN_BYTES) {
|
|
|
|
// Convert number of bytes into samples
|
|
|
|
// Convert number of bytes into samples
|
|
|
|
uint32_t nsamples = len / 4 - SKIQ_RX_HEADER_SIZE_IN_WORDS;
|
|
|
|
uint32_t nsamples = len / 4 - SKIQ_RX_HEADER_SIZE_IN_WORDS;
|
|
|
@ -94,17 +79,6 @@ static void* reader_thread(void* arg)
|
|
|
|
return NULL;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int rf_skiq_card_update_gain_table(rf_skiq_card_t* q)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
for (uint8_t i = q->param.rx_param->gain_index_min; i <= q->param.rx_param->gain_index_max; i++) {
|
|
|
|
|
|
|
|
if (skiq_read_rx_cal_offset_by_gain_index(q->card, skiq_rx_hdl_A1, i, &q->rx_gain_table_db[i])) {
|
|
|
|
|
|
|
|
ERROR("Reading calibrated Rx gain index %d", i);
|
|
|
|
|
|
|
|
return SRSRAN_ERROR;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return SRSRAN_SUCCESS;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int rf_skiq_card_init(rf_skiq_card_t* q, uint8_t card, uint8_t nof_ports, const rf_skiq_port_opts_t* opts)
|
|
|
|
int rf_skiq_card_init(rf_skiq_card_t* q, uint8_t card, uint8_t nof_ports, const rf_skiq_port_opts_t* opts)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
q->card = card;
|
|
|
|
q->card = card;
|
|
|
@ -283,29 +257,33 @@ double rf_skiq_card_set_tx_gain_db(rf_skiq_card_t* q, uint32_t port_idx, double
|
|
|
|
|
|
|
|
|
|
|
|
double rf_skiq_card_set_rx_gain_db(rf_skiq_card_t* q, uint32_t port_idx, double gain_db)
|
|
|
|
double rf_skiq_card_set_rx_gain_db(rf_skiq_card_t* q, uint32_t port_idx, double gain_db)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// Find the nearest gain index in the table
|
|
|
|
// From Sidekiq API doc:
|
|
|
|
int gain_idx = -1;
|
|
|
|
//
|
|
|
|
double gain_min_diff = INFINITY;
|
|
|
|
// For Sidekiq mPCIe (skiq_mpcie), Sidekiq M.2 (skiq_m2), Sidekiq Stretch (skiq_m2_2280), Sidekiq Z2
|
|
|
|
for (int i = q->param.rx_param->gain_index_min; i <= q->param.rx_param->gain_index_max; i++) {
|
|
|
|
//(skiq_z2), and Matchstiq Z3u (skiq_z3u) each increment of the gain index value results in approxi-
|
|
|
|
double gain_diff = fabs(q->rx_gain_table_db[i] - gain_db);
|
|
|
|
// mately 1 dB of gain, with approximately 76 dB of total gain available. For details on the gain table,
|
|
|
|
if (gain_diff < gain_min_diff) {
|
|
|
|
// refer to p. 37 of AD9361 Reference Manual UG-570
|
|
|
|
gain_min_diff = gain_diff;
|
|
|
|
|
|
|
|
gain_idx = i;
|
|
|
|
// Check gain range
|
|
|
|
}
|
|
|
|
if (gain_db < q->param.rx_param->gain_index_min || gain_db > q->param.rx_param->gain_index_max) {
|
|
|
|
|
|
|
|
ERROR("Error port %d:%d the selected gain (%.2f dB) is out of range (%d to %d dB).\n",
|
|
|
|
|
|
|
|
q->card,
|
|
|
|
|
|
|
|
port_idx,
|
|
|
|
|
|
|
|
gain_db,
|
|
|
|
|
|
|
|
q->param.rx_param->gain_index_min,
|
|
|
|
|
|
|
|
q->param.rx_param->gain_index_max);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (gain_idx >= 0) {
|
|
|
|
// Calculate attenuation index
|
|
|
|
gain_db = q->rx_gain_table_db[gain_idx];
|
|
|
|
uint16_t gain_idx = (uint16_t)floor(gain_db);
|
|
|
|
if (port_idx < q->nof_ports) {
|
|
|
|
|
|
|
|
// Set single port gain
|
|
|
|
if (port_idx < q->nof_ports) {
|
|
|
|
q->issued_rx_gain_db[port_idx] = gain_db;
|
|
|
|
// Set single port gain
|
|
|
|
skiq_write_rx_gain(q->card, (skiq_rx_hdl_t)port_idx, gain_idx);
|
|
|
|
skiq_write_rx_gain(q->card, (skiq_rx_hdl_t)port_idx, gain_idx);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
// Set all gains
|
|
|
|
// Set all gains
|
|
|
|
for (int i = 0; i < q->nof_ports; i++) {
|
|
|
|
for (int i = 0; i < q->nof_ports; i++) {
|
|
|
|
q->issued_rx_gain_db[i] = gain_db;
|
|
|
|
skiq_write_rx_gain(q->card, (skiq_rx_hdl_t)i, gain_idx);
|
|
|
|
skiq_write_rx_gain(q->card, (skiq_rx_hdl_t)i, gain_idx);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -511,16 +489,6 @@ double rf_skiq_card_set_rx_freq_hz(rf_skiq_card_t* q, uint32_t port_idx, double
|
|
|
|
q->suspend = true;
|
|
|
|
q->suspend = true;
|
|
|
|
rf_skiq_rx_port_set_lo(&q->rx_ports[port_idx], (uint64_t)freq_hz);
|
|
|
|
rf_skiq_rx_port_set_lo(&q->rx_ports[port_idx], (uint64_t)freq_hz);
|
|
|
|
q->suspend = false;
|
|
|
|
q->suspend = false;
|
|
|
|
|
|
|
|
|
|
|
|
// Update gains for only port 0
|
|
|
|
|
|
|
|
if (port_idx == 0) {
|
|
|
|
|
|
|
|
// Update gain table
|
|
|
|
|
|
|
|
rf_skiq_card_update_gain_table(q);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Set previous issued gain in dB for the new tables
|
|
|
|
|
|
|
|
rf_skiq_card_set_rx_gain_db(q, q->nof_ports, q->issued_rx_gain_db[port_idx]);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return freq_hz;
|
|
|
|
return freq_hz;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|