OriginalCircuit.Electronics.TestEquipment 1.0.7

dotnet add package OriginalCircuit.Electronics.TestEquipment --version 1.0.7                
NuGet\Install-Package OriginalCircuit.Electronics.TestEquipment -Version 1.0.7                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="OriginalCircuit.Electronics.TestEquipment" Version="1.0.7" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add OriginalCircuit.Electronics.TestEquipment --version 1.0.7                
#r "nuget: OriginalCircuit.Electronics.TestEquipment, 1.0.7"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install OriginalCircuit.Electronics.TestEquipment as a Cake Addin
#addin nuget:?package=OriginalCircuit.Electronics.TestEquipment&version=1.0.7

// Install OriginalCircuit.Electronics.TestEquipment as a Cake Tool
#tool nuget:?package=OriginalCircuit.Electronics.TestEquipment&version=1.0.7                

.NET Networked Test Equipment Library

This library allows you to easily connect to and control your electronics test equipment over a network connection. It is implemented entirely in native .NET, so it does not require any additional equipment libraries or vendor specific DLLs. The library can be used on all platforms that support .NET, including Windows, Mac, Linux, and embedded systems.

Using this library, you can perform a variety of tasks such as sending commands, reading measurements, and configuring settings on your test equipment. It is designed to be easy to use, with a straightforward API and detailed documentation to help you get started.

Supported Equipment

Currently, the following equipment is supported:

  • Rigol MSO5000 Series Oscilloscopes
  • Rigol DP800 Series Power Supplies
  • Rigol DL3000 Series DC Loads
  • Rohde & Schwarz LCX Series LCR Meters
  • Siglent SDG series arbitrary function generators

However, the code should work with most oscilloscopes/supplies/loads with most functions.

Examples

Connecting to a DP800 Power Supply and Setting Channel Voltage

Connecting to a power supply instrument and setting the output to 3.3V:

    var dp800a = new RigolDP800Instrument();
    await dp800a.Connect("10.0.0.132");
    
    Console.WriteLine($"Connected to {dp800a.Identity.Manufacturer} {dp800a.Identity.Model}");
    
    await dp800a.SelectChannel(Channel.CH1);
    await dp800a.Output.Disable();
    await dp800a.Voltage.Set(3.3);
    await dp800a.WaitForOperationComplete(1);
    await dp800a.Output.Enable();

Oscilloscope Usage

The NetworkedTestEquipmentLibrary provides a class called RigolMSO5000Instrument for interacting with a Rigol MSO5000 oscilloscope.

The following example methods show how you can use the RigolMSO5000Instrument class to perform various tasks with an oscilloscope.

Connect to an MSO5000 Oscilloscope

var mso5000 = new RigolMSO5000Instrument();
await mso5000.Connect("10.0.0.74");

Enabling/Disabling Oscilloscope Channel Display

You can use the SetDisplay method of the Channel object to enable or disable the display of a specific channel on the oscilloscope. The method takes a bool indicating whether to enable (true) or disable (false) the display, and a ScopeChannel representing the channel to enable or disable (e.g. CH1, CH2, etc.).

Here is an example of how to disable the display of all channels and then re-enable the display of channels 1 and 2:

if (mso != null)
{
    // Disable display of all channels
    await mso.Channel.SetDisplay(false, ScopeChannel.CH1);
    await mso.Channel.SetDisplay(false, ScopeChannel.CH2);
    await mso.Channel.SetDisplay(false, ScopeChannel.CH3);
    await mso.Channel.SetDisplay(false, ScopeChannel.CH4);

    // Wait for all operations to complete before continuing
    await mso.WaitForOperationComplete();

    // Enable display of channels 1 and 2
    await mso.Channel.SetDisplay(true, ScopeChannel.CH1);
    await mso.Channel.SetDisplay(true, ScopeChannel.CH2);
}

Setting a Scope Channel to Display a Current Waveform

To set a scope channel to display a current waveform and adjust the channel scaling and offset, you can use the following methods of the Channel object:

  • SetCoupling(Coupling coupling, ScopeChannel channel): Sets the coupling type (DC or AC) of the specified channel.
  • SetUnits(ChannelUnits units, ScopeChannel channel): Sets the units (e.g. voltage, current) of the specified channel.
  • SetFineAdjustment(bool enable, ScopeChannel channel): Enables or disables fine adjustment of the specified channel.
  • SetScale(double scale, ScopeChannel channel): Sets the scale (vertical sensitivity) of the specified channel.
  • SetOffset(double offset, ScopeChannel channel): Sets the offset (vertical position) of the specified channel.

Here is an example of how to use these methods to set a scope channel (e.g. CHAN1) to display a current waveform:

if (mso == null) return;

var current = 0.1; // 100mA

