This page was generated from docs/examples/Parameters/Parameter-With-Setpoints-defined-on-a-different-instrument.ipynb. Interactive online version: Binder badge.

ParameterWithSetpoints with setpoints defined on another instrument.

This notebook provides an example for writing a ParameterWithSetpoints that gets it setpoints from a different instrument. This is meant as an extension Simple Example of ParameterWithSetpoints which you should read before reading this notebook.

This is meant for the situation where an instrument has the capability to capture data into a buffer. This could be either by measuring a time series or by capturing each datapoint in the buffer via an external trigger. Such an instrument could capture the data into a ParameterWithSetpoints that user the time or the index of the buffer as setpoints. However, this is typically not very useful as the setpoints that are relevant for your experiment are often set by another instrument that is being swept as you read data into the buffer of the first instrument. This notebook shows an example of how you can generate the setpoints from the sweep settings of another instrument.

[1]:
import os

import numpy as np
[2]:
from qcodes.dataset import Measurement, plot_dataset
from qcodes.instrument import Instrument
from qcodes.validators import Arrays, Numbers
Logging hadn't been started.
Activating auto-logging. Current session state plus future input saved.
Filename       : /home/runner/.qcodes/logs/command_history.log
Mode           : append
Output logging : True
Raw input log  : False
Timestamping   : True
State          : active
Qcodes Logfile : /home/runner/.qcodes/logs/240912-19786-qcodes.log
[3]:
from qcodes.dataset import initialise_or_create_database_at, load_or_create_experiment
[4]:
from qcodes.parameters import DelegateParameter, Parameter, ParameterWithSetpoints

First, we define a dummy instrument that returns something like a current measurement buffer starting from a DelegateParameter given by sweep_start to one given by sweep_stop in n_points steps.

A function is added that allows you to set the parameters that sweep_start and sweep_stop delegates to.

[5]:
class GeneratedSetPoints(Parameter):
    """
    A parameter that generates a setpoint array from start, stop and num points
    parameters.
    """

    def __init__(self, startparam, stopparam, numpointsparam, *args, **kwargs):
        super().__init__(*args, **kwargs)
        self._startparam = startparam
        self._stopparam = stopparam
        self._numpointsparam = numpointsparam

    def get_raw(self):
        return np.linspace(
            self._startparam(), self._stopparam(), self._numpointsparam()
        )


class DummyArray(ParameterWithSetpoints):
    def get_raw(self):
        npoints = self.root_instrument.sweep_n_points.get_latest()
        return np.random.rand(npoints)


class DummyBufferedDMM(Instrument):
    def __init__(self, name, **kwargs):
        super().__init__(name, **kwargs)

        self.add_parameter(
            "sweep_start", source=None, parameter_class=DelegateParameter
        )

        self.add_parameter("sweep_stop", source=None, parameter_class=DelegateParameter)

        self.add_parameter(
            "sweep_n_points",
            unit="",
            initial_value=10,
            vals=Numbers(1, 1e3),
            get_cmd=None,
            set_cmd=None,
        )

        self.add_parameter(
            "setpoints",
            parameter_class=GeneratedSetPoints,
            startparam=self.sweep_start,
            stopparam=self.sweep_stop,
            numpointsparam=self.sweep_n_points,
            vals=Arrays(shape=(self.sweep_n_points.get_latest,)),
        )

        self.add_parameter(
            "current",
            get_cmd=self._get_current_data,
            unit="A",
            setpoints=(self.setpoints,),
            label="Current",
            parameter_class=ParameterWithSetpoints,
            vals=Arrays(shape=(self.sweep_n_points.get_latest,)),
        )

    def _get_current_data(self):
        npoints = self.sweep_n_points.get_latest()
        return np.random.rand(npoints)

    def set_sweep_parameters(self, start_parameter, stop_parameter, label=None):
        if start_parameter.unit != stop_parameter.unit:
            raise TypeError("You must sweep from and to parameters with the same unit")
        self.sweep_start.source = start_parameter
        self.sweep_stop.source = stop_parameter
        self.setpoints.unit = start_parameter.unit
        if label is not None:
            self.setpoints.label = label


class DummyWaveformGenerator(Instrument):
    def __init__(self, name, **kwargs):
        super().__init__(name, **kwargs)

        self.add_parameter(
            "v_start",
            initial_value=0,
            unit="V",
            label="v start",
            vals=Numbers(0, 1e3),
            get_cmd=None,
            set_cmd=None,
        )

        self.add_parameter(
            "v_stop",
            initial_value=1,
            unit="V",
            label="v stop",
            vals=Numbers(1, 1e3),
            get_cmd=None,
            set_cmd=None,
        )
