diff --git a/lib/include/srslte/phy/rf/rf.h b/lib/include/srslte/phy/rf/rf.h index ab83bdd49..4ef9c76dd 100644 --- a/lib/include/srslte/phy/rf/rf.h +++ b/lib/include/srslte/phy/rf/rf.h @@ -29,6 +29,8 @@ #include "srslte/config.h" +#define RF_PARAM_LEN (256) + typedef struct { void* handler; void* dev; diff --git a/lib/src/phy/rf/rf_helper.h b/lib/src/phy/rf/rf_helper.h index b86cdb772..ebce5373a 100644 --- a/lib/src/phy/rf/rf_helper.h +++ b/lib/src/phy/rf/rf_helper.h @@ -21,6 +21,10 @@ // A bunch of helper functions to process device arguments +#include "srslte/config.h" +#include "srslte/phy/rf/rf.h" +#include +#include #include #define REMOVE_SUBSTRING_WITHCOMAS(S, TOREMOVE) \ @@ -47,4 +51,62 @@ static inline void copy_subdev_string(char* dst, char* src) n++; } dst[n] = '\0'; +} + +static inline int parse_string(char* args, const char* config_arg_base, int channel_index, char param_dst[RF_PARAM_LEN]) +{ + int ret = SRSLTE_ERROR; + + char config_key[RF_PARAM_LEN] = {0}; + char config_str[RF_PARAM_LEN] = {0}; + char* config_ptr = NULL; + + // try to parse parameter without index as is + snprintf(config_key, RF_PARAM_LEN, "%s=", config_arg_base); + config_ptr = strstr(args, config_key); + + // check if we have a match + if (!config_ptr) { + // Couldn't find param, add channel index and parse again + snprintf(config_key, RF_PARAM_LEN, "%s%d=", config_arg_base, channel_index); + config_ptr = strstr(args, config_key); + } + + if (config_ptr) { + copy_subdev_string(config_str, config_ptr + strlen(config_key)); + if (channel_index >= 0) { + printf("CH%d %s=%s\n", channel_index, config_arg_base, config_str); + } else { + printf("CHx %s=%s\n", config_arg_base, config_str); + } + + strncpy(param_dst, config_str, RF_PARAM_LEN); + param_dst[RF_PARAM_LEN - 1] = 0; + + // concatenate key=value and remove both (avoid removing the same value twice if it occurs twice in rf_args) + char config_pair[RF_PARAM_LEN * 2] = {0}; + snprintf(config_pair, RF_PARAM_LEN * 2, "%s%s", config_key, config_str); + remove_substring(args, config_pair); + + ret = SRSLTE_SUCCESS; + } + + return ret; +} + +static inline int parse_double(char* args, const char* config_arg_base, int channel_index, double* value) +{ + char tmp_value[RF_PARAM_LEN] = {0}; + int ret = parse_string(args, config_arg_base, channel_index, tmp_value); + *value = strtod(tmp_value, NULL); + return ret; +} + +static inline int parse_uint32(char* args, const char* config_arg_base, int channel_index, uint32_t* value) +{ + double tmp_value; + int ret = parse_double(args, config_arg_base, channel_index, &tmp_value); + *value = tmp_value; + + return ret; } \ No newline at end of file diff --git a/lib/src/phy/rf/rf_zmq_imp.c b/lib/src/phy/rf/rf_zmq_imp.c index cf2c8a252..395b7645f 100644 --- a/lib/src/phy/rf/rf_zmq_imp.c +++ b/lib/src/phy/rf/rf_zmq_imp.c @@ -44,17 +44,14 @@ typedef struct { double rx_gain; uint32_t tx_freq_mhz[SRSLTE_MAX_CHANNELS]; uint32_t rx_freq_mhz[SRSLTE_MAX_CHANNELS]; - bool tx_used; + bool tx_off; + char id[RF_PARAM_LEN]; // Server void* context; rf_zmq_tx_t transmitter[SRSLTE_MAX_CHANNELS]; rf_zmq_rx_t receiver[SRSLTE_MAX_CHANNELS]; - char rx_port[PARAM_LEN]; - char tx_port[PARAM_LEN]; - char id[PARAM_LEN_SHORT]; - // Various sample buffers cf_t* buffer_decimation[SRSLTE_MAX_CHANNELS]; cf_t* buffer_tx; @@ -194,20 +191,6 @@ int rf_zmq_open(char* args, void** h) return rf_zmq_open_multi(args, h, 1); } -static inline int parse_double(const char* args, const char* config_arg, double* value) -{ - int ret = SRSLTE_ERROR; - if (args && config_arg && value) { - char* config_ptr = strstr(args, config_arg); - - if (config_ptr) { - *value = strtod(config_ptr + strlen(config_arg), NULL); - ret = SRSLTE_SUCCESS; - } - } - return ret; -} - int rf_zmq_open_multi(char* args, void** h, uint32_t nof_channels) { int ret = SRSLTE_ERROR; @@ -250,104 +233,57 @@ int rf_zmq_open_multi(char* args, void** h, uint32_t nof_channels) // parse args if (args && strlen(args)) { // base_srate - { - const char config_arg[] = "base_srate="; - char config_str[PARAM_LEN] = {0}; - char* config_ptr = strstr(args, config_arg); - if (config_ptr) { - copy_subdev_string(config_str, config_ptr + strlen(config_arg)); - printf("Using base rate=%s\n", config_str); - handler->base_srate = (uint32_t)strtod(config_str, NULL); - remove_substring(args, config_arg); - remove_substring(args, config_str); - } - } + parse_uint32(args, "base_srate", -1, &handler->base_srate); // id - { - const char config_arg[] = "id="; - char config_str[PARAM_LEN_SHORT] = {0}; - char* config_ptr = strstr(args, config_arg); - if (config_ptr) { - copy_subdev_string(config_str, config_ptr + strlen(config_arg)); - printf("Using ID=%s\n", config_str); - strncpy(handler->id, config_str, PARAM_LEN_SHORT); - handler->id[PARAM_LEN_SHORT - 1] = 0; - remove_substring(args, config_arg); - remove_substring(args, config_str); - } - } + parse_string(args, "id", -1, handler->id); + // rx_type - { - const char config_arg[] = "rx_type="; - char config_str[PARAM_LEN_SHORT] = {0}; - char* config_ptr = strstr(args, config_arg); - if (config_ptr) { - copy_subdev_string(config_str, config_ptr + strlen(config_arg)); - if (!strcmp(config_str, "sub")) { - rx_opts.socket_type = ZMQ_SUB; - printf("Using ZMQ_SUB for rx socket\n"); - } else { - printf("Unsupported socket type %s. Using ZMQ_REQ for rx socket\n", config_str); - } - remove_substring(args, config_arg); - remove_substring(args, config_str); + char tmp[RF_PARAM_LEN] = {0}; + parse_string(args, "rx_type", -1, tmp); + if (strlen(tmp) > 0) { + if (!strcmp(tmp, "sub")) { + rx_opts.socket_type = ZMQ_SUB; + } else { + printf("Unsupported socket type %s\n", tmp); + goto clean_exit; } } + // rx_format - { - const char config_arg[] = "rx_format="; - char config_str[PARAM_LEN_SHORT] = {0}; - char* config_ptr = strstr(args, config_arg); - if (config_ptr) { - copy_subdev_string(config_str, config_ptr + strlen(config_arg)); - rx_opts.sample_format = ZMQ_TYPE_FC32; - if (!strcmp(config_str, "sc16")) { - rx_opts.sample_format = ZMQ_TYPE_SC16; - printf("Using sc16 format for rx socket\n"); - } else { - printf("Unsupported sample format %s. Using fc32 for rx socket\n", config_str); - } - remove_substring(args, config_arg); - remove_substring(args, config_str); + parse_string(args, "rx_format", -1, tmp); + rx_opts.sample_format = ZMQ_TYPE_FC32; + if (strlen(tmp) > 0) { + if (!strcmp(tmp, "sc16")) { + rx_opts.sample_format = ZMQ_TYPE_SC16; + } else { + printf("Unsupported sample format %s\n", tmp); + goto clean_exit; } } + // tx_type - { - const char config_arg[] = "tx_type="; - char config_str[PARAM_LEN_SHORT] = {0}; - char* config_ptr = strstr(args, config_arg); - if (config_ptr) { - copy_subdev_string(config_str, config_ptr + strlen(config_arg)); - if (!strcmp(config_str, "pub")) { - tx_opts.socket_type = ZMQ_PUB; - printf("Using ZMQ_PUB for tx socket\n"); - } else { - printf("Unsupported socket type %s. Using ZMQ_REP for tx socket\n", config_str); - } - remove_substring(args, config_arg); - remove_substring(args, config_str); + parse_string(args, "tx_type", -1, tmp); + if (strlen(tmp) > 0) { + if (!strcmp(tmp, "pub")) { + tx_opts.socket_type = ZMQ_PUB; + } else { + printf("Unsupported socket type %s\n", tmp); + goto clean_exit; } } + // tx_format - { - const char config_arg[] = "tx_format="; - char config_str[PARAM_LEN_SHORT] = {0}; - char* config_ptr = strstr(args, config_arg); - if (config_ptr) { - copy_subdev_string(config_str, config_ptr + strlen(config_arg)); - tx_opts.sample_format = ZMQ_TYPE_FC32; - if (!strcmp(config_str, "sc16")) { - tx_opts.sample_format = ZMQ_TYPE_SC16; - printf("Using sc16 format for tx socket\n"); - } else { - printf("Unsupported sample format %s. Using fc32 for tx socket\n", config_str); - } - remove_substring(args, config_arg); - remove_substring(args, config_str); + parse_string(args, "tx_format", -1, tmp); + tx_opts.sample_format = ZMQ_TYPE_FC32; + if (strlen(tmp) > 0) { + if (!strcmp(tmp, "sc16")) { + tx_opts.sample_format = ZMQ_TYPE_SC16; + } else { + printf("Unsupported sample format %s\n", tmp); + goto clean_exit; } } - } else { fprintf(stderr, "[zmq] Error: RF device args are required for ZMQ no-RF module\n"); goto clean_exit; @@ -363,104 +299,46 @@ int rf_zmq_open_multi(char* args, void** h, uint32_t nof_channels) } for (int i = 0; i < handler->nof_channels; i++) { - // rxport - { - char config_arg[PARAM_LEN] = "rx_port="; - char config_str[PARAM_LEN] = {0}; + char rx_port[RF_PARAM_LEN] = {}; + char tx_port[RF_PARAM_LEN] = {}; - if (i > 0) { - snprintf(config_arg, PARAM_LEN, "rx_port%d=", i + 1); - } - - char* config_ptr = strstr(args, config_arg); - - if (config_ptr) { - copy_subdev_string(config_str, config_ptr + strlen(config_arg)); - printf("Channel %d. Using rx_port=%s\n", i, config_str); - strncpy(handler->rx_port, config_str, PARAM_LEN); - handler->rx_port[PARAM_LEN - 1] = 0; - remove_substring(args, config_arg); - remove_substring(args, config_str); - } - } + // rx_port + parse_string(args, "rx_port", i, rx_port); // rx_freq - { - char config_arg[PARAM_LEN] = "rx_freq="; - if (i > 0) { - snprintf(config_arg, PARAM_LEN, "rx_freq%d=", i + 1); - } - - double freq = 0.0; - if (!parse_double(args, config_arg, &freq)) { - rx_opts.frequency_mhz = (uint32_t)(freq / 1e6); - printf("Channel %d. Using rx_freq=%dMHz\n", i, rx_opts.frequency_mhz); - } - } + double rx_freq = 0.0f; + parse_double(args, "rx_freq", i, &rx_freq); + rx_opts.frequency_mhz = (uint32_t)(rx_freq / 1e6); - // txport - { - char config_arg[PARAM_LEN] = "tx_port="; - char config_str[PARAM_LEN] = {0}; - - if (i > 0) { - snprintf(config_arg, PARAM_LEN, "tx_port%d=", i + 1); - } - - char* config_ptr = strstr(args, config_arg); - - if (config_ptr) { - copy_subdev_string(config_str, config_ptr + strlen(config_arg)); - printf("Channel %d. Using tx_port=%s\n", i, config_str); - strncpy(handler->tx_port, config_str, PARAM_LEN); - handler->tx_port[PARAM_LEN - 1] = 0; - remove_substring(args, config_arg); - remove_substring(args, config_str); - } - } + // tx_port + parse_string(args, "tx_port", i, tx_port); // tx_freq - { - char config_arg[PARAM_LEN] = "tx_freq="; - if (i > 0) { - snprintf(config_arg, PARAM_LEN, "tx_freq%d=", i + 1); - } - - double freq = 0.0; - if (!parse_double(args, config_arg, &freq)) { - tx_opts.frequency_mhz = (uint32_t)(freq / 1e6); - printf("Channel %d. Using tx_freq=%dMHz\n", i, tx_opts.frequency_mhz); - } - } + double tx_freq = 0.0f; + parse_double(args, "tx_freq", i, &tx_freq); + tx_opts.frequency_mhz = (uint32_t)(tx_freq / 1e6); // fail_on_disconnect - { - const char config_arg[] = "fail_on_disconnect="; - char config_str[PARAM_LEN_SHORT] = {0}; - char* config_ptr = strstr(args, config_arg); - if (config_ptr) { - copy_subdev_string(config_str, config_ptr + strlen(config_arg)); - if (strncmp(config_str, "true", PARAM_LEN_SHORT) == 0 || strncmp(config_str, "yes", PARAM_LEN_SHORT) == 0) { - rx_opts.fail_on_disconnect = true; - } - remove_substring(args, config_arg); - remove_substring(args, config_str); - } + char tmp[RF_PARAM_LEN] = {}; + parse_string(args, "fail_on_disconnect", i, tmp); + if (strncmp(tmp, "true", RF_PARAM_LEN) == 0 || strncmp(tmp, "yes", RF_PARAM_LEN) == 0) { + rx_opts.fail_on_disconnect = true; } // initialize transmitter - if (strlen(handler->tx_port) != 0) { - if (rf_zmq_tx_open(&handler->transmitter[i], tx_opts, handler->context, handler->tx_port) != SRSLTE_SUCCESS) { + if (strlen(tx_port) != 0) { + if (rf_zmq_tx_open(&handler->transmitter[i], tx_opts, handler->context, tx_port) != SRSLTE_SUCCESS) { fprintf(stderr, "[zmq] Error: opening transmitter\n"); goto clean_exit; } } else { fprintf(stdout, "[zmq] %s Tx port not specified. Disabling transmitter.\n", handler->id); + handler->tx_off = true; } // initialize receiver - if (strlen(handler->rx_port) != 0) { - if (rf_zmq_rx_open(&handler->receiver[i], rx_opts, handler->context, handler->rx_port) != SRSLTE_SUCCESS) { + if (strlen(rx_port) != 0) { + if (rf_zmq_rx_open(&handler->receiver[i], rx_opts, handler->context, rx_port) != SRSLTE_SUCCESS) { fprintf(stderr, "[zmq] Error: opening receiver\n"); goto clean_exit; } @@ -965,7 +843,7 @@ int rf_zmq_send_timed_multi(void* h, rf_zmq_info(handler->id, "Tx %d samples (%d B)\n", nsamples, nbytes); // return if transmitter is switched off - if (strlen(handler->tx_port) == 0) { + if (handler->tx_off) { return SRSLTE_SUCCESS; } @@ -1037,7 +915,6 @@ int rf_zmq_send_timed_multi(void* h, } } } - handler->tx_used = true; } ret = SRSLTE_SUCCESS; diff --git a/lib/src/phy/rf/rf_zmq_imp.h b/lib/src/phy/rf/rf_zmq_imp.h index 07df529f4..b313ceb64 100644 --- a/lib/src/phy/rf/rf_zmq_imp.h +++ b/lib/src/phy/rf/rf_zmq_imp.h @@ -26,8 +26,6 @@ #include "srslte/phy/rf/rf.h" #define DEVNAME_ZMQ "ZeroMQ" -#define PARAM_LEN (256) -#define PARAM_LEN_SHORT (PARAM_LEN / 2) SRSLTE_API int rf_zmq_open(char* args, void** handler); diff --git a/lib/src/phy/rf/rf_zmq_test.c b/lib/src/phy/rf/rf_zmq_test.c index 62babf095..ce43d1283 100644 --- a/lib/src/phy/rf/rf_zmq_test.c +++ b/lib/src/phy/rf/rf_zmq_test.c @@ -41,9 +41,9 @@ pthread_t rx_thread; void* ue_rx_thread_function(void* args) { - char rf_args[PARAM_LEN]; - strncpy(rf_args, (char*)args, PARAM_LEN - 1); - rf_args[PARAM_LEN - 1] = 0; + char rf_args[RF_PARAM_LEN]; + strncpy(rf_args, (char*)args, RF_PARAM_LEN - 1); + rf_args[RF_PARAM_LEN - 1] = 0; // sleep(1); @@ -73,9 +73,9 @@ void* ue_rx_thread_function(void* args) void enb_tx_function(const char* tx_args, bool timed_tx) { - char rf_args[PARAM_LEN]; - strncpy(rf_args, tx_args, PARAM_LEN - 1); - rf_args[PARAM_LEN - 1] = 0; + char rf_args[RF_PARAM_LEN]; + strncpy(rf_args, tx_args, RF_PARAM_LEN - 1); + rf_args[RF_PARAM_LEN - 1] = 0; printf("opening tx device with args=%s\n", rf_args); if (srslte_rf_open_devname(&enb_radio, "zmq", rf_args, NOF_RX_ANT)) { @@ -182,8 +182,52 @@ exit: return ret; } +int param_test(const char* args_param, const int num_channels) +{ + char rf_args[RF_PARAM_LEN] = {}; + strncpy(rf_args, (char*)args_param, RF_PARAM_LEN - 1); + rf_args[RF_PARAM_LEN - 1] = 0; + + printf("opening tx device with args=%s\n", rf_args); + if (srslte_rf_open_devname(&enb_radio, "zmq", rf_args, num_channels)) { + fprintf(stderr, "Error opening rf\n"); + return SRSLTE_ERROR; + } + + return SRSLTE_SUCCESS; +} + int main() { + // two Rx ports + if (param_test("rx_port=ipc://dl0,rx_port1=ipc://dl1", 2)) { + fprintf(stderr, "Param test failed!\n"); + return SRSLTE_ERROR; + } + + // multiple rx ports, no channel index provided + if (param_test("rx_port=ipc://dl0,rx_port=ipc://dl1,rx_port=ipc://dl2,rx_port=ipc://dl3,base_srate=1.92e6", 4)) { + fprintf(stderr, "Param test failed!\n"); + return SRSLTE_ERROR; + } + + // One Rx, one Tx and all generic options + if (param_test("rx_port0=tcp://" + "localhost:2000,rx_format=sc16,tx_format=sc16,tx_type=pub,rx_type=sub,base_srate=1.92e6,id=test", + 1)) { + fprintf(stderr, "Param test failed!\n"); + return SRSLTE_ERROR; + } + + // 1 port, 2 antennas, MIMO freq config + if (param_test( + "tx_port0=tcp://*:2001,tx_port1=tcp://*:2003,rx_port0=tcp://localhost:2000,rx_port1=tcp://" + "localhost:2002,id=ue,base_srate=23.04e6,tx_freq0=2510e6,tx_freq1=2510e6,rx_freq0=2630e6,,rx_freq1=2630e6", + 2)) { + fprintf(stderr, "Param test failed!\n"); + return SRSLTE_ERROR; + } + // single tx, single rx with continuous transmissions (no timed tx) using IPC transport if (run_test("rx_port=ipc://link1,id=ue,base_srate=1.92e6", "tx_port=ipc://link1,id=enb,base_srate=1.92e6", false) != SRSLTE_SUCCESS) { @@ -207,5 +251,5 @@ int main() return -1; } - return 0; + return SRSLTE_SUCCESS; }