// Set channel 1 to display a DC current waveform
await mso.Channel.SetCoupling(Coupling.DC, ScopeChannel.CHAN1);
// Set the units of channel 1 to Ampere
await mso.Channel.SetUnits(ChannelUnits.Ampere, ScopeChannel.CHAN1);
// Enable fine adjustment for channel 1
await mso.Channel.SetFineAdjustment(true, ScopeChannel.CHAN1);
// Set the scale of channel 1 to give 1 division of headroom
await mso.Channel.SetScale(Math.Round(current / 6, 2), ScopeChannel.CHAN1);
// Set the offset of channel 1 to give 1 division of footspace
await mso.Channel.SetOffset(-(current / 2), ScopeChannel.CHAN1);

// Wait for the operation to be completed
await mso.WaitForOperationComplete();

Note that the WaitForOperationComplete method can be used to wait for the oscilloscope to complete the current operation before continuing with the next operation. This is useful to ensure that the oscilloscope has finished updating the display and is ready to accept new commands.

Setting the Trigger

To set up the trigger on an oscilloscope channel, you can use the following steps:

  1. Set the coupling of the trigger to DC using the SetCoupling method.
  2. Set the trigger mode to EDGE using the SetMode method.
  3. Set the trigger edge source to CHAN1 using the SetSource method of the Edge property.
  4. Set the trigger edge slope to POS (positive) using the SetSlope method of the Edge property.
  5. Set the trigger level to a desired value (e.g. 8.2 volts) using the SetLevel method.
  6. Set the trigger sweep mode to NORM (normal) using the SetSweep method.
  7. Use the WaitToContinue method to wait for the oscilloscope to complete the current operation before continuing with the next operation.
  8. Use the SendKeypress method to hide the trigger menu by pressing the MOFF (Menu Off) button on the scope.
  9. Use the WaitForOperationComplete method to wait for the oscilloscope to complete the current operation.
if (mso == null) return;

double triggerLevel = 8.2; // 8.2volts

await mso.Trigger.SetCoupling(Coupling.DC);
await mso.Trigger.SetMode(TriggerMode.EDGE);
await mso.Trigger.Edge.SetSource(TriggerEdgeSource.CHAN1);
await mso.Trigger.Edge.SetSlope(TriggerSlope.POS);
await mso.Trigger.Edge.SetLevel(triggerLevel);
await mso.Trigger.SetSweep(TriggerSweep.NORM);
await mso.WaitToContinue();

// press the Menu Off (MOFF) button on the scope to hide the trigger menu
await mso.SendKeypress(ScopeKey.MOFF);

await mso.WaitForOperationComplete();

This code sets up the trigger on the oscilloscope to be an edge trigger on channel 1 with a positive slope and a trigger level of 8.2 volts. The sweep mode is set to NORM, which means that the scope will trigger based on the configured trigger settings. Finally, the MOFF button is pressed to hide the trigger menu on the scope.

Note: The TriggerSweep enumeration has three values: AUTO (auto trigger as well as configured trigger), NORM (only trigger based on configuration), or SIGN (single trigger, then stop the scope). You can choose the appropriate trigger sweep mode based on your needs.

Displaying Measurements and Statistics

To display measurements or measurements with statistics on the oscilloscope, you can use the EnableMeasurement and EnableStatistic methods of the Measure object. These methods take a MeasureFunction representing the measurement function to display (e.g. VPP, VMAX, VRMS) and a MeasureChannel representing the channel to display the measurement for (e.g. CHAN1, CHAN2).

Here is an example of how to use these methods to display various measurements and statistics on the oscilloscope:

if (mso != null)
{
    // Enable statistical measurements for VPP and VMAX on channel 1
    await mso.Measure.EnableStatistic(MeasureFunction.VPP, MeasureChannel.CHAN1);
    await mso.Measure.EnableStatistic(MeasureFunction.VMAX, MeasureChannel.CHAN1);

    // Enable statistical measurement for VMIN on channel 2
    await mso.Measure.EnableStatistic(MeasureFunction.VMIN, MeasureChannel.CHAN2);

    // Enable measurements for VRMS and VAVG on channel 2
    await mso.Measure.EnableMeasurement(MeasureFunction.VRMS, MeasureChannel.CHAN2);
    await mso.Measure.EnableMeasurement(MeasureFunction.VAVG, MeasureChannel.CHAN1);

    // Wait for all operations to complete before continuing
    await mso.WaitToContinue();
}

Removing Measurements from the Display

To remove measurements from the display of the oscilloscope, you can use the Clear method of the Measure object. Passing in the value MeasureItem.ALL as the parameter will remove all measurements from the display, regardless of which channel or function they are for.

Here is an example of how to use the Clear method:

if (mso != null)
{
	// Remove the first measurement item
	await scope.Measure.Clear(MeasureItem.ITEM1);
	
    // Remove all measurements from the display
    await mso.Measure.Clear(MeasureItem.ALL);
}

