Source code for instruments.tektronix.tektds5xx

#!/usr/bin/env python
# -*- coding: utf-8 -*-
#
# tektds5xx.py: Driver for the Tektronix TDS 5xx series oscilloscope.
#
# © 2014 Chris Schimp (silverchris@gmail.com)
#
# Modified from tektds224.py
# © 2013 Steven Casagrande (scasagrande@galvant.ca).
#
# This file is a part of the InstrumentKit project.
# Licensed under the AGPL version 3.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
"""
Provides support for the Tektronix DPO 500 oscilloscope series.

Originally contributed by Chris Schimp (silverchris@gmail.com) in 2014.
Based off of tektds224.py written by Steven Casagrande.
"""

# IMPORTS #####################################################################

from __future__ import absolute_import
from __future__ import division
from functools import reduce

import time
from time import sleep
from datetime import datetime
import operator
import struct

from builtins import range, map, round
from enum import Enum

import numpy as np

from instruments.abstract_instruments import (
    OscilloscopeChannel,
    OscilloscopeDataSource,
    Oscilloscope,
)
from instruments.generic_scpi import SCPIInstrument
from instruments.util_fns import ProxyList

# CLASSES #####################################################################


class _TekTDS5xxMeasurement(object):

    """
    Class representing a measurement channel on the Tektronix TDS5xx
    """

    def __init__(self, tek, idx):
        self._tek = tek
        self._id = idx + 1
        resp = self._tek.query('MEASU:MEAS{}?'.format(self._id))
        self._data = dict(zip(['enabled', 'type', 'units', 'src1', 'src2',
                               'edge1', 'edge2', 'dir'], resp.split(';')))

    def read(self):
        """
        Gets the current measurement value of the channel, and returns a dict
        of all relevent information

        :rtype: `dict` of measurement parameters
        """
        if int(self._data['enabled']):
            resp = self._tek.query('MEASU:MEAS{}:VAL?'.format(self._id))
            self._data['value'] = float(resp)
            return self._data

        return self._data


class _TekTDS5xxDataSource(OscilloscopeDataSource):

    """
    Class representing a data source (channel, math, or ref) on the Tektronix
    TDS 5xx.

    .. warning:: This class should NOT be manually created by the user. It is
        designed to be initialized by the `TekTDS5xx` class.
    """

    @property
    def name(self):
        """
        Gets the name of this data source, as identified over SCPI.

        :type: `str`
        """
        return self._name

    def read_waveform(self, bin_format=True):
        """
        Read waveform from the oscilloscope.
        This function is all inclusive. After reading the data from the
        oscilloscope, it unpacks the data and scales it accordingly.

        Supports both ASCII and binary waveform transfer. For 2500 data
        points, with a width of 2 bytes, transfer takes approx 2 seconds for
        binary, and 7 seconds for ASCII over Galvant Industries' GPIBUSB
        adapter.

        Function returns a tuple (x,y), where both x and y are numpy arrays.

        :param bool bin_format: If `True`, data is transfered
            in a binary format. Otherwise, data is transferred in ASCII.

        :rtype: two item `tuple` of `numpy.ndarray`
        """
        with self:

            if not bin_format:
                # Set the data encoding format to ASCII
                self._parent.sendcmd('DAT:ENC ASCI')
                raw = self._parent.query('CURVE?')
                raw = raw.split(',')  # Break up comma delimited string
                raw = map(float, raw)  # Convert each list element to int
                raw = np.array(raw)  # Convert into numpy array
            else:
                # Set encoding to signed, big-endian
                self._parent.sendcmd('DAT:ENC RIB')
                data_width = self._parent.data_width
                self._parent.sendcmd('CURVE?')
                # Read in the binary block, data width of 2 bytes
                raw = self._parent.binblockread(data_width)

                # pylint: disable=protected-access
                self._parent._file.flush_input()  # Flush input buffer

            # Retrieve Y offset
            yoffs = self._parent.query('WFMP:{}:YOF?'.format(self.name))
            # Retrieve Y multiply
            ymult = self._parent.query('WFMP:{}:YMU?'.format(self.name))
            # Retrieve Y zero
            yzero = self._parent.query('WFMP:{}:YZE?'.format(self.name))

            y = ((raw - float(yoffs)) * float(ymult)) + float(yzero)

            # Retrieve X incr
            xincr = self._parent.query('WFMP:{}:XIN?'.format(self.name))
            # Retrieve number of data points
            ptcnt = self._parent.query('WFMP:{}:NR_P?'.format(self.name))

            x = np.arange(float(ptcnt)) * float(xincr)

            return (x, y)


