Рейтинг@Mail.ru

Управление измерительными приборами Rohde-Schwarz по интерфейсу RSIB

автор:
Be the first to comment! Программирование
Print Friendly, PDF & Email

Измерительная аппаратура компании Rohde&Schwarz – это профессиональная аппаратура, которая используется в рабочих процессах на многих предприятиях по всему миру. Зачастую необходимо обеспечивать удалённый доступ и управление измерениями и режимами работы измерительной аппаратуры. Для этого у приборов Rohde&Schwarz есть, как правило, несколько возможностей. Одна из них – интерфейс RSIB. Рассмотрим его подробнее.

1Описание интерфейсаудалённого управления RSIB

Интерфейс RSIB использует динамически загружаемую библиотеку RSIB32.DLL, которая содержит несколько экспортируемых функций для работы с измерительными приборами. Эти функции позволяют передавать прибору по локальной сети команды и директивы, а также получать его ответы. Список экспортируемых функций приведён ниже.

Список экспортируемых функций rsib32.dll (Rohde&Schwarz)
Список экспортируемых функций rsib32.dll (Rohde&Schwarz)

Вот какие функции имеются в библиотеке.

Описание экспортируемых функций библиотеки RSIB32.DLL
ФункцияНазначение
RSDLLibfind()Пытается подключиться к измерительному прибору по IP-адресу.
RSDLLibonl()Переводит прибор в режим online/offline. После перевода в режим offline необходимо заново подключаться к прибору с помощью RSDLLibfind().
RSDLLibwrt()Отправляет прибору строку.
RSDLLilwrt()Отправляет прибору массив байтов.
RSDLLibwrtf()Отправляет прибору содержимое файла.
RSDLLibrd()Читает из прибора строку.
RSDLLilrd()Читает из прибора заданное число байтов.
RSDLLibrdf()Читает из прибора данные в файл.
RSDLLibloc()Временно переводит прибор в режим LOCAL.
RSDLLibeot()Включает или выключает отправку конца сообщения после операции записи.
RSDLLibrsp()Опрашивает регистр статуса прибора.
RSDLLibtmo()Задаёт таймаут для функций RSIB.
RSDLLibsre()Переводит прибор в режим LOCAL или режим REMOTE.
RSDLLTestSRQ()Проверяет значение бита SRQ прибора.
RSDLLWaitSrq()Ожидает одно из двух событий: 1) прибор установит бит SRQ; 2) бит SRQ не будет установлен прибором в течение времени, заданного тайм-аутом.
RSDLLSwapBytes()Переключает отображение байтов с Big Endian на Little Endian и обратно. Требуется на платформах, отличных от Intel (x86).

Предлагается сделать класс-обёртку для данной библиотеки в объектно-ориентированном стиле. Для использования базовой функциональности библиотеки некоторые её функции можно оставить без реализации. Нам точно пригодятся функции для подключения к прибору, перевода в режим REMOTE/LOCAL, записи и чтения строк. Эти функции будут нужны для отправки прибору команд SCPI.

Программный код для работы по интерфейсу RSIB (разворачивается)
Imports System.Runtime.InteropServices
Imports System.Text