[6]:
tutorial_db_path = os.path.join(os.getcwd(), "tutorial_paramter_with_setpoints.db")
initialise_or_create_database_at(tutorial_db_path)
load_or_create_experiment(
    experiment_name="tutorial_ParameterWithSetpoints", sample_name="no sample"
)
[6]:
tutorial_ParameterWithSetpoints#no sample#1@/home/runner/work/Qcodes/Qcodes/docs/examples/Parameters/tutorial_paramter_with_setpoints.db
----------------------------------------------------------------------------------------------------------------------------------------
[7]:
dmm = DummyBufferedDMM("dmm")
wg = DummyWaveformGenerator("wg")

First, we assume that we have wired up our instruments such that the current buffer will correspond to a voltage sweep from v_start to v_stop

For a real world experiment this would probably be wired such that the DMM is triggered at the start of the voltage sweep and then automatically measures sweep_n_points in the time that it takes the waveform generator to sweep to v_stop

[8]:
dmm.set_sweep_parameters(wg.v_start, wg.v_stop, label="Voltage")
[9]:
wg.v_start(0)
wg.v_stop(1)
dmm.sweep_n_points(501)

And we can grab the setpoints axis

[10]:
dmm.sweep_n_points()
[10]:
501
[11]:
sp_axis = dmm.setpoints()
len(sp_axis)
[11]:
501
[12]:
sp_axis[:10]
[12]:
array([0.   , 0.002, 0.004, 0.006, 0.008, 0.01 , 0.012, 0.014, 0.016,
       0.018])

As expected we get a result wit 501 points as we asked for an axis with 501 points.

[13]:
dmm.setpoints.validate(dmm.setpoints.get())

Naturally, we can also get the current buffer.

[14]:
current = dmm.current.get()
len(current)
[14]:
501

When we validate a ParameterWithSetpoints, we automatically validate that the shape is consistent between the parameters and its setpoints. As well as validating the shape as above.

[15]:
dmm.current.validate(dmm.current.get())

The consistent shapes can be validated explicitly.

[16]:
dmm.current.validate_consistent_shape()

We can inspect the setpoints of the spectrum.

[17]:
dmm.current.setpoints
[17]:
(<__main__.GeneratedSetPoints: setpoints at 140394974984528>,)

Measurement

[18]:
meas = Measurement()
meas.register_parameter(dmm.current)

with meas.run() as datasaver:
    datasaver.add_result((dmm.current, dmm.current()))

    dataid = datasaver.run_id
plot_dataset(datasaver.dataset)
Starting experimental run with id: 1.
[18]:
([<Axes: title={'center': 'Run #1, Experiment tutorial_ParameterWithSetpoints (no sample)'}, xlabel='Voltage (V)', ylabel='Current (mA)'>],
 [None])
../../_images/examples_Parameters_Parameter-With-Setpoints-defined-on-a-different-instrument_27_2.png
[19]:
wg.v_stop(10)
[20]:
meas = Measurement()
meas.register_parameter(dmm.current)

with meas.run() as datasaver:
    datasaver.add_result((dmm.current, dmm.current()))

    dataid = datasaver.run_id
plot_dataset(datasaver.dataset)
Starting experimental run with id: 2.
[20]:
([<Axes: title={'center': 'Run #2, Experiment tutorial_ParameterWithSetpoints (no sample)'}, xlabel='Voltage (V)', ylabel='Current (mA)'>],
 [None])
../../_images/examples_Parameters_Parameter-With-Setpoints-defined-on-a-different-instrument_29_2.png

Now imagine that we change our wiring such that we are sweeping a magnetic field while samling the current.

[21]:
class DummyMagnetPS(Instrument):
    """
    We assume this is a powersupply for an magnet that allows
    you to set the magnetic field.
    """

    def __init__(self, name, **kwargs):
        super().__init__(name, **kwargs)

        self.add_parameter(
            "b_start",
            initial_value=0,
            unit="T",
            label="B start",
            vals=Numbers(0, 7),
            get_cmd=None,
            set_cmd=None,
        )

        self.add_parameter(
            "b_stop",
            initial_value=1,
            unit="T",
            label="b stop",
            vals=Numbers(0, 7),
            get_cmd=None,
            set_cmd=None,
        )
[22]:
mpsu = DummyMagnetPS(name="psu1")
[23]:
dmm.set_sweep_parameters(mpsu.b_start, mpsu.b_stop, label="Magnetic field")
[24]:
meas = Measurement()
meas.register_parameter(dmm.current)

with meas.run() as datasaver:
    datasaver.add_result((dmm.current, dmm.current()))

    dataid = datasaver.run_id
plot_dataset(datasaver.dataset)
Starting experimental run with id: 3.
[24]:
([<Axes: title={'center': 'Run #3, Experiment tutorial_ParameterWithSetpoints (no sample)'}, xlabel='Magnetic field (T)', ylabel='Current (mA)'>],
 [None])
../../_images/examples_Parameters_Parameter-With-Setpoints-defined-on-a-different-instrument_34_2.png