class _TekTDS5xxChannel(_TekTDS5xxDataSource, OscilloscopeChannel):

    """
    Class representing a channel on the Tektronix TDS 5xx.

    This class inherits from `_TekTDS5xxDataSource`.

    .. warning:: This class should NOT be manually created by the user. It is
        designed to be initialized by the `TekTDS5xx` class.
    """

    def __init__(self, parent, idx):
        super(_TekTDS5xxChannel, self).__init__(parent, "CH{}".format(idx + 1))
        self._idx = idx + 1

    @property
    def coupling(self):
        """
        Gets/sets the coupling setting for this channel.

        :type: `TekTDS5xx.Coupling`
        """
        return TekTDS5xx.Coupling(
            self._parent.query("CH{}:COUPL?".format(self._idx))
        )

    @coupling.setter
    def coupling(self, newval):
        if not isinstance(newval, TekTDS5xx.Coupling):
            raise TypeError("Coupling setting must be a `TekTDS5xx.Coupling`"
                            " value, got {} instead.".format(type(newval)))

        self._parent.sendcmd("CH{}:COUPL {}".format(self._idx, newval.value))

    @property
    def bandwidth(self):
        """
        Gets/sets the Bandwidth setting for this channel.

        :type: `TekTDS5xx.Bandwidth`
        """
        return TekTDS5xx.Bandwidth(
            self._parent.query("CH{}:BAND?".format(self._idx))
        )

    @bandwidth.setter
    def bandwidth(self, newval):
        if not isinstance(newval, TekTDS5xx.Bandwidth):
            raise TypeError("Bandwidth setting must be a `TekTDS5xx.Bandwidth`"
                            " value, got {} instead.".format(type(newval)))

        self._parent.sendcmd("CH{}:BAND {}".format(self._idx, newval.value))

    @property
    def impedance(self):
        """
        Gets/sets the impedance setting for this channel.

        :type: `TekTDS5xx.Impedance`
        """
        return TekTDS5xx.Impedance(
            self._parent.query("CH{}:IMP?".format(self._idx))
        )

    @impedance.setter
    def impedance(self, newval):
        if not isinstance(newval, TekTDS5xx.Impedance):
            raise TypeError("Impedance setting must be a `TekTDS5xx.Impedance`"
                            " value, got {} instead.".format(type(newval)))

        self._parent.sendcmd("CH{}:IMP {}".format(self._idx, newval.value))

    @property
    def probe(self):
        """
        Gets the connected probe value for this channel

        :type: `float`
        """
        return round(1 / float(self._parent.query("CH{}:PRO?".format(self._idx))), 0)

    @property
    def scale(self):
        """
        Gets/sets the scale setting for this channel.

        :type: `TekTDS5xx.Impedance`
        """
        return float(self._parent.query("CH{}:SCA?".format(self._idx)))

    @scale.setter
    def scale(self, newval):
        self._parent.sendcmd("CH{0}:SCA {1:.3E}".format(self._idx, newval))
        resp = float(self._parent.query("CH{}:SCA?".format(self._idx)))
        if newval != resp:
            raise ValueError("Tried to set CH{0} Scale to {1} but got {2}"
                             " instead".format(self._idx, newval, resp))


