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..74c9091b8 --- /dev/null +++ b/adi/tools/adistream.py @@ -0,0 +1,393 @@ +# 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 + self.stream._ctx.set_timeout(100000) + + # 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( # noqa: F841 ,variable is being used in the eval function below + 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 = [ + "ad579x", + "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" 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"