From e0e8d1fe754a74741a764f589848971af8103c5f Mon Sep 17 00:00:00 2001 From: SGudla Date: Mon, 6 Nov 2023 15:08:51 +0530 Subject: [PATCH 1/2] Add adistream CLI tool Add adistream CLI tool to support data generation and streaming for tx devices like DACs. Update doc to include new tools optional dependency. Add doc page for tools. Signed-off-by: SGudla --- MANIFEST.in | 1 + adi/__init__.py | 5 + adi/tools/__init__.py | 3 + adi/tools/adistream.py | 392 +++++++++++++++++++++ doc/source/devices/adi.tools.adistream.rst | 7 + doc/source/devices/adi.tools.rst | 7 + doc/source/guides/quick.rst | 10 + pyproject.toml | 6 + 8 files changed, 431 insertions(+) create mode 100644 adi/tools/__init__.py create mode 100644 adi/tools/adistream.py create mode 100644 doc/source/devices/adi.tools.adistream.rst create mode 100644 doc/source/devices/adi.tools.rst diff --git a/MANIFEST.in b/MANIFEST.in index e67e0b8a0..469ab9511 100644 --- a/MANIFEST.in +++ b/MANIFEST.in @@ -1,6 +1,7 @@ include LICENSE include README.md include adi/* +include adi/tools/* exclude setup.cfg diff --git a/adi/__init__.py b/adi/__init__.py index dd8372a0a..4169a8319 100644 --- a/adi/__init__.py +++ b/adi/__init__.py @@ -109,5 +109,10 @@ except ImportError: pass +try: + from adi.tools.adistream import run_adi_stream +except ImportError: + pass + __version__ = "0.0.17" name = "Analog Devices Hardware Interfaces" diff --git a/adi/tools/__init__.py b/adi/tools/__init__.py new file mode 100644 index 000000000..4fd12de58 --- /dev/null +++ b/adi/tools/__init__.py @@ -0,0 +1,3 @@ +# Copyright (C) 2023-2024 Analog Devices, Inc. +# +# SPDX short identifier: ADIBSD diff --git a/adi/tools/adistream.py b/adi/tools/adistream.py new file mode 100644 index 000000000..f5eecda0c --- /dev/null +++ b/adi/tools/adistream.py @@ -0,0 +1,392 @@ +# Copyright (C) 2023-2024 Analog Devices, Inc. +# +# SPDX short identifier: ADIBSD + +import argparse +import sys +from time import sleep + +import adi +import numpy as np +from genalyzer import WaveformGen +from pynput import keyboard + + +class ADIStream(object): + """ ADIStream Class for DACs""" + + def __init__( + self, + classname, + uri, + v_ref_n, + v_ref_p, + npts, + chn_list, + v_req_l, + v_req_u, + out_freq, + code_sel, + wave_list, + device_name, + duty_cycle, + channel_order, + ): + """Constructor for ADIStream Class + + :param classname: class name to stream data to + :param uri: uri of the target device + :param v_ref_n: negative reference voltage in volts + :param v_ref_p: positive reference voltage in volts + :param npts: number of data points required per wave + :param chn_list: channels list to stream data to + :param v_req_l: lower end Voltage required in volts + :param v_req_u: upper end Voltage required in volts + :param out_freq: output frequency in Hz + :param code_sel: code select to stream data in + :param wave_list: list of waveform type required + :param device_name: part name if the class supports more than 1 generic + :param duty_cycle: duty cycle in percent for pwm wave + :param channel_order: channel wise data order streamed to the DAC""" + + self.sampling_freq = None + self.data = [] + self.data_stream_abort = False + self.classname = classname + self.npts = npts + self.v_ref_n = v_ref_n + self.v_ref_p = v_ref_p + self.v_req_l = v_req_l + self.v_req_u = v_req_u + self.out_freq = out_freq + self.code_sel = code_sel + self.duty_cyc = duty_cycle + self.channel_order = channel_order + + self.stream = eval( + "adi." + + classname + + "(uri='" + + uri + + "', device_name='" + + device_name + + "')" + ) + self.stream.tx_cyclic_buffer = True + self.wave_list = wave_list + + # Sort channels list + chn_list.sort() + + # Get channel id and set enabled channels + self.chan_list = [] + ch_name = self.stream.ctx.devices[0].channels[0].id + ch_name = ch_name[:-1] + for val in chn_list: + self.chan_list += [ch_name + str(val)] + self.stream.tx_enabled_channels = self.chan_list + self.chn_cnt = len(chn_list) + + # assign sine waveform as the default waveform type for channels with no waveform type provided + wave_cnt = len(self.wave_list) + while self.chn_cnt - wave_cnt: + self.wave_list[wave_cnt] = "sine" + wave_cnt += 1 + + # get the device resolution + try: + self.resolution = self.stream.output_bits[0] + except AttributeError: + self.resolution = self.stream.ctx.devices[0].channels[0].data_format.bits + + if "sampling_frequency" in self.stream.ctx.devices[0].attrs.keys(): + self.sampling_freq = self.stream._get_iio_dev_attr("sampling_frequency") + else: + raise Exception("No attribute with name sampling_frequency found") + + if self.out_freq is None: + self.out_freq = self.sampling_freq / self.npts + else: + self.stream.sampling_frequency = self.out_freq * self.npts + self.sampling_freq = int(self.stream.sampling_frequency) + + if self.v_req_l is None: + self.v_req_l = self.v_ref_n + + if self.v_req_u is None: + self.v_req_u = self.v_ref_p + + def write_buffered_data(self): + # Send data from device + if self.channel_order == "desc": + # Stream the numpy array of channels data in descending order + self.data = self.data[::-1] + self.stream.tx(self.data) + print("Data streaming started") + + def get_data(self): + if self.out_freq > self.sampling_freq / self.npts: + print( + f"The nearest possible output frequency that can be generated " + f"with the current config is {self.sampling_freq / self.npts} Hz" + ) + self.out_freq = self.sampling_freq / self.npts + elif self.out_freq < self.sampling_freq / self.npts: + print( + "Actual generated output frequency: ", + str(int(self.sampling_freq) // self.npts), + ) + + gen_obj = WaveformGen( + self.npts, + self.out_freq, + self.code_sel, + self.resolution, + self.v_ref_n, + self.v_ref_p, + self.v_req_l, + self.v_req_u, + ) + + # get data using the genalyzer apis + for val in range(self.chn_cnt): + if self.chn_cnt == 1: + if self.wave_list[val] == "pwm": + self.data = eval( + "gen_obj.gen_pwm_wave(duty_cycle = " + + str(self.duty_cyc / 100) + + ")" + ) + else: + self.data = eval("gen_obj.gen_" + self.wave_list[val] + "_wave()") + else: + if self.wave_list[val] == "pwm": + self.data.append( + eval( + "gen_obj.gen_pwm_wave(duty_cycle = " + + str(self.duty_cyc / 100) + + ")" + ) + ) + else: + self.data.append( + eval("gen_obj.gen_" + self.wave_list[val] + "_wave()") + ) + + self.data = np.array(self.data) + + def key_press_event(self, key): + if key == keyboard.Key.esc: + self.data_stream_abort = True + + def do_data_streaming(self): + self.get_data() + + listener = keyboard.Listener(on_press=self.key_press_event) + listener.start() + + print("Press " "escape" " key to stop cyclic data streaming..") + sleep(2) + + self.write_buffered_data() + + while not self.data_stream_abort: + # This should halt the control for the cyclic mode to work + pass + + self.exit_data_streaming() + + def exit_data_streaming(self): + self.stream.tx_destroy_buffer() + self.stream._ctx.__del__() + print("\r\nData streaming finished") + + +def run_adi_stream(argv=None, test_flag=False): + parser = argparse.ArgumentParser( + description="ADI data streaming app", + formatter_class=argparse.RawTextHelpFormatter, + usage="\nFor help, \n\t python adistream.py -h" + "\nTo go with default configurations, provide the positional arguments" + "\n\t python adistream.py ad579x serial:COM13,230400 -10 10" + "\nTo use available options," + "\nThe following command will generate a square wave at 1KHz" + "\n\t python adistream.py ad3530r serial:COM13,230400 0 2.5 -w square -f 1000" + "\nThe following command will stream data to channels 0, 1, and 2" + "\n\t python adistream.py ad3530r serial:COM13,230400 0 2.5 -cl 0 1 2", + ) + parser.add_argument( + "class", help="pyadi class name to stream data to", action="store" + ) + parser.add_argument("uri", help="URI of target device", action="store") + parser.add_argument( + "neg_voltage_ref", + help="Negative reference voltage of DAC in volts. \nInput 0 for unipolar DACs", + action="store", + type=float, + ) + parser.add_argument( + "pos_voltage_ref", + help="Positive reference voltage of DAC in volts.", + action="store", + type=float, + ) + parser.add_argument( + "-d", + "--device_name", + help="Part name if the class supports more than 1 generic", + action="store", + default="", + ) + parser.add_argument( + "-n", + "--data_points_per_wave", + help="Number of data points required per wave\n" + "Default value is 100 samples per wave", + action="store", + default=100, + type=int, + ) + parser.add_argument( + "-cl", + "--chn_list", + help="Channels list to stream data to\nE.g. --chn_list 0 2 3 to stream data to channels 0, 2, 3\n" + "Default is channel 0", + nargs="+", + action="store", + default=[0], + type=int, + ) + parser.add_argument( + "-vl", + "--v_lower_req", + help="Lower end Voltage required in volts. Should be in the accepted FSR.", + action="store", + type=float, + ) + parser.add_argument( + "-vu", + "--v_upper_req", + help="Upper end Voltage required in volts. Should be in the accepted FSR.", + action="store", + type=float, + ) + parser.add_argument( + "-f", + "--output_freq", + help="Output frequency required in Hz.\nNote: The effective output frequency per " + "channel might vary based on the device and number of active channels.\n" + "Default will be the maximum sampling frequency supported by the app.", + action="store", + type=int, + ) + parser.add_argument( + "-c", + "--code_sel", + help="code data format to stream data in. \nAccepted:\n\t0 for offset binary" + "\n\t1 for 2s-complement.\nDefault is 0 (offset binary)", + action="store", + default=0, + type=int, + choices=[0, 1], + ) + parser.add_argument( + "-w", + "--wave_types", + help="list of waveform type required. " + "\nAccepted: \n\tsine" + "\n\tcosine " + "\n\ttriangular " + "\n\tsquare " + "\n\tpwm" + "\nDefault is sine" + "\nE.g. --wave_types sine square triangular ", + nargs="+", + action="store", + default=["sine"], + choices=["sine", "cosine", "triangular", "square", "pwm"], + ) + parser.add_argument( + "-dc", + "--duty_cycle", + help="duty cycle in percent for pwm wave. " + "\nAccepted: \n\t1-99" + "\nDefault is 30", + action="store", + type=int, + default=30, + ) + parser.add_argument( + "-o", + "--channel_order", + help="order of the generated channel wise data array to be streamed." + "\nNote: This can be useful if the application assumes the data to be in a particular order." + "\nAccepted: \n\tasc -- default order (ascending)" + "\n\tdesc -- (descending)", + action="store", + default="asc", + choices=["asc", "desc"], + ) + parser.add_argument( + "-t", + "--test_device", + help="flag to test devices that are not part of the supported device list", + action="store_true", + ) + + args = vars(parser.parse_args(argv)) + + if args["test_device"]: + print("Device test enabled") + else: + supported_devices = [ + "ad578x", + "ad3552r", + "ad3530r", + "ad5754r", + "ad9152", + "ad9172", + ] + + device_name = args["class"] + if device_name not in supported_devices: + raise Exception( + f"The device {device_name} is not supported. Supported devices are" + f": {','.join(supported_devices)}" + f"\nEnable test_device (-t) flag to test devices not part of the supported devices list." + ) + + if args["data_points_per_wave"] <= 0: + raise argparse.ArgumentTypeError("data points must be greater than 0") + + if args["output_freq"] is not None and args["output_freq"] <= 0: + raise argparse.ArgumentTypeError( + "output frequency required cannot be a negative number" + ) + + if args["neg_voltage_ref"] >= args["pos_voltage_ref"]: + raise argparse.ArgumentTypeError( + "negative reference cannot be greater than the positive reference" + ) + + if args["duty_cycle"] < 0 or args["duty_cycle"] > 100: + raise argparse.ArgumentTypeError("Duty cycle shall be in between 0 and 100") + + app = ADIStream( + args["class"], + args["uri"], + args["neg_voltage_ref"], + args["pos_voltage_ref"], + args["data_points_per_wave"], + args["chn_list"], + args["v_lower_req"], + args["v_upper_req"], + args["output_freq"], + args["code_sel"], + args["wave_types"], + args["device_name"], + args["duty_cycle"], + args["channel_order"], + ) + app.data_stream_abort = test_flag + app.do_data_streaming() diff --git a/doc/source/devices/adi.tools.adistream.rst b/doc/source/devices/adi.tools.adistream.rst new file mode 100644 index 000000000..58a192b3e --- /dev/null +++ b/doc/source/devices/adi.tools.adistream.rst @@ -0,0 +1,7 @@ +tools.adistream +================= + +.. automodule:: adi.tools.adistream + :members: + :undoc-members: + :show-inheritance: diff --git a/doc/source/devices/adi.tools.rst b/doc/source/devices/adi.tools.rst new file mode 100644 index 000000000..b408101e5 --- /dev/null +++ b/doc/source/devices/adi.tools.rst @@ -0,0 +1,7 @@ +tools +================= + +.. automodule:: adi.tools + :members: + :undoc-members: + :show-inheritance: diff --git a/doc/source/guides/quick.rst b/doc/source/guides/quick.rst index 7895e32d5..a540fbc6a 100644 --- a/doc/source/guides/quick.rst +++ b/doc/source/guides/quick.rst @@ -30,6 +30,16 @@ To install the optional dependencies for JESD debugging and control Note that this is only needed for the ADRV9009-ZU11EG multi-SOM configuration. +To install the optional dependencies for leverage the built-in CLI tools + +.. code-block:: bash + + (sudo) pip install pyadi-iio[tools] + +.. note:: + + For the adistream CLI tool to generate and stream data to the device, it's also needed to install the `genalyzer `_ and its python bindings. + .. note:: On Linux the libiio python bindings are sometimes installed in locations not on path when building from source. On Ubuntu this is a common fix diff --git a/pyproject.toml b/pyproject.toml index 2ebf5c698..20b0493f7 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -45,12 +45,18 @@ namespaces = true jesd = [ "paramiko" ] +tools = [ + "pynput" +] [project.urls] homepage = "https://analogdevicesinc.github.io/pyadi-iio/" documentation = "https://analogdevicesinc.github.io/pyadi-iio/" repository = "https://github/analogdevicesinc/pyadi-iio" +[project.gui-scripts] +adistream = "adi.tools.adistream:run_adi_stream" + [tool.isort] multi_line_output=3 include_trailing_comma="True" From acdcc9b3733361d498dc51fd336740117edef383 Mon Sep 17 00:00:00 2001 From: SGudla Date: Thu, 4 Jan 2024 16:28:02 +0530 Subject: [PATCH 2/2] Add tests for adistream cli tool Signed-off-by: SGudla --- adi/tools/adistream.py | 5 +- test/test_adistream.py | 194 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 197 insertions(+), 2 deletions(-) create mode 100644 test/test_adistream.py diff --git a/adi/tools/adistream.py b/adi/tools/adistream.py index f5eecda0c..74c9091b8 100644 --- a/adi/tools/adistream.py +++ b/adi/tools/adistream.py @@ -74,6 +74,7 @@ def __init__( ) self.stream.tx_cyclic_buffer = True self.wave_list = wave_list + self.stream._ctx.set_timeout(100000) # Sort channels list chn_list.sort() @@ -137,7 +138,7 @@ def get_data(self): str(int(self.sampling_freq) // self.npts), ) - gen_obj = WaveformGen( + gen_obj = WaveformGen( # noqa: F841 ,variable is being used in the eval function below self.npts, self.out_freq, self.code_sel, @@ -340,7 +341,7 @@ def run_adi_stream(argv=None, test_flag=False): print("Device test enabled") else: supported_devices = [ - "ad578x", + "ad579x", "ad3552r", "ad3530r", "ad5754r", diff --git a/test/test_adistream.py b/test/test_adistream.py new file mode 100644 index 000000000..4d08a6990 --- /dev/null +++ b/test/test_adistream.py @@ -0,0 +1,194 @@ +import argparse + +import pytest + +try: + from adi.tools.adistream import run_adi_stream +except: + pytest.skip(allow_module_level=True) + +hardware = "ad5780" +ref_neg = "-10" +ref_pos = "+10" + + +######################################### +@pytest.mark.iio_hardware(hardware, True) +@pytest.mark.parametrize("classname", [("ad579x")]) +@pytest.mark.parametrize("code_sel", ["0", "1"]) +@pytest.mark.parametrize( + "wave_types", ["sine", "cosine", "triangular", "square", "pwm"] +) +@pytest.mark.parametrize("v_lower_req, v_upper_req", [("-5", "5"), ("0", "10")]) +@pytest.mark.parametrize("channel_order", ["asc", "desc"]) +@pytest.mark.parametrize("output_freq", ["100", "500"]) +def test_adistream_vaild_inputs( + capsys, + classname, + iio_uri, + code_sel, + wave_types, + v_lower_req, + v_upper_req, + channel_order, + output_freq, +): + run_adi_stream( + [ + classname, + iio_uri, + ref_neg, + ref_pos, + "--output_freq", + output_freq, + "--code_sel", + code_sel, + "--wave_types", + wave_types, + "--v_lower_req", + v_lower_req, + "--v_upper_req", + v_upper_req, + "--channel_order", + channel_order, + ], + True, + ) + captured = capsys.readouterr() + assert ( + captured.out == "Press escape key to stop cyclic data streaming..\n" + "Data streaming started\n\r\n" + "Data streaming finished\n" + ) + + +######################################### +@pytest.mark.iio_hardware(hardware, True) +@pytest.mark.parametrize("wave_types", ["pwm"]) +@pytest.mark.parametrize("duty_cycle", ["10", "50", "75", "90"]) +@pytest.mark.parametrize("classname", [("ad579x")]) +def test_adistream_dutycycle(capsys, classname, iio_uri, wave_types, duty_cycle): + run_adi_stream( + [ + classname, + iio_uri, + ref_neg, + ref_pos, + "--wave_types", + wave_types, + "--duty_cycle", + duty_cycle, + ], + True, + ) + captured = capsys.readouterr() + assert ( + captured.out == "Press escape key to stop cyclic data streaming..\n" + "Data streaming started\n\r\n" + "Data streaming finished\n" + ) + + +######################################### +@pytest.mark.iio_hardware(hardware, True) +@pytest.mark.parametrize("upper_req", ["15"]) +@pytest.mark.parametrize("classname", [("ad579x")]) +def test_adistream_voltages_1(classname, iio_uri, upper_req): + with pytest.raises(Exception) as context: + run_adi_stream( + [classname, iio_uri, ref_neg, ref_pos, "--v_upper_req", upper_req], True + ) + + assert ( + str(context.value) + == "required upper voltage cannot be greater than upper voltage reference" + ) + + +######################################### +@pytest.mark.iio_hardware(hardware, True) +@pytest.mark.parametrize("lower_req", ["-15"]) +@pytest.mark.parametrize("classname", [("ad579x")]) +def test_adistream_voltages_2(classname, iio_uri, lower_req): + with pytest.raises(Exception) as context: + run_adi_stream( + [classname, iio_uri, ref_neg, ref_pos, "--v_lower_req", lower_req], True + ) + + assert ( + str(context.value) + == "required lower voltage cannot be less than lower voltage reference" + ) + + +######################################### +@pytest.mark.iio_hardware(hardware, True) +@pytest.mark.parametrize("data_points", [("-10"), ("0")]) +@pytest.mark.parametrize("classname", [("ad579x")]) +def test_adistream_datapoints(classname, iio_uri, data_points): + with pytest.raises(argparse.ArgumentTypeError) as context: + run_adi_stream( + [ + classname, + iio_uri, + ref_neg, + ref_pos, + "--data_points_per_wave", + data_points, + ], + True, + ) + + assert str(context.value) == "data points must be greater than 0" + + +######################################### +@pytest.mark.iio_hardware(hardware, True) +@pytest.mark.parametrize("classname", [("ad1234"), ("ad1123x")]) +def test_adistream_invalid_classnames_1(classname, iio_uri): + with pytest.raises(Exception) as context: + run_adi_stream([classname, iio_uri, ref_neg, ref_pos], True) + + assert ( + str(context.value) + == f"The device {classname} is not supported. Supported devices are: " + "ad579x,ad3552r,ad3530r,ad5754r,ad9152,ad9172\n" + "Enable test_device (-t) flag to test devices not part of the supported " + "devices list." + ) + + +######################################### +@pytest.mark.iio_hardware(hardware, True) +@pytest.mark.parametrize("classname", [("ad1234"), ("ad1123x")]) +def test_adistream_invalid_classnames_2(classname, iio_uri): + with pytest.raises(AttributeError) as context: + run_adi_stream([classname, iio_uri, ref_neg, ref_pos, "--test_device"], True) + + assert str(context.value) == f"module 'adi' has no attribute '{classname}'" + + +######################################### +@pytest.mark.iio_hardware(hardware, True) +@pytest.mark.parametrize("output_freq", [("-10"), ("0")]) +@pytest.mark.parametrize("classname", [("ad579x")]) +def test_adistream_output_freq(classname, iio_uri, output_freq): + with pytest.raises(argparse.ArgumentTypeError) as context: + run_adi_stream( + [classname, iio_uri, ref_neg, ref_pos, "--output_freq", output_freq], True + ) + + assert str(context.value) == "output frequency required cannot be a negative number" + + +######################################### +@pytest.mark.iio_hardware(hardware, True) +@pytest.mark.parametrize("duty_cycle", [("-10"), ("110")]) +@pytest.mark.parametrize("classname", [("ad579x")]) +def test_adistream_invalid_duty_cycle(classname, iio_uri, duty_cycle): + with pytest.raises(argparse.ArgumentTypeError) as context: + run_adi_stream( + [classname, iio_uri, ref_neg, ref_pos, "--duty_cycle", duty_cycle], True + ) + + assert str(context.value) == "Duty cycle shall be in between 0 and 100"