[docs]class TekTDS5xx(SCPIInstrument, Oscilloscope): """ Support for the TDS5xx series of oscilloscopes Implemented from: | TDS Family Digitizing Oscilloscopes | (TDS 410A, 420A, 460A, 520A, 524A, 540A, 544A, | 620A, 640A, 644A, 684A, 744A & 784A) | Tektronix Document: 070-8709-07 """ # ENUMS ##
[docs] class Coupling(Enum): """ Available coupling options for input sources and trigger """ ac = "AC" dc = "DC" ground = "GND"
[docs] class Bandwidth(Enum): """ Bandwidth in MHz """ Twenty = "TWE" OneHundred = "HUN" TwoHundred = "TWO" FULL = "FUL"
[docs] class Impedance(Enum): """ Available options for input source impedance """ Fifty = "FIF" OneMeg = "MEG"
[docs] class Edge(Enum): """ Available Options for trigger slope """ Rising = 'RIS' Falling = 'FALL'
[docs] class Trigger(Enum): """ Available Trigger sources (AUX not Available on TDS520A/TDS540A) """ CH1 = 'CH1' CH2 = 'CH2' CH3 = 'CH3' CH4 = 'CH4' AUX = 'AUX' LINE = 'LINE'
[docs] class Source(Enum): """ Available Data sources """ CH1 = "CH1" CH2 = "CH2" CH3 = "CH3" CH4 = "CH4" Math1 = "MATH1" Math2 = "MATH2" Math3 = "MATH3" Ref1 = "REF1" Ref2 = "REF2" Ref3 = "REF3" Ref4 = "REF4"
# PROPERTIES ## @property def measurement(self): """ Gets a specific oscilloscope measurement object. The desired channel is specified like one would access a list. :rtype: `_TDS5xxMeasurement` """ return ProxyList(self, _TekTDS5xxMeasurement, range(3)) @property def channel(self): """ Gets a specific oscilloscope channel object. The desired channel is specified like one would access a list. For instance, this would transfer the waveform from the first channel:: >>> tek = ik.tektronix.TekTDS5xx.open_tcpip('192.168.0.2', 8888) >>> [x, y] = tek.channel[0].read_waveform() :rtype: `_TekTDS5xxChannel` """ return ProxyList(self, _TekTDS5xxChannel, range(4)) @property def ref(self): """ Gets a specific oscilloscope reference channel object. The desired channel is specified like one would access a list. For instance, this would transfer the waveform from the first channel:: >>> tek = ik.tektronix.TekTDS5xx.open_tcpip('192.168.0.2', 8888) >>> [x, y] = tek.ref[0].read_waveform() :rtype: `_TekTDS5xxDataSource` """ return ProxyList( self, lambda s, idx: _TekTDS5xxDataSource(s, "REF{}".format(idx + 1)), range(4) ) @property def math(self): """ Gets a data source object corresponding to the MATH channel. :rtype: `_TekTDS5xxDataSource` """ return ProxyList( self, lambda s, idx: _TekTDS5xxDataSource(s, "MATH{}".format(idx + 1)), range(3) ) @property def sources(self): """ Returns list of all active sources :rtype: `list` """ active = [] channels = map(int, self.query('SEL?').split(';')[0:11]) for idx in range(0, 4): if channels[idx]: active.append(_TekTDS5xxChannel(self, idx)) for idx in range(4, 7): if channels[idx]: active.append(_TekTDS5xxDataSource(self, "MATH{}".format( idx - 3))) for idx in range(7, 11): if channels[idx]: active.append( _TekTDS5xxDataSource(self, "REF{}".format(idx - 6))) return active @property def data_source(self): """ Gets/sets the the data source for waveform transfer. :type: `TekTDS5xx.Source` or `_TekTDS5xxDataSource` :rtype: '_TekTDS5xxDataSource` """ name = self.query("DAT:SOU?") if name.startswith("CH"): return _TekTDS5xxChannel(self, int(name[2:]) - 1) return _TekTDS5xxDataSource(self, name) @data_source.setter def data_source(self, newval): if isinstance(newval, _TekTDS5xxDataSource): newval = TekTDS5xx.Source[newval.name] if not isinstance(newval, TekTDS5xx.Source): raise TypeError("Source setting must be a `TekTDS5xx.Source`" " value, got {} instead.".format(type(newval))) self.sendcmd("DAT:SOU {}".format(newval.value)) time.sleep(0.01) # Let the instrument catch up. @property def data_width(self): """ Gets/Sets the data width for waveform transfers :type: `int` """ return int(self.query("DATA:WIDTH?")) @data_width.setter def data_width(self, newval): if int(newval) not in [1, 2]: raise ValueError("Only one or two byte-width is supported.") self.sendcmd("DATA:WIDTH {}".format(newval)) @property def force_trigger(self): raise NotImplementedError @property def horizontal_scale(self): """ Get/Set Horizontal Scale :type: `float` """ return float(self.query('HOR:MAI:SCA?')) @horizontal_scale.setter def horizontal_scale(self, newval): self.sendcmd("HOR:MAI:SCA {0:.3E}".format(newval)) resp = float(self.query('HOR:MAI:SCA?')) if newval != resp: raise ValueError("Tried to set Horizontal Scale to {} but got {}" " instead".format(newval, resp)) @property def trigger_level(self): """ Get/Set trigger level :type: `float` """ return float(self.query('TRIG:MAI:LEV?')) @trigger_level.setter def trigger_level(self, newval): self.sendcmd("TRIG:MAI:LEV {0:.3E}".format(newval)) resp = float(self.query('TRIG:MAI:LEV?')) if newval != resp: raise ValueError("Tried to set trigger level to {} but got {}" " instead".format(newval, resp)) @property def trigger_coupling(self): """ Get/Set trigger coupling :type: `TekTDS5xx.Coupling` """ return TekTDS5xx.Coupling[self.query("TRIG:MAI:EDGE:COUP?")] @trigger_coupling.setter def trigger_coupling(self, newval): if not isinstance(newval, TekTDS5xx.Coupling): raise TypeError("Coupling setting must be a `TekTDS5xx.Coupling`" " value, got {} instead.".format(type(newval))) self.sendcmd("TRIG:MAI:EDGE:COUP {}".format(newval.value)) @property def trigger_slope(self): """ Get/Set trigger slope :type: `TekTDS5xx.Edge` """ return TekTDS5xx.Edge(self.query("TRIG:MAI:EDGE:SLO?")) @trigger_slope.setter def trigger_slope(self, newval): if not isinstance(newval, TekTDS5xx.Edge): raise TypeError("Edge setting must be a `TekTDS5xx.Edge`" " value, got {} instead.".format(type(newval))) self.sendcmd("TRIG:MAI:EDGE:SLO {}".format(newval.value)) @property def trigger_source(self): """ Get/Set trigger source :type: `TekTDS5xx.Trigger` """ return TekTDS5xx.Trigger(self.query("TRIG:MAI:EDGE:SOU?")) @trigger_source.setter def trigger_source(self, newval): if not isinstance(newval, TekTDS5xx.Trigger): raise TypeError("Trigger source setting must be a" "`TekTDS5xx.source` value, got {} instead.".format(type(newval))) self.sendcmd("TRIG:MAI:EDGE:SOU {}".format(newval.value)) @property def clock(self): """ Get/Set oscilloscope clock :type: `datetime.datetime` """ resp = self.query('DATE?;:TIME?') return datetime.strptime(resp, '"%Y-%m-%d";"%H:%M:%S"') @clock.setter def clock(self, newval): if not isinstance(newval, datetime): raise ValueError("Expected datetime.datetime" "but got {} instead".format(type(newval))) self.sendcmd(newval.strftime('DATE "%Y-%m-%d";:TIME "%H:%M:%S"')) @property def display_clock(self): """ Get/Set the visibility of clock on the display :type: `bool` """ return bool(int(self.query('DISPLAY:CLOCK?'))) @display_clock.setter def display_clock(self, newval): if not isinstance(newval, bool): raise ValueError("Expected bool but got" "{} instead".format(type(newval))) self.sendcmd('DISPLAY:CLOCK {}'.format(int(newval)))
[docs] def get_hardcopy(self): """ Gets a screenshot of the display :rtype: `string` """ self.sendcmd('HARDC:PORT GPI;HARDC:LAY PORT;:HARDC:FORM BMP') self.sendcmd('HARDC START') sleep(1) header = self.query("", size=54) # Get BMP Length in kilobytes from DIB header, because file header is # bad length = reduce( operator.mul, struct.unpack('<iihh', header[18:30])) / 8 length = int(length) + 8 # Add 8 bytes for our monochrome colour table data = header + self.query("", size=length) self._file.flush_input() # Flush input buffer return data