aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--host/examples/python/CMakeLists.txt1
-rwxr-xr-xhost/examples/python/benchmark_rate.py481
2 files changed, 482 insertions, 0 deletions
diff --git a/host/examples/python/CMakeLists.txt b/host/examples/python/CMakeLists.txt
index 669c80203..758a69af1 100644
--- a/host/examples/python/CMakeLists.txt
+++ b/host/examples/python/CMakeLists.txt
@@ -8,6 +8,7 @@ SET(python_examples
rx_to_file.py
tx_waveforms.py
curses_fft.py
+ benchmark_rate.py
)
UHD_INSTALL(PROGRAMS ${python_examples} DESTINATION ${PKG_LIB_DIR}/examples/python COMPONENT examples)
diff --git a/host/examples/python/benchmark_rate.py b/host/examples/python/benchmark_rate.py
new file mode 100755
index 000000000..b89ec4d91
--- /dev/null
+++ b/host/examples/python/benchmark_rate.py
@@ -0,0 +1,481 @@
+#!/usr/bin/env python
+#
+# Copyright 2018 Ettus Research, a National Instruments Company
+#
+# SPDX-License-Identifier: GPL-3.0-or-later
+#
+"""
+Benchmark rate using Python API
+"""
+
+import argparse
+from datetime import datetime, timedelta
+import sys
+import time
+import threading
+import logging
+import numpy as np
+import uhd
+
+
+CLOCK_TIMEOUT = 1000 # 1000mS timeout for external clock locking
+INIT_DELAY = 0.05 # 50mS initial delay before transmit
+
+def parse_args():
+ """Parse the command line arguments"""
+ description = """UHD Benchmark Rate (Python API)
+
+ Utility to stress test a USRP device.
+
+ Specify --rx_rate for a receive-only test.
+ Specify --tx_rate for a transmit-only test.
+ Specify both options for a full-duplex test.
+ """
+ parser = argparse.ArgumentParser(formatter_class=argparse.RawTextHelpFormatter,
+ description=description)
+ parser.add_argument("-a", "--args", default="", type=str, help="single uhd device address args")
+ parser.add_argument("-d", "--duration", default=10.0, type=float,
+ help="duration for the test in seconds")
+ parser.add_argument("--rx_subdev", type=str, help="specify the device subdev for RX")
+ parser.add_argument("--tx_subdev", type=str, help="specify the device subdev for TX")
+ parser.add_argument("--rx_rate", type=float, help="specify to perform a RX rate test (sps)")
+ parser.add_argument("--tx_rate", type=float, help="specify to perform a TX rate test (sps)")
+ parser.add_argument("--rx_otw", type=str, default="sc16",
+ help="specify the over-the-wire sample mode for RX")
+ parser.add_argument("--tx_otw", type=str, default="sc16",
+ help="specify the over-the-wire sample mode for TX")
+ parser.add_argument("--rx_cpu", type=str, default="fc32",
+ help="specify the host/cpu sample mode for RX")
+ parser.add_argument("--tx_cpu", type=str, default="fc32",
+ help="specify the host/cpu sample mode for TX")
+ parser.add_argument("--ref", type=str,
+ help="clock reference (internal, external, mimo, gpsdo)")
+ parser.add_argument("--pps", type=str, help="PPS source (internal, external, mimo, gpsdo)")
+ parser.add_argument("--random", action="store_true", default=False,
+ help="Run with random values of samples in send() and recv() to stress-test"
+ " the I/O.")
+ parser.add_argument("-c", "--channels", default=[0], nargs="+", type=int,
+ help="which channel(s) to use (specify \"0\", \"1\", \"0 1\", etc)")
+ parser.add_argument("--rx_channels", nargs="+", type=int,
+ help="which RX channel(s) to use (specify \"0\", \"1\", \"0 1\", etc)")
+ parser.add_argument("--tx_channels", nargs="+", type=int,
+ help="which TX channel(s) to use (specify \"0\", \"1\", \"0 1\", etc)")
+ return parser.parse_args()
+
+
+class LogFormatter(logging.Formatter):
+ """Log formatter which prints the timestamp with fractional seconds"""
+ @staticmethod
+ def pp_now():
+ """Returns a formatted string containing the time of day"""
+ now = datetime.now()
+ return "{:%H:%M}:{:05.2f}".format(now, now.second + now.microsecond / 1e6)
+ # return "{:%H:%M:%S}".format(now)
+
+ def formatTime(self, record, datefmt=None):
+ converter = self.converter(record.created)
+ if datefmt:
+ formatted_date = converter.strftime(datefmt)
+ else:
+ formatted_date = LogFormatter.pp_now()
+ return formatted_date
+
+
+def setup_ref(usrp, ref, num_mboards):
+ """Setup the reference clock"""
+ if ref == "mimo":
+ if num_mboards != 2:
+ logger.error("ref = \"mimo\" implies 2 motherboards; "
+ "your system has %d boards", num_mboards)
+ return False
+ usrp.set_clock_source("mimo", 1)
+ else:
+ usrp.set_clock_source(ref)
+
+ # Lock onto clock signals for all mboards
+ if ref != "internal":
+ logger.debug("Now confirming lock on clock signals...")
+ end_time = datetime.now() + timedelta(milliseconds=CLOCK_TIMEOUT)
+ for i in range(num_mboards):
+ if ref == "mimo" and i == 0:
+ continue
+ is_locked = usrp.get_mboard_sensor("ref_locked", i)
+ while (not is_locked) and (datetime.now() < end_time):
+ time.sleep(1e-3)
+ is_locked = usrp.get_mboard_sensor("ref_locked", i)
+ if not is_locked:
+ logger.error("Unable to confirm clock signal locked on board %d", i)
+ return False
+ return True
+
+
+def setup_pps(usrp, pps, num_mboards):
+ """Setup the PPS source"""
+ if pps == "mimo":
+ if num_mboards != 2:
+ logger.error("ref = \"mimo\" implies 2 motherboards; "
+ "your system has %d boards", num_mboards)
+ return False
+ # make mboard 1 a slave over the MIMO Cable
+ usrp.set_time_source("mimo", 1)
+ else:
+ usrp.set_time_source(pps)
+ return True
+
+
+def check_channels(usrp, args):
+ """Check that the device has sufficient RX and TX channels available"""
+ # Check RX channels
+ if args.rx_rate:
+ if args.rx_channels:
+ rx_channels = args.rx_channels
+ else:
+ rx_channels = args.channels
+ # Check that each channel specified is less than the number of total number of rx channels
+ # the device can support
+ dev_rx_channels = usrp.get_rx_num_channels()
+ if not all(map((lambda chan: chan < dev_rx_channels), rx_channels)):
+ logger.error("Invalid RX channel(s) specified.")
+ return [], []
+ else:
+ rx_channels = []
+ # Check TX channels
+ if args.tx_rate:
+ if args.tx_channels:
+ tx_channels = args.tx_channels
+ else:
+ tx_channels = args.channels
+ # Check that each channel specified is less than the number of total number of tx channels
+ # the device can support
+ dev_tx_channels = usrp.get_tx_num_channels()
+ if not all(map((lambda chan: chan < dev_tx_channels), tx_channels)):
+ logger.error("Invalid TX channel(s) specified.")
+ return [], []
+ else:
+ tx_channels = []
+ return rx_channels, tx_channels
+
+
+def benchmark_rx_rate(usrp, rx_streamer, random, timer_elapsed_event, rx_statistics):
+ """Benchmark the receive chain"""
+ logger.info("Testing receive rate {:.3f} Msps on {:d} channels".format(
+ usrp.get_rx_rate()/1e6, rx_streamer.get_num_channels()))
+
+ # Make a receive buffer
+ num_channels = rx_streamer.get_num_channels()
+ max_samps_per_packet = rx_streamer.get_max_num_samps()
+ # TODO: The C++ code uses rx_cpu type here. Do we want to use that to set dtype?
+ recv_buffer = np.empty((num_channels, max_samps_per_packet), dtype=np.complex64)
+ metadata = uhd.types.RXMetadata()
+
+ # Craft and send the Stream Command
+ stream_cmd = uhd.types.StreamCMD(uhd.types.StreamMode.start_cont)
+ stream_cmd.stream_now = (num_channels == 1)
+ stream_cmd.time_spec = uhd.types.TimeSpec(usrp.get_time_now().get_real_secs() + INIT_DELAY)
+ rx_streamer.issue_stream_cmd(stream_cmd)
+
+ # To estimate the number of dropped samples in an overflow situation, we need the following
+ # On the first overflow, set had_an_overflow and record the time
+ # On the next ERROR_CODE_NONE, calculate how long its been since the recorded time, and use the
+ # tick rate to estimate the number of dropped samples. Also, reset the tracking variables
+ had_an_overflow = False
+ last_overflow = uhd.types.TimeSpec(0)
+ # Setup the statistic counters
+ num_rx_samps = 0
+ num_rx_dropped = 0
+ num_rx_overruns = 0
+ num_rx_seqerr = 0
+ num_rx_timeouts = 0
+ num_rx_late = 0
+
+ rate = usrp.get_rx_rate()
+ # Receive until we get the signal to stop
+ while not timer_elapsed_event.is_set():
+ if random:
+ stream_cmd.num_samps = np.random.randint(1, max_samps_per_packet+1, dtype=int)
+ rx_streamer.issue_stream_cmd(stream_cmd)
+ try:
+ num_rx_samps += rx_streamer.recv(recv_buffer, metadata) * num_channels
+ except RuntimeError as ex:
+ logger.error("Runtime error in receive: %s", ex)
+ return
+
+ # Handle the error codes
+ if metadata.error_code == uhd.types.RXMetadataErrorCode.none:
+ # Reset the overflow flag
+ if had_an_overflow:
+ had_an_overflow = False
+ num_rx_dropped += uhd.types.TimeSpec(
+ metadata.time_spec.get_real_secs() - last_overflow.get_real_secs()
+ ).to_ticks(rate)
+ elif metadata.error_code == uhd.types.RXMetadataErrorCode.overflow:
+ had_an_overflow = True
+ last_overflow = metadata.time_spec
+ # If we had a sequence error, record it
+ if metadata.out_of_sequence:
+ num_rx_seqerr += 1
+ logger.warning("Detected RX sequence error.")
+ # Otherwise just count the overrun
+ else:
+ num_rx_overruns += 1
+ elif metadata.error_code == uhd.types.RXMetadataErrorCode.late:
+ logger.warning("Receiver error: %s, restarting streaming...", metadata.strerror())
+ num_rx_late += 1
+ # Radio core will be in the idle state. Issue stream command to restart streaming.
+ stream_cmd.time_spec = uhd.types.TimeSpec(
+ usrp.get_time_now().get_real_secs() + INIT_DELAY)
+ stream_cmd.stream_now = (num_channels == 1)
+ rx_streamer.issue_stream_cmd(stream_cmd)
+ elif metadata.error_code == uhd.types.RXMetadataErrorCode.timeout:
+ logger.warning("Receiver error: %s, continuing...", metadata.strerror())
+ num_rx_timeouts += 1
+ else:
+ logger.error("Receiver error: %s", metadata.strerror())
+ logger.error("Unexpected error on receive, continuing...")
+
+ # Return the statistics to the main thread
+ rx_statistics["num_rx_samps"] = num_rx_samps
+ rx_statistics["num_rx_dropped"] = num_rx_dropped
+ rx_statistics["num_rx_overruns"] = num_rx_overruns
+ rx_statistics["num_rx_seqerr"] = num_rx_seqerr
+ rx_statistics["num_rx_timeouts"] = num_rx_timeouts
+ rx_statistics["num_rx_late"] = num_rx_late
+ # After we get the signal to stop, issue a stop command
+ rx_streamer.issue_stream_cmd(uhd.types.StreamCMD(uhd.types.StreamMode.stop_cont))
+
+
+def benchmark_tx_rate(usrp, tx_streamer, random, timer_elapsed_event, tx_statistics):
+ """Benchmark the transmit chain"""
+ logger.info("Testing transmit rate %.3f Msps on %d channels",
+ usrp.get_tx_rate() / 1e6, tx_streamer.get_num_channels())
+
+ # Make a transmit buffer
+ num_channels = tx_streamer.get_num_channels()
+ max_samps_per_packet = tx_streamer.get_max_num_samps()
+ # TODO: The C++ code uses rx_cpu type here. Do we want to use that to set dtype?
+ transmit_buffer = np.zeros((num_channels, max_samps_per_packet), dtype=np.complex64)
+ metadata = uhd.types.TXMetadata()
+ metadata.time_spec = uhd.types.TimeSpec(usrp.get_time_now().get_real_secs() + INIT_DELAY)
+ metadata.has_time_spec = bool(num_channels)
+
+ # Setup the statistic counters
+ num_tx_samps = 0
+ # TODO: The C++ has a single randomly sized packet sent here, then the thread returns
+ num_timeouts_tx = 0
+ # Transmit until we get the signal to stop
+ if random:
+ while not timer_elapsed_event.is_set():
+ total_num_samps = np.random.randint(1, max_samps_per_packet + 1, dtype=int)
+ num_acc_samps = 0
+ while num_acc_samps < total_num_samps:
+ num_tx_samps += tx_streamer.send(
+ transmit_buffer, metadata) * num_channels
+ num_acc_samps += min(total_num_samps - num_acc_samps,
+ tx_streamer.get_max_num_samps())
+ else:
+ while not timer_elapsed_event.is_set():
+ try:
+ num_tx_samps_now = tx_streamer.send(transmit_buffer, metadata) * num_channels
+ num_tx_samps += num_tx_samps_now
+ if num_tx_samps_now == 0:
+ num_timeouts_tx += 1
+ if (num_timeouts_tx % 10000) == 1:
+ logger.warning("Tx timeouts: %d", num_timeouts_tx)
+ metadata.has_time_spec = False
+
+ except RuntimeError as ex:
+ logger.error("Runtime error in transmit: %s", ex)
+ return
+
+ tx_statistics["num_tx_samps"] = num_tx_samps
+
+ # Send a mini EOB packet
+ metadata.end_of_burst = True
+ tx_streamer.send(np.zeros((num_channels, 0), dtype=np.complex64), metadata)
+
+
+def benchmark_tx_rate_async_helper(tx_streamer, timer_elapsed_event, tx_async_statistics):
+ """Receive and process the asynchronous TX messages"""
+ async_metadata = uhd.types.TXAsyncMetadata()
+
+ # Setup the statistic counters
+ num_tx_seqerr = 0
+ num_tx_underrun = 0
+ num_tx_timeouts = 0 # TODO: Not populated yet
+
+ try:
+ while not timer_elapsed_event.is_set():
+ # Receive the async metadata
+ if not tx_streamer.recv_async_msg(async_metadata, 0.1):
+ continue
+
+ # Handle the error codes
+ if async_metadata.event_code == uhd.types.TXMetadataEventCode.burst_ack:
+ return
+ elif ((async_metadata.event_code == uhd.types.TXMetadataEventCode.underflow) or
+ (async_metadata.event_code == uhd.types.TXMetadataEventCode.underflow_in_packet)):
+ num_tx_seqerr += 1
+ elif ((async_metadata.event_code == uhd.types.TXMetadataEventCode.seq_error) or
+ (async_metadata.event_code == uhd.types.TXMetadataEventCode.seq_error_in_packet)):
+ num_tx_underrun += 1
+ else:
+ logger.warning("Unexpected event on async recv (%s), continuing.",
+ async_metadata.event_code)
+
+ finally:
+ # Write the statistics back
+ tx_async_statistics["num_tx_seqerr"] = num_tx_seqerr
+ tx_async_statistics["num_tx_underrun"] = num_tx_underrun
+ tx_async_statistics["num_tx_timeouts"] = num_tx_timeouts
+
+
+def print_statistics(rx_statistics, tx_statistics, tx_async_statistics):
+ """Print TRX statistics in a formatted block"""
+ logger.debug("RX Statistics Dictionary: %s", rx_statistics)
+ logger.debug("TX Statistics Dictionary: %s", tx_statistics)
+ logger.debug("TX Async Statistics Dictionary: %s", tx_async_statistics)
+ # Print the statistics
+ statistics_msg = """Benchmark rate summary:
+ Num received samples: {}
+ Num dropped samples: {}
+ Num overruns detected: {}
+ Num transmitted samples: {}
+ Num sequence errors (Tx): {}
+ Num sequence errors (Rx): {}
+ Num underruns detected: {}
+ Num late commands: {}
+ Num timeouts (Tx): {}
+ Num timeouts (Rx): {}""".format(
+ rx_statistics.get("num_rx_samps", 0),
+ rx_statistics.get("num_rx_dropped", 0),
+ rx_statistics.get("num_rx_overruns", 0),
+ tx_statistics.get("num_tx_samps", 0),
+ tx_async_statistics.get("num_tx_seqerr", 0),
+ rx_statistics.get("num_rx_seqerr", 0),
+ tx_async_statistics.get("num_tx_underrun", 0),
+ rx_statistics.get("num_rx_late", 0),
+ rx_statistics.get("num_rx_timeouts", 0),
+ tx_async_statistics.get("num_tx_timeouts", 0))
+ logger.info(statistics_msg)
+
+
+def main():
+ """RX samples and write to file"""
+ args = parse_args()
+ # Setup some argument parsing
+ if not (args.rx_rate or args.tx_rate):
+ logger.error("Please specify --rx_rate and/or --tx_rate")
+ return False
+
+ # Setup a usrp device
+ usrp = uhd.usrp.MultiUSRP(args.args)
+ if usrp.get_mboard_name() == "USRP1":
+ logger.warning(
+ "Benchmark results will be inaccurate on USRP1 due to insufficient features.")
+
+ # Always select the subdevice first, the channel mapping affects the other settings
+ if args.rx_subdev:
+ usrp.set_rx_subdev_spec(uhd.usrp.SubdevSpec(args.rx_subdev))
+ if args.tx_subdev:
+ usrp.set_tx_subdev_spec(uhd.usrp.SubdevSpec(args.tx_subdev))
+
+ logger.info("Using Device: %s", usrp.get_pp_string())
+
+ # Set the reference clock
+ if args.ref and not setup_ref(usrp, args.ref, usrp.get_num_mboards()):
+ # If we wanted to set a reference clock and it failed, return
+ return False
+
+ # Set the PPS source
+ if args.pps and not setup_pps(usrp, args.pps, usrp.get_num_mboards()):
+ # If we wanted to set a PPS source and it failed, return
+ return False
+ # At this point, we can assume our device has valid and locked clock and PPS
+
+ rx_channels, tx_channels = check_channels(usrp, args)
+ if not rx_channels and not tx_channels:
+ # If the check returned two empty channel lists, that means something went wrong
+ return False
+ logger.info("Selected %s RX channels and %s TX channels",
+ rx_channels if rx_channels else "no",
+ tx_channels if tx_channels else "no")
+
+ logger.info("Setting device timestamp to 0...")
+ # If any of these conditions are met, we need to synchronize the channels
+ if args.pps == "mimo" or args.ref == "mimo" or len(rx_channels) > 1 or len(tx_channels) > 1:
+ usrp.set_time_unknown_pps(uhd.types.TimeSpec(0.0))
+ else:
+ usrp.set_time_now(uhd.types.TimeSpec(0.0))
+
+ threads = []
+ # Make a signal for the threads to stop running
+ quit_event = threading.Event()
+ # Create a dictionary for the RX statistics
+ # Note: we're going to use this without locks, so don't access it from the main thread until
+ # the worker has joined
+ rx_statistics = {}
+ # Spawn the receive test thread
+ if args.rx_rate:
+ usrp.set_rx_rate(args.rx_rate)
+ st_args = uhd.usrp.StreamArgs(args.rx_cpu, args.rx_otw)
+ st_args.channels = rx_channels
+ rx_streamer = usrp.get_rx_stream(st_args)
+ rx_thread = threading.Thread(target=benchmark_rx_rate,
+ args=(usrp, rx_streamer, args.random, quit_event,
+ rx_statistics))
+ threads.append(rx_thread)
+ rx_thread.start()
+ rx_thread.setName("bmark_rx_stream")
+
+ # Create a dictionary for the RX statistics
+ # Note: we're going to use this without locks, so don't access it from the main thread until
+ # the worker has joined
+ tx_statistics = {}
+ tx_async_statistics = {}
+ # Spawn the transmit test thread
+ if args.tx_rate:
+ usrp.set_tx_rate(args.tx_rate)
+ st_args = uhd.usrp.StreamArgs(args.tx_cpu, args.tx_otw)
+ st_args.channels = tx_channels
+ tx_streamer = usrp.get_tx_stream(st_args)
+ tx_thread = threading.Thread(target=benchmark_tx_rate,
+ args=(usrp, tx_streamer, args.random, quit_event,
+ tx_statistics))
+ threads.append(tx_thread)
+ tx_thread.start()
+ tx_thread.setName("bmark_tx_stream")
+
+ tx_async_thread = threading.Thread(target=benchmark_tx_rate_async_helper,
+ args=(tx_streamer, quit_event, tx_async_statistics))
+ threads.append(tx_async_thread)
+ tx_async_thread.start()
+ tx_async_thread.setName("bmark_tx_helper")
+
+ # Sleep for the required duration
+ # If we have a multichannel test, add some time for initialization
+ if len(rx_channels) > 1 or len(tx_channels) > 1:
+ args.duration += INIT_DELAY * 1000
+ time.sleep(args.duration)
+ # Interrupt and join the threads
+ logger.debug("Sending signal to stop!")
+ quit_event.set()
+ for thr in threads:
+ thr.join()
+
+ print_statistics(rx_statistics, tx_statistics, tx_async_statistics)
+
+ return True
+
+
+if __name__ == "__main__":
+ # Setup the logger with our custom timestamp formatting
+ global logger
+ logger = logging.getLogger(__name__)
+ logger.setLevel(logging.DEBUG)
+ console = logging.StreamHandler()
+ logger.addHandler(console)
+ formatter = LogFormatter(fmt='[%(asctime)s] [%(levelname)s] (%(threadName)-10s) %(message)s')
+ console.setFormatter(formatter)
+
+ # Vamos, vamos, vamos!
+ sys.exit(not main())