From bf7f26b67d776fa57009a898d1a1c8085ed0012e Mon Sep 17 00:00:00 2001 From: RibhuDP Date: Thu, 2 Nov 2023 01:06:37 +0530 Subject: [PATCH 1/2] adi:Generic Add a generic driver 1. Add a generic driver and example script Signed-off-by: RibhuDP --- +adi/+Generic/Rx.m | 248 +++++++++++++++++++++++++++++++++ examples/Generic_DataCapture.m | 37 +++++ 2 files changed, 285 insertions(+) create mode 100644 +adi/+Generic/Rx.m create mode 100644 examples/Generic_DataCapture.m diff --git a/+adi/+Generic/Rx.m b/+adi/+Generic/Rx.m new file mode 100644 index 0000000..399f212 --- /dev/null +++ b/+adi/+Generic/Rx.m @@ -0,0 +1,248 @@ +classdef Rx < adi.common.Rx & matlabshared.libiio.base & ... + adi.common.Attribute & adi.common.DeviceAttribute & adi.common.Channel + % Generic Precision ADC Class + % adi.Generic.Rx Receives data from the connected ADC + % The adi.Generic.Rx System object is a signal source that can receive + % data from any generic ADI precision ADC. + % + % rx = adi.Generic.Rx('adxxxx', 'ip:analog.local'); + % + + properties + % DeviceAttributeNames Device Attribute Names + % Array of device IIO attribute names as defined in the + % driver/firmware, and queried from the hardware + % at initialization time + DeviceAttributeNames + + % ChannelAttributeNames Device Attribute Names + % Array of channel IIO attribute names as defined in the + % driver/firmware, and queried from the hardware + % at initialization time + ChannelAttributeNames + end + + properties (Nontunable) + % SamplesPerFrame Samples Per Frame + % Number of samples per frame, specified as an even positive + % integer. + SamplesPerFrame = 1024 + end + + properties (Nontunable, Hidden) + channel_names = {'voltage0'} + Timeout = Inf + kernelBuffersCount = 1 + dataTypeStr = 'int32' + phyDevName = 'adxxxx' + devName = 'adxxxx' + end + + properties (Nontunable, Hidden, Constant) + Type = 'Rx' + ComplexData = false + end + + properties (Hidden, Nontunable, Access = protected) + isOutput = false + end + + properties(Hidden, Access = protected) + cachedAttrWrites = {} + end + + methods + + %% Constructor + function obj = Rx(Args) + arguments + Args.devName + Args.phyDevName + Args.uri + end + + obj = obj@matlabshared.libiio.base(); + obj.enableExplicitPolling = false; + obj.EnabledChannels = 1; + obj.BufferTypeConversionEnable = true; + + % Set device name, phyDevName, uri from input arguments + try + obj.devName = Args.devName; + obj.phyDevName = Args.phyDevName; + obj.uri = Args.uri; + catch + e = MException('MATLAB:notEnoughInputs',['Not enough input ' ... + 'arguments. Specify devName, phyDevName and uri']); + throw(e) + end + + % Determine datatype using trial and error + possibleDataTypeStr = {'int8', 'int16', 'int32', 'int64'}; + for k = 1:length(possibleDataTypeStr) + try + obj.dataTypeStr = possibleDataTypeStr{k}; + obj.setup(); + break; + catch + end + end + assert(obj.ConnectedToDevice == 1, "Connection could not be established with the specified device") + + % Check is the platform is running ADI Kuiper Linux. + % Presence of 'local,kernel' context attribute would + % indicate usage of Linux platform + isLinuxPLatform = false; + try + obj.iio_context_get_attr_value(obj.iioCtx, 'local,kernel'); + isLinuxPLatform = true; + catch + end + + release(obj); + + % If platform is running ADI Kuiper Linux, kernelBuffersCount + % can be increased + if isLinuxPLatform + obj.kernelBuffersCount = 4; + disp("Detected ADI Kuiper Linux platform. Changing buffer count to 4..") + end + + end + + function flush(obj) + % Flush the buffer contents + flushBuffers(obj); + end + + function SetDeviceAttrValue(obj, attr, val) + % Apply the device attribute value when device is + % connected + % rx.SetDeviceAttrValue('sampling_frequency', '1000000') + obj.cachedAttrWrites{end+1} = {attr, val}; + end + + + function val = GetDeviceAttrValue(obj, attr) + % Fetch the device attribute value if device is + % connected. + % rx.GetDeviceAttrValue('sampling_frequency') + if obj.ConnectedToDevice + val = obj.getDeviceAttributeRAW(attr, 128); + end + end + + function SetChannelAttrValue(obj, chnID, attr, val) + % Apply the channel attribute value when device is + % connected. To write to the first channel attribute use + % rx.SetChannelAttrValue('voltage0', 'scale', '2') or + % rx.SetChannelAttrValue(1, 'scale', '2') + if isnumeric(chnID) + chnID = obj.channel_names(chanID); + end + obj.cachedAttrWrites{end+1} = {chnID, attr, val}; + end + + function val = GetChannelAttrValue(obj, chnID, attr) + % Fetch the channel attribute value if device is + % connected. To get the first channel attribute value, use + % rx.GetChannelAttrValue('voltage0', 'scale') or + % rx.GetChannelAttrValue(1, 'scale') + if obj.ConnectedToDevice + if isnumeric(chnID) + chnID = obj.channel_names(chanID); + end + val = obj.getAttributeRAW(attr, chnID, obj.isOutput); + end + end + + end + + %% API Functions + methods (Hidden, Access = protected) + + function setupInit(obj) + + % Write all attributes to device once connected through set + % methods + % Do writes directly to hardware without using set methods. + % This is required since Simulink support doesn't support + % modification to nontunable variables at SetupImpl + + %Fetch all the attributes and channel names + if ~isempty(obj.channel_names) + obj.fetch_channel_names(); + end + + if isempty(obj.DeviceAttributeNames) + obj.fetch_device_attributes(); + end + + if isempty(obj.ChannelAttributeNames) + obj.fetch_channel_attributes(); + end + + % Apply attribute settings in hardware + if ~isempty(obj.cachedAttrWrites) + for i = 1:length(obj.cachedAttrWrites) + if length(obj.cachedAttrWrites{i}) == 2 + % Apply device attribute + attr = obj.cachedAttrWrites{i}{1}; + val = obj.cachedAttrWrites{i}{2}; + obj.setDeviceAttributeRAW(attr, val); + elseif length(obj.cachedAttrWrites{i}) == 3 + % Apply channel attribute + chan = obj.cachedAttrWrites{i}{1}; + attr = obj.cachedAttrWrites{i}{2}; + val = obj.cachedAttrWrites{i}{3}; + obj.setAttributeRAW(chan, attr, val, obj.isOutput); + end + end + end + + obj.cachedAttrWrites = {}; + + end + + function fetch_channel_names(obj) + % Update the channel names + obj.channel_names = {}; + phydev = getDev(obj, obj.devName); + chanCount = obj.iio_device_get_channels_count(phydev); + for c = 1:chanCount + chanPtr = obj.iio_device_get_channel(phydev, c - 1); + obj.channel_names{end + 1} = obj.iio_channel_get_name(chanPtr); + end + end + + function fetch_device_attributes(obj) + % Store the attribute names + devPtr = obj.getDev(obj.phyDevName); + devAttrCnt = obj.iio_device_get_attrs_count(devPtr); + for i=1:devAttrCnt + obj.DeviceAttributeNames{end+1} = obj.iio_device_get_attr(devPtr, i-1); + end + end + + function fetch_channel_attributes(obj) + % Store channel attribute names, if there's channels present + devPtr = obj.getDev(obj.phyDevName); + + if ~isempty(obj.channel_names) + % Fetch first channel + chanPtr = calllib(obj.libName, 'iio_device_get_channel', devPtr, 0); + + % Fetch count of channel attributes + chanAttrCnt = obj.iio_channel_get_attrs_count(chanPtr); + + % Fetch each attribute for the channel + for j=1:chanAttrCnt + obj.ChannelAttributeNames{j} = obj.iio_channel_get_attr(chanPtr, j-1); + end + end + + end + + + end +end diff --git a/examples/Generic_DataCapture.m b/examples/Generic_DataCapture.m new file mode 100644 index 0000000..af901fd --- /dev/null +++ b/examples/Generic_DataCapture.m @@ -0,0 +1,37 @@ +%% Script for capturing buffers of data from a connected IIO device + +% Instantiate the Generic Rx system object, and specify the device name, uri +% and phyDevName in the manner shown below +rx = adi.Generic.Rx("devName", 'ad4630-24', 'phyDevName', 'ad4630-24',... + 'uri', 'ip:analog.local'); + +rx.SamplesPerFrame = 4096; % Using values less than 3660 can yield poor +% performance, generally + +% Enable channels for data capture +rx.EnabledChannels = [1]; + +% Display device and channel attribute names +rx.DeviceAttributeNames +rx.ChannelAttributeNames + +% Read and write attribute values +% For example, with ad4630-24 +% rx.GetDeviceAttrValue('sampling_frequency') +% rx.SetDeviceAttrValue('sampling_frequency', '2000000'); +% rx.SetChannelAttrValue('differential0', 'hardwaregain', '2'); + +% Capture data +data = rx(); + +% Plot samples +enabledChannels = size(data, 2); +figure(1); +for i = 1:enabledChannels + subplot(enabledChannels, 1, i); + plot(data(1:rx.SamplesPerFrame, i)); + title("Channel " + num2str(rx.EnabledChannels(i))); +end + +% Delete the system object +release(rx); \ No newline at end of file From 7ba47451f283eba50eb07782dd4eea6b37bf07ad Mon Sep 17 00:00:00 2001 From: RibhuDP Date: Fri, 22 Dec 2023 16:50:22 +0530 Subject: [PATCH 2/2] adi:Generic: Address the review comments 1. Add docstrings for properties 2. Remove extra line 3. Fix type in docstring 4. Remove extraneous comment Signed-off-by: RibhuDP --- +adi/+Generic/Rx.m | 22 ++++++++++++++++++++-- examples/Generic_DataCapture.m | 10 +++++----- 2 files changed, 25 insertions(+), 7 deletions(-) diff --git a/+adi/+Generic/Rx.m b/+adi/+Generic/Rx.m index 399f212..dc62d80 100644 --- a/+adi/+Generic/Rx.m +++ b/+adi/+Generic/Rx.m @@ -6,7 +6,6 @@ % data from any generic ADI precision ADC. % % rx = adi.Generic.Rx('adxxxx', 'ip:analog.local'); - % properties % DeviceAttributeNames Device Attribute Names @@ -15,7 +14,7 @@ % at initialization time DeviceAttributeNames - % ChannelAttributeNames Device Attribute Names + % ChannelAttributeNames Channel Attribute Names % Array of channel IIO attribute names as defined in the % driver/firmware, and queried from the hardware % at initialization time @@ -30,11 +29,27 @@ end properties (Nontunable, Hidden) + % channel_names Channel Names channel_names = {'voltage0'} + + % Timeout Timeout for the IIO client Timeout = Inf + + % kernelBuffersCount Kernel Buffers Count + % For Tinyiiod firmware, set the value to 1. + % For Linux platforms, value can be greater than 1 aa well kernelBuffersCount = 1 + + % dataTypeStr Data Type String + % Data type to store the samples dataTypeStr = 'int32' + + % phyDevName Physical Device Name + % Set this to match the IIO device name phyDevName = 'adxxxx' + + % devName Device Name + % Set this to match the IIO device name devName = 'adxxxx' end @@ -48,6 +63,9 @@ end properties(Hidden, Access = protected) + % cachedAttrWrites Cached Attribute Writes + % Stores the attr write values locally until hardware is + % connected. cachedAttrWrites = {} end diff --git a/examples/Generic_DataCapture.m b/examples/Generic_DataCapture.m index af901fd..43185e7 100644 --- a/examples/Generic_DataCapture.m +++ b/examples/Generic_DataCapture.m @@ -5,15 +5,15 @@ rx = adi.Generic.Rx("devName", 'ad4630-24', 'phyDevName', 'ad4630-24',... 'uri', 'ip:analog.local'); -rx.SamplesPerFrame = 4096; % Using values less than 3660 can yield poor -% performance, generally +% Specify number of samples to fetch using buffered capture +rx.SamplesPerFrame = 4096; % Enable channels for data capture rx.EnabledChannels = [1]; -% Display device and channel attribute names -rx.DeviceAttributeNames -rx.ChannelAttributeNames +% Uncomment to display device and channel attribute names +%rx.DeviceAttributeNames +%rx.ChannelAttributeNames % Read and write attribute values % For example, with ad4630-24