Querying Measurements

You can use the Measure.QueryMeasurement method to retrieve the value of a specific measurement (e.g. VMAX, VPP, PERIOD, etc) on the specific channel (CH1, CH2, MATH1, etc.) from the oscilloscope. If you do not specify the channel, it will use the channel you selecting with the Measure.SetSource method. The method below returns a double representing the voltage measurement.

internal async Task<double> QueryScopeMeasurement(MeasureFunction function, MeasureChannel channel)
{
    // Check if oscilloscope object is available
    if (mso == null)
        return double.NaN;

    // Initialize measurement and attempt counter
    double meas = double.NaN;
    int attempts = 0;

    // Make multiple attempts to retrieve measurement
    while (attempts++ < 20)
    {
        meas = await mso.Measure.QueryMeasurement(function, channel);

        // Return measurement if it is valid (not double.NaN)
        if (!double.IsNaN(meas))
            return meas;
    }

    // Return measurement (could be double.NaN if all attempts failed)
    return meas;
}

Note: A measurement may return double.NaN if the oscilloscope is busy updating the measurement when you attempt to read it. In this case, the method will make multiple attempts until a valid measurement is returned or the maximum number of attempts is reached.

Querying Statistics

You can use the Measure.QueryStatistic method to retrieve the value of a specific measurement (e.g. VMAX, VPP, PERIOD, etc) on the specific channel (CH1, CH2, MATH1, etc.) from the oscilloscope. This will allow you to read the statistics from the measurement window (e.g. Maximum, Average, Deviation.) If you do not specify the channel, it will use the channel you selecting with the Measure.SetSource method. The method below returns a double representing the voltage measurement.

internal async Task<double> QueryScopeStatistic(MeasureFunction function, MeasureType type, MeasureChannel channel)
{
    // Check if oscilloscope object is available
    if (mso == null)
        return double.NaN;

    // Initialize measurement and attempt counter
    double meas = double.NaN;
    int attempts = 0;

    // Make multiple attempts to retrieve measurement
    while (double.IsNaN(meas) && attempts++ < 20)
    {
        meas = await mso.Measure.QueryStatistic(function, type, channel);
    }

    // Return measurement (could be double.NaN if all attempts failed)
    return meas;
}

Note: A measurement may return double.NaN if the oscilloscope is busy updating the measurement when you attempt to read it. In this case, the method will make multiple attempts until a valid measurement is returned or the maximum number of attempts is reached.

Capturing and Saving the Oscilloscope Display

To capture the display of the oscilloscope and save it to a file, you can use the Capture method of the Display object and the SaveAsPngAsync method of the Image object. The Capture method returns an Image object representing the current display of the oscilloscope. The SaveAsPngAsync method takes a file path as a parameter and saves the image to the specified location in PNG format.

Here is an example of how to use these methods to capture and save the oscilloscope display:

if (mso != null)
{
    // Capture the oscilloscope display as an Image object
    var display = await mso.Display.Capture();

    // Save the image to a file if it was successfully captured
    if (display != null)
    {
        await display.SaveAsPngAsync("C:\\temp\\screenshot.png");
    }
}

Note: You will need to have the SixLabors.ImageSharp NuGet package installed in order to use this code.

Capturing Waveform Data from the Oscilloscope

To capture waveform data from the oscilloscope, you can use the Data method of the Waveform object. This method takes a WaveformSource representing the source of the waveform data (e.g. CHAN1, CHAN2) and a WaveformMode representing the mode of the waveform data (e.g. NORM, MAX, RAW). The WaveformMode can be one of the following:

  • NORM: Reads the waveform data currently displayed on the screen.
  • MAX: Reads the waveform data displayed on the screen when the oscilloscope is in the Run state; reads the waveform data in the internal memory when the oscilloscope is in the Stop state.
  • RAW: Reads the waveform data in the internal memory. Note: The data in the internal memory can only be read when the oscilloscope is in the Stop state. You are not allowed to operate the instrument when it is reading data.

The Data method returns a List<WaveformDataPoint> object, where each WaveformDataPoint object represents a point on the waveform with a Time and Voltage value.

Here is an example of how to use the Data method to capture waveform data from the oscilloscope:

if (mso != null)
{
    // Capture waveform data from channel 1 in normal mode
    var scopeWave = await mso.Waveform.Data(WaveformSource.CHAN1, WaveformMode.NORM);
}
Product Compatible and additional computed target framework versions.
.NET net6.0 is compatible.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 was computed.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.0.7 341 4/15/2023
1.0.6 226 3/21/2023
1.0.5 251 3/19/2023
1.0.4 303 2/5/2023
1.0.3 317 1/20/2023
1.0.2 318 1/20/2023
1.0.1 316 1/20/2023
1.0.0 318 12/29/2022

Added Siglent SDG function generator