Namespace RS

    ''' <summary>
    ''' Работа с устройствами Rohde-Schwarz по протоколу RSIB.
    ''' </summary>
    Public Class RSIB
        Implements IDisposable

#Region "CTOR"

		Private Const LibPath As String = "rsib32.dll"
        Private Const CLOSED_HANDLE As UShort = &H8000US

        ''' <summary>
        ''' Дескриптор устройства. Присваивается во время открытия устройства.
        ''' </summary>
        Private ReadOnly Property DevHandle As UShort
            Get
                Return _DevHandle
            End Get
        End Property
        Private _DevHandle As UShort = CLOSED_HANDLE

        ''' <summary>
        ''' Ищет устройство RSIB и подключается к нему по IP-адресу <paramref name="ipAddr"/>.
        ''' </summary>
        ''' <param name="ipAddr">IP-адрес устройства.</param>
        Public Sub New(ipAddr As String)
            Dim ibsta As Status 'флаги статуса
            Dim iberr As Errors 'код ошибки
            Dim ibcntl As UInteger 'счётчик показывает число переданных байтов
            _DevHandle = RSDLLibfind(ipAddr, ibsta, iberr, ibcntl)
            CheckResult(ibsta, iberr)
        End Sub

        <DllImport(LibPath, CharSet:=CharSet.Ansi)>
        Private Shared Function RSDLLibfind(udName As String, ByRef ibsta As Status, ByRef iberr As Errors, ByRef ibcntl As UInteger) As UShort
        End Function

        ''' <summary>
        ''' Отключается от устройства.
        ''' </summary>
        ''' <remarks>
        ''' После закрытия устройства программный доступ к устройству становится невозможен.
        ''' </remarks>
        Public Sub Close()
            If (_DevHandle <> CLOSED_HANDLE) Then
                Try
                    Dim ibsta As Status
                    Dim iberr As Errors
                    Dim ibcntl As UInteger
                    Dim res As Integer = RSDLLibonl(DevHandle, Modes.Local, ibsta, iberr, ibcntl)
                    _DevHandle = CLOSED_HANDLE
                Catch ex As Exception
                    Debug.WriteLine(ex)
                End Try
            End If
        End Sub

        <DllImport(LibPath, CharSet:=CharSet.Ansi)>
        Private Shared Function RSDLLibonl(udName As Integer, mode As Modes, ByRef ibsta As Status, ByRef iberr As Errors, ByRef ibcntl As UInteger) As UShort
        End Function

#End Region '/CTOR

#Region "ЗАПИСЬ, ЧТЕНИЕ"

        ''' <summary>
        ''' Отправляет команду и читает ответ.
        ''' </summary>
        Public Function SendCommand(command As String) As String
            Dim ibsta As Status
            Dim iberr As Errors
            Dim ibcntl As UInteger
            Dim res As Integer = RSDLLibwrt(DevHandle, command & vbNullChar, ibsta, iberr, ibcntl)
            CheckResult(ibsta, iberr)
            If command.Contains("?"c) Then
                Dim sbAns As New StringBuilder(100)
                Dim ans As Integer = RSDLLibrd(DevHandle, sbAns, ibsta, iberr, ibcntl)
                CheckResult(ibsta, iberr)
                Return sbAns.ToString()
            End If
            Return String.Empty
        End Function

        <DllImport(LibPath, CharSet:=CharSet.Ansi)>
        Private Shared Function RSDLLibwrt(udName As Integer, data As String, ByRef ibsta As Status, ByRef iberr As Errors, ByRef ibcntl As UInteger) As Integer
        End Function

        <DllImport(LibPath, CharSet:=CharSet.Ansi)>
        Private Shared Function RSDLLibrd(udName As Integer, data As StringBuilder, ByRef ibsta As Status, ByRef iberr As Errors, ByRef ibcntl As UInteger) As UShort
        End Function

#End Region '/ЗАПИСЬ, ЧТЕНИЕ 

#Region "УПРАВЛЕНИЕ РЕЖИМОМ"

        ''' <summary>
        ''' Переключает устройство в режим <paramref name="mode"/>.
        ''' </summary>
        ''' <remarks>
        ''' После переключения в режим <see cref="Modes.TemporarilyLocal"/> устройство может управляться вручную через фронтальную панель.
        ''' При следующем удалённом доступе устройство снова переключится в режим <see cref="Modes.Remote"/>.
        ''' </remarks>
        Public Sub SetMode(mode As Modes)
            Dim ibsta As Status
            Dim iberr As Errors
            Dim ibcntl As UInteger
            Select Case mode
                Case Modes.Local, Modes.Remote
                    Dim res As Integer = RSDLLibsre(DevHandle, mode, ibsta, iberr, ibcntl)
                Case Modes.TemporarilyLocal
                    Dim res As Integer = RSDLLibloc(DevHandle, ibsta, iberr, ibcntl)
            End Select
            CheckResult(ibsta, iberr)
        End Sub

        <DllImport(LibPath, CharSet:=CharSet.Ansi)>
        Private Shared Function RSDLLibsre(udName As Integer, mode As Modes, ByRef ibsta As Status, ByRef iberr As Errors, ByRef ibcntl As UInteger) As UShort
        End Function

        <DllImport(LibPath, CharSet:=CharSet.Ansi)>
        Private Shared Function RSDLLibloc(udName As Integer, ByRef ibsta As Status, ByRef iberr As Errors, ByRef ibcntl As UInteger) As UShort
        End Function

        ''' <summary>
        ''' Включает или выключает отправку конца сообщения после операции записи.
        ''' </summary>
        ''' <param name="sendEnd">Если False, то команды могут отправляться несколькими последовательными вызовами <see cref="SendCommand(String)"/>.
        ''' True должно быть снова установлено перед отправкой последнего блока данных.
        ''' </param>
        ''' <remarks>
        ''' If the END message is disabled, the data of a command can be sent with several successive calls of write functions. 
        ''' The END message must be enabled again before sending the last data block.
        ''' </remarks>
        Public Sub SetEndOfMessage(sendEnd As Boolean)
            Dim ibsta As Status
            Dim iberr As Errors
            Dim ibcntl As UInteger
            Dim res As Integer = RSDLLibeot(DevHandle, sendEnd, ibsta, iberr, ibcntl)
            CheckResult(ibsta, iberr)
        End Sub

        <DllImport(LibPath, CharSet:=CharSet.Ansi)>
        Private Shared Function RSDLLibeot(udName As Integer, <MarshalAs(UnmanagedType.Bool)> sendEnd As Boolean, ByRef ibsta As Status, ByRef iberr As Errors, ByRef ibcntl As UInteger) As UShort
        End Function

