From 6af6ac32c30d2dc0efa6eab61e4a3920649e3e62 Mon Sep 17 00:00:00 2001 From: Trung Tran Date: Fri, 3 Aug 2018 09:12:55 -0700 Subject: examples: add rfnoc_radio_loopback This example will allow an RF->RF loopback using RFNoC devices. --- host/examples/CMakeLists.txt | 1 + host/examples/rfnoc_radio_loopback.cpp | 213 +++++++++++++++++++++++++++++++++ 2 files changed, 214 insertions(+) create mode 100644 host/examples/rfnoc_radio_loopback.cpp (limited to 'host/examples') diff --git a/host/examples/CMakeLists.txt b/host/examples/CMakeLists.txt index 48b31bdfb..4b15f8ddf 100644 --- a/host/examples/CMakeLists.txt +++ b/host/examples/CMakeLists.txt @@ -35,6 +35,7 @@ IF(ENABLE_RFNOC) example_sources rfnoc_nullsource_ce_rx.cpp rfnoc_rx_to_file.cpp + rfnoc_radio_loopback.cpp ) ENDIF(ENABLE_RFNOC) diff --git a/host/examples/rfnoc_radio_loopback.cpp b/host/examples/rfnoc_radio_loopback.cpp new file mode 100644 index 000000000..a9a8232f2 --- /dev/null +++ b/host/examples/rfnoc_radio_loopback.cpp @@ -0,0 +1,213 @@ +// +// Copyright 2016 Ettus Research LLC +// Copyright 2018 Ettus Research, a National Instruments Company +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +// Example UHD/RFNoC application: Connect an rx radio to a tx radio and +// run a loopback. + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +namespace po = boost::program_options; +using uhd::rfnoc::radio_ctrl; +using uhd::rfnoc::block_ctrl; + +/**************************************************************************** + * SIGINT handling + ***************************************************************************/ +static bool stop_signal_called = false; +void sig_int_handler(int){stop_signal_called = true;} + +/**************************************************************************** + * main + ***************************************************************************/ +int UHD_SAFE_MAIN(int argc, char *argv[]){ + uhd::set_thread_priority_safe(); + + //variables to be set by po + std::string args, rx_args, tx_args, rx_ant, tx_ant, rx_blockid, tx_blockid, ref; + size_t total_num_samps, spp, rx_chan, tx_chan, tx_delay; + double rate, rx_freq, tx_freq, rx_gain, tx_gain, bw, total_time, setup_time; + bool rx_timestamps; + + //setup the program options + po::options_description desc("Allowed options"); + desc.add_options() + ("help", "help message") + ("args", po::value(&args)->default_value(""), "UHD device address args") + ("rx_args", po::value(&rx_args)->default_value(""), "Block args for the receive radio") + ("tx_args", po::value(&tx_args)->default_value(""), "Block args for the transmit radio") + ("spp", po::value(&spp)->default_value(0), "Samples per packet (reduce for lower latency)") + ("rx-freq", po::value(&rx_freq)->default_value(0.0), "Rx RF center frequency in Hz") + ("tx-freq", po::value(&tx_freq)->default_value(0.0), "Tx RF center frequency in Hz") + ("rx-gain", po::value(&rx_gain)->default_value(0.0), "Rx RF center gain in Hz") + ("tx-gain", po::value(&tx_gain)->default_value(0.0), "Tx RF center gain in Hz") + ("rx-ant", po::value(&rx_ant), "Receive antenna selection") + ("tx-ant", po::value(&tx_ant), "Transmit antenna selection") + ("rx-blockid", po::value(&rx_blockid)->default_value("0/Radio_0"), "Receive radio block ID") + ("tx-blockid", po::value(&tx_blockid)->default_value("0/Radio_1"), "Transmit radio block ID") + ("rx-chan", po::value(&rx_chan)->default_value(0), "Channel index on receive radio") + ("tx-chan", po::value(&tx_chan)->default_value(0), "Channel index on transmit radio") + ("rx-timestamps", po::value(&rx_timestamps)->default_value(false), "Set timestamps on RX") + ("tx-delay", po::value(&tx_delay)->default_value(10), "Ticks delay") + ("setup", po::value(&setup_time)->default_value(1.0), "seconds of setup time") + ("nsamps", po::value(&total_num_samps)->default_value(0), "total number of samples to receive") + ("rate", po::value(&rate)->default_value(0.0), "Sampling rate") + ("duration", po::value(&total_time)->default_value(0), "total number of seconds to receive") + ("int-n", "Tune USRP with integer-N tuning") + ("bw", po::value(&bw), "Analog frontend filter bandwidth in Hz (Rx and Tx)") + ("ref", po::value(&ref)->default_value("internal"), "reference source (internal, external, mimo)") + ; + po::variables_map vm; + po::store(po::parse_command_line(argc, argv, desc), vm); + po::notify(vm); + + //print the help message + if (vm.count("help")) { + std::cout << boost::format("RFNoC: Radio loopback test %s") % desc << std::endl; + std::cout + << std::endl + << "This application streams data from one radio to another using RFNoC.\n" + << std::endl; + return ~0; + } + + // Create a device session + std::cout << std::endl; + std::cout << boost::format("Creating the usrp device with: %s...") % args << std::endl; + uhd::device3::sptr dev = boost::dynamic_pointer_cast(uhd::device::make(args)); + if (not dev) { + std::cout << "Error: Could not find an RFNoC-compatible device." << std::endl; + return EXIT_FAILURE; + } + + // Access block controllers + if (not dev->has_block(rx_blockid) + or not dev->has_block(tx_blockid)) { + std::cout << "Error: Could not access at least one of these blocks:\n" + << "- " << rx_blockid + << "- " << tx_blockid + << std::endl; + std::cout << "Please confirm these blocks are actually available on the current loaded device." + << std::endl; + return EXIT_FAILURE; + } + radio_ctrl::sptr rx_radio_ctrl = dev->get_block_ctrl(rx_blockid); + radio_ctrl::sptr tx_radio_ctrl = dev->get_block_ctrl(tx_blockid); + + // Configure Rx radio + std::cout << "Configuring Rx radio..." << std::endl; + rx_radio_ctrl->set_args(rx_args); + if (spp) { + rx_radio_ctrl->set_arg("spp", spp, rx_chan); + + } + std::cout << "Setting Rx rate: " << (rate/1e6) << " Msps" << std::endl; + double actual_rx_rate = rx_radio_ctrl->set_rate(rate); + std::cout << "Actual Rx rate: " << (actual_rx_rate/1e6) << " Msps" << std::endl; + std::cout << "Setting Rx frequency: " << (rx_freq/1e6) << " MHz." << std::endl; + std::cout << "Actual Rx frequency: " << (rx_radio_ctrl->set_rx_frequency(rx_freq, rx_chan)/1e6) << " MHz." << std::endl; + if (rx_gain) { + std::cout << "Setting Rx gain: " << (rx_gain) << " dB." << std::endl; + std::cout << "Actual Rx gain: " << (rx_radio_ctrl->set_rx_gain(rx_gain, rx_chan)) << " dB." << std::endl; + } + if (not rx_ant.empty()) { + std::cout << "Setting Rx antenna: " << (rx_ant) << "." << std::endl; + rx_radio_ctrl->set_rx_antenna(rx_ant, rx_chan); + std::cout << "Actual Rx antenna: " << rx_radio_ctrl->get_rx_antenna(rx_chan) << "." << std::endl; + } + if (!rx_timestamps) { + std::cout << "Disabling timestamps on RX... (direct loopback, may underrun)" << std::endl; + } + rx_radio_ctrl->enable_rx_timestamps(rx_timestamps, 0); + // Configure Tx radio + std::cout << "Configuring Tx radio..." << std::endl; + tx_radio_ctrl->set_args(tx_args); + std::cout << "Setting Tx rate: " << (rate/1e6) << " Msps" << std::endl; + double actual_tx_rate = tx_radio_ctrl->set_rate(rate); + std::cout << "Actual Tx rate: " << (actual_tx_rate/1e6) << " Msps" << std::endl; + std::cout << "Setting Tx frequency: " << (tx_freq/1e6) << " MHz." << std::endl; + std::cout << "Actual Tx frequency: " << (tx_radio_ctrl->set_tx_frequency(tx_freq, tx_chan)/1e6) << " MHz." << std::endl; + if (tx_gain) { + std::cout << "Setting Tx gain: " << (tx_gain) << " dB." << std::endl; + std::cout << "Actual Tx gain: " << (tx_radio_ctrl->set_tx_gain(tx_gain, tx_chan)) << " dB." << std::endl; + } + if (not tx_ant.empty()) { + std::cout << "Setting Tx antenna: " << (tx_ant) << "." << std::endl; + tx_radio_ctrl->set_tx_antenna(tx_ant, tx_chan); + std::cout << "Actual Tx antenna: " << tx_radio_ctrl->get_tx_antenna(tx_chan) << "." << std::endl; + } + + // Compare rates + if (not uhd::math::frequencies_are_equal(actual_rx_rate, actual_tx_rate)) { + std::cout << "Error: Failed to set receive and transmit radios to same sampling rate!" << std::endl; + return EXIT_FAILURE; + } + // Create graph and connect blocks + uhd::rfnoc::graph::sptr graph = dev->create_graph("radio_loopback"); + std::cout << "Connecting radios..." << std::endl; + try { + graph->connect( + rx_blockid, rx_chan, + tx_blockid, tx_chan + ); + } catch (const uhd::runtime_error &ex) { + std::cout << "Error connecting blocks: " << std::endl; + std::cout << ex.what() << std::endl; + return EXIT_FAILURE; + } + tx_radio_ctrl->set_tx_streamer(true, tx_chan); + rx_radio_ctrl->set_rx_streamer(true, rx_chan); + + // Allow for some setup time + boost::this_thread::sleep(boost::posix_time::seconds(setup_time)); + + // Arm SIGINT handler + std::signal(SIGINT, &sig_int_handler); + + // Calculate timeout and set timers + if (total_time == 0 and total_num_samps > 0) { + const double buffer_time = 1.0; // seconds + total_time = (1.0/rate) * total_num_samps + buffer_time; + } + long total_time_milliseconds = total_time * 1000; + boost::posix_time::ptime start_time = boost::posix_time::microsec_clock::local_time(); + boost::posix_time::time_duration elapsed; + + // Start streaming + uhd::stream_cmd_t stream_cmd((total_num_samps == 0)? + uhd::stream_cmd_t::STREAM_MODE_START_CONTINUOUS: + uhd::stream_cmd_t::STREAM_MODE_NUM_SAMPS_AND_DONE + ); + stream_cmd.num_samps = size_t(total_num_samps); + stream_cmd.stream_now = true; + stream_cmd.time_spec = uhd::time_spec_t(); + std::cout << "Issuing start stream cmd..." << std::endl; + rx_radio_ctrl->issue_stream_cmd(stream_cmd, rx_chan); + std::cout << "Wait..." << std::endl; + + // Wait until we can exit + while (not stop_signal_called) { + boost::this_thread::sleep_for(boost::chrono::milliseconds(100)); + } + + // Stop radio + stream_cmd.stream_mode = uhd::stream_cmd_t::STREAM_MODE_STOP_CONTINUOUS; + std::cout << "Issuing stop stream cmd..." << std::endl; + rx_radio_ctrl->issue_stream_cmd(stream_cmd); + std::cout << "Done" << std::endl; + + return EXIT_SUCCESS; +} + -- cgit v1.2.3