#End Region '/УПРАВЛЕНИЕ РЕЖИМОМ       

#Region "ERRORS, STATUS"

        ''' <summary>
        ''' Опрашивает устройство и возвращает его регистр статуса.
        ''' </summary>
        Public Function GetDeviceStatusRegister() As Byte
            Dim ibsta As Status
            Dim iberr As Errors
            Dim ibcntl As UInteger
            Dim spr As Byte
            Dim res As Integer = RSDLLibrsp(DevHandle, spr, ibsta, iberr, ibcntl)
            CheckResult(ibsta, iberr)
            Return spr
        End Function

        <DllImport(LibPath, CharSet:=CharSet.Ansi)>
        Private Shared Function RSDLLibrsp(udName As Integer, ByRef spr As Byte, ByRef ibsta As Status, ByRef iberr As Errors, ByRef ibcntl As UInteger) As UShort
        End Function

        ''' <summary>
        ''' Проверяет статус и ошибки.
        ''' </summary>
        Private Sub CheckResult(stat As Status, err As Errors)
            If IsErrorOccured(stat) Then
                Throw New Exception(GetErrorText(err))
            ElseIf IsTimeout(stat) Then
                Throw New Exception("Превышено время ожидания.")
            End If
        End Sub

        ''' <summary>
        ''' Возвращает текст ошибки.
        ''' </summary>
        Private Function GetErrorText(err As Errors) As String
            Select Case err
                Case Errors.IBERR_TIMEOUT
                    Return "Превышено время ожидания."
                Case Errors.IBERR_BUSY
                    Return "Протокол RSIB заблокирован функцией, которая ещё выполняется."
                Case Errors.IBERR_CONNECT
                    Return "Подключение к инструменту не удалось."
                Case Errors.IBERR_NO_DEVICE
                    Return "Функция была вызвана с некорректным идентификатором устройства."
                Case Errors.IBERR_MEM
                    Return "Нет свободной памяти."
                Case Errors.IBERR_FILE
                    Return "Ошибка во время чтения или записи в файл."
                Case Errors.IBERR_SEMA
                    Return "Ошибка создания или присваивания семафора (только для UNIX)."
                Case Else
                    Return "Ошибка во время работы с устройством."
            End Select
        End Function

        Private Function IsAnswerReady(stat As Status) As Boolean
            Return ((stat And Status.CMPL) = Status.CMPL)
        End Function

        Private Function IsTimeout(stat As Status) As Boolean
            Return ((stat And Status.TIMO) = Status.TIMO)
        End Function

        Private Function IsErrorOccured(stat As Status) As Boolean
            Return ((stat And Status.ERR) = Status.ERR)
        End Function

#End Region '/ERRORS, STATUS

#Region "ENUMS"

        ''' <summary>
        ''' Состяния устройства.
        ''' </summary>
        <Flags()>
        Public Enum Status As UShort
            ''' <summary>
            ''' Флаг установлен, когда произошла ошибка во время вызова функции.
            ''' </summary>
            ''' <remarks>
            ''' При наличии этого флага в статусе необходимо проверить код ошибки для выяснения деталей.
            ''' </remarks>
            ERR = &H8000
            ''' <summary>
            ''' Флаг установлен, когда превышено время ожидания во время вызова функции.
            ''' </summary>
            TIMO = &H4000
            ''' <summary>
            ''' Флаг установлен, когда устройство завершило обработку принятой команды.
            ''' Флаг сброшен, когда ответ устройства прочитан или длина буфера недостаточна для ответа.
            ''' </summary>
            CMPL = &H100
        End Enum

        ''' <summary>
        ''' Коды ошибок.
        ''' </summary>
        Public Enum Errors As UShort
            None = 0
            ''' <summary>
            ''' Подключение к инструменту не удалось.
            ''' </summary>
            IBERR_CONNECT = 2
            ''' <summary>
            ''' Функция была вызвана с некорректным идентификатором устройства.
            ''' </summary>
            IBERR_NO_DEVICE = 3
            ''' <summary>
            ''' Нет свободной памяти.
            ''' </summary>
            IBERR_MEM = 4
            ''' <summary>
            ''' Превышено время ожидания.
            ''' </summary>
            IBERR_TIMEOUT = 5
            ''' <summary>
            ''' Протокол RSIB заблокирован функцией, которая ещё выполняется.
            ''' </summary>
            IBERR_BUSY = 6
            ''' <summary>
            ''' Ошибка во время чтения или записи в файл.
            ''' </summary>
            IBERR_FILE = 7
            ''' <summary>
            ''' Ошибка создания или присваивания семафора (только для UNIX).
            ''' </summary>
            IBERR_SEMA = 8
        End Enum

        Public Enum Modes As UShort
            Local = 0
            Remote = 1
            ''' <summary>
            ''' Временно переключить в локальный режим. После получения команды снова переходит в удалённый режим.
            ''' </summary>
            TemporarilyLocal = 2
        End Enum

#End Region '/ENUMS

#Region "IDISPOSABLE"

        Private DisposedValue As Boolean

        Protected Overridable Sub Dispose(disposing As Boolean)
            If (Not DisposedValue) Then
                Close()
                DisposedValue = True
            End If
        End Sub

        Public Sub Dispose() Implements IDisposable.Dispose
            Dispose(disposing:=True)
            GC.SuppressFinalize(Me)
        End Sub

#End Region '/IDISPOSABLE

    End Class

End Namespace

Последняя версия кода всегда доступна на Гитхабе, а детальное описание работы по интерфейсу RSIB приведены на официальном сайте Rohde&Schwarz. Библиотеку rsib32.dll можно скачать в конце статьи.

2Использование класса-обёртки RSIBпри работе с аппаратурой Rohde&Schwarz

Использование приведённого программного кода очень простое. Например, запросим у прибора его идентификатор, используя синтаксис SCPI, который поддерживается большинством измерительной аппаратуры (и приборами фирмы Rohde&Schwarz в частности), а затем переведём его в локальный режим:

    Using rs As New RS.RSIB("192.168.0.200") 'ввести свой IP-адрес прибора R&S
        Dim id As String = rs.SendCommand("*IDN?")
        rs.SetMode(RSIB.Modes.Local)
        Console.WriteLine($"Прибор {id} переведён в локальный режим.")
    End Using

Естественно, что необходимо выполнять данный код в блоке обработки ошибок Try…Catch, т.к. если программа не найдёт по указанному IP-адресу измерительный прибор, она выбросит исключение.

Download attachments:

Last modified onЧетверг, 28 Октябрь 2021 08:04 Read 4708 times
Ключевые слова: :

Поблагодарить автора:

Поделиться

Print Friendly, PDF & Email