# Open Neuro Interface Specification Version 0.2

### Jonathan P. Newman, Wilson Lab, MIT

### October 28, 2019

5 Abstract

This document specifies requirements for implementing an Open Neuro Interface (ONI) acquisition system in hardware and software. This specification entails two basic elements: (1) Communication protocols between acquisition firmware and host software and (2) an application programming interface (API) for utilizing this communication protocol. This document is incomplete and we gratefully welcome criticisms and amendments.

### Contents

| 11 | Intentions and capabilities                                 | 2  |
|----|-------------------------------------------------------------|----|
| 12 | FPGA/Host PC communication                                  | 3  |
| 13 | Signal channel (8-bit, asynchronous, read only)             | 3  |
| 14 | Configuration channel (32-bit, synchronous, read and write) |    |
| 15 | Data read channel (32-bit, asynchronous, read-only)         |    |
| 16 | Data write channel (32-bit, asynchronous, write-only)       |    |
| 17 | Device Driver Specification                                 | 6  |
| 18 | Required API Types and Behavior                             | 7  |
| 19 | Context                                                     |    |
| 20 | Device                                                      | 7  |
| 21 | Frame                                                       | 8  |
| 22 | liboepcie: An Open Ephys ++ API Implementation              | 9  |
| 23 | Scope and External Dependencies                             | 9  |
| 24 | License                                                     | 9  |
| 25 | Types                                                       | 9  |
| 26 | oe_create_ctx                                               | 12 |
| 27 | oe_init_ctx                                                 | 13 |
| 28 | oe_destroy_ctx                                              | 13 |
| 29 | oe get opt                                                  |    |
| 30 | oe set opt                                                  | 16 |
| 31 | oe_read_reg                                                 | 19 |
| 32 | oe_write_reg                                                | 19 |
| 33 | oe read frame                                               | 20 |
| 34 | oe destroy frame                                            | 20 |
| 35 | oe write                                                    | 21 |
| 36 | oe version                                                  |    |
| 37 | oe error st                                                 |    |
| 38 | oe_device_str                                               |    |
| 39 | Example Headstage Serialization Protocol                    | 23 |

## 40 Intentions and capabilities

- Potential for low latency round trip times (sub millisecond)
- Potential for high bandwidth communication (> 1000 neural data channels)
  - Bidirectional communication
    - Acquisition and control of arbitrary of hardware components using a single communication medium
      - Support generic mixes of hardware elements from multiple, asynchronous pieces of hardware
      - Generic hardware configuration
      - Generic data input streams
      - Generic data output streams
  - Support multiple acquisition systems on one computer
    - Cross platform

41

43

44

45

46

47

48

49

51

• Low level: aimed at the creation of firmware, APIs, language bindings and application-specific libraries

## 52 FPGA/Host PC communication

- 53 Communication between the acquisition board firmware and API shall occur at least four communication channels:
  - 1. Signal: Read-only, short-message, asynchronous hardware events. Only one signal channel is permitted.
  - 2. Configuration: Bidirectional, register-based, synchronous configuration setting and getting. Only one configuration channel is permitted.
    - 3. Input: Read-only, asynchronous, high-bandwidth firmware to host streaming. More than one input channel is permitted.
  - 4. Output: Write-only, asynchronous, high-bandwidth host to firmware streaming. More than one output channel is permitted.
- 61 Required characteristics of these channels are described in the following paragraphs.

### Signal channel (8-bit, asynchronous, read only)

The *signal* channel provides a way for the FPGA firmware to inform host of configuration results, which may be provided with a significant delay. Additionally, it allows the host to read the device map supported by the FPGA firmware. The behavior of the signal channel is equivalent to a read-only, blocking UNIX named pipe. Signal data is framed into packets using Consistent Overhead Byte Stuffing (COBS). Within this scheme, packets are delimited using 0's and always have the following format:

```
3 ... | PACKET_FLAG data | ...
```

54

55

56

57

58

59

60

where PACKET\_FLAG is 32-bit unsigned integer with a single unique bit setting, | represents a packet delimiter, and ... represents other packets. This stream can be read and ignored until a desired packet is received. Reading this stream shall block if no data is available, which allows asynchronous configuration acknowledgment. Valid PACKET\_FLAGs are:

```
enum signal {
    NULLSIG = (1u << 0), // Null signal, ignored by host
    CONFIGWACK = (1u << 1), // Configuration write-acknowledgment
    CONFIGWNACK = (1u << 2), // Configuration no-write-acknowledgment
    CONFIGRACK = (1u << 3), // Configuration read-acknowledgment
    CONFIGRNACK = (1u << 4), // Configuration no-read-acknowledgment
    DEVICEMAPACK = (1u << 5), // Device map start acknowledgment
    DEVICEINST = (1u << 6), // Device map instance
};</pre>
```

Following a hardware reset, the signal channel is used to provide the device map to the host using the following packet sequence:

```
75 ... | DEVICEMAPACK, uint32_t num_devices | DEVICEINST device dev_0 | DEVICEINST device dev_1 | ... | DEVICEINST device dev_n | ...
```

Following a device register read or write (see configuration channel), ACK or NACK signals are pushed onto the signal stream by the firmware. For instance, on a successful register read:

```
9 ... | CONFIGRACK, uint32_t register value | ...
```

### Configuration channel (32-bit, synchronous, read and write)

The configuration channel supports seeking to, reading, and writing a set of configuration registers. Its behavior is equivalent to that of a normal UNIX file. There are two classes of registers handled by the configuration channel: the first set of registers encapsulates a generic device register programming interface. The remaining registers are for global context control and configuration and provide access to acquisition parameters and state control. SEEK locations of each configuration register, relative to the start of the stream, should be hard-coded into the API implementation file and used in the background to manipulate register state.

#### 87 Device register programming interface

89

91

92

93

95

97

98

100

101

102

104

105

106

108

109

111

112

113

116

117

118

119

120

121

122

124

125

126

127

128

129

130

131

132

134

135

- The device programming interface is composed of the following configuration channel registers:
  - uint32\_t config\_device\_idx: Device index register. Specify a device endpoint as enumerated in the device map by the firmware (e.g. an Intan chip, or a IMU chip) and to which communication will be directed using config\_reg\_addr and config\_reg\_value, as described below.
  - uint32\_t config\_reg\_addr: The register address of configuration to be written
  - uint32\_t config\_reg\_value: configuration value to be written to or read from and that corresponds to config\_reg\_addr on device config\_device\_id
    - uint32\_t config\_rw: A flag indicating if a read or write should be performed. 0 indicates read operation. A value > 0 indicates write operation.
    - uint32\_t config\_trig: Set > 0 to trigger either register read or write operation depending on the state of config\_rw. If config\_rw is 0, a read is performed. In this case config\_reg\_value is updated with value stored at config\_reg\_addr on device at config\_device\_id. If config\_rw is 1, config\_reg\_value is written to register at config\_reg\_addr on device config\_device\_id. The config\_trig register is always be set low by the firmware following transmission even if it is not successful or does not make sense given the address register values.

Appropriate values of config\_reg\_addr and config\_reg\_value are determined by:

- Looking at a device's data sheet if the device is an integrated circuit
- Examining the open ephys++ devices header file (oedevices.h) which contains off register addresses and descriptions for devices officially supported by this project.

When a host requests a device register read, the following following actions take place:

- 1. The value of config\_trig is checked.
  - If it is 0x00, the function call proceeds.
  - Else, the function call returns with an error specifying a retrigger.
- 2. dev\_idx is copied to the config\_device\_id register on the host FPGA.
- 3. addr is copied to the config\_reg\_addr register on the host FPGA.
- 4. The config\_rw register on the host FPGA is set to 0x00.
- 5. The config\_read\_trig register on the host FPGA is set to 0x01, triggering configuration transmission by the firmware.
  - 6. (Firmware) A configuration read is performed by the firmware.
    - 7. (Firmware) config\_trig is set to 0x00 by the firmware.
  - 8. (Firmware) CONFIGRACK is pushed onto the signal stream by the firmware.
  - 9. The signal stream is pumped until either CONFIGRACK or CONFIGRNACK is received indicating that the host FPGA has either:
    - Completed reading the specified device register and copied its value to the config reg value register.
    - Failed to read the register in which case the value of config\_reg\_value contains garbage.

When a host requests a device register write, the following following actions take place:

- 1. The value of config\_trig is checked.
  - If it is 0x00, the function call proceeds.
  - Else, the function call returns with an error specifying a retrigger.
- 2. dev\_idx is copied to the config\_device\_id register on the host FPGA.
  - 3. addr is copied to the config\_reg\_addr register on the host FPGA.
- 4. value is copied to the config\_reg\_value register on the host FPGA.
- 5. The config\_rw register on the host FPGA is set to 0x01.
- 6. The config\_trig register on the host FPGA is set to 0x01, triggering configuration transmission by the firmware.
- 7. (Firmware) A configuration write is performed by the firmware.
  - 8. (Firmware) config\_trig is set to 0x00 by the firmware.
    - 9. (Firmware) CONFIGWACK is pushed onto the signal stream by the firmware.

- 10. The signal stream is pumped until either CONFIGWACK or CONFIGWNACK is received indicating that the host FPGA has either:
  - Successfully completed writing the specified device register
  - Failed to write the register

Following successful or unsuccessful device register read or write, the appropriate ACK or NACK packets *must* be passed to the signal channel. If they are not, the register read and write calls will block indefinitely.

### 142 Global acquisition registers

136

137

138

139

144

145

147

148

149

The following global acquisition registers provide information about, and control over, the entire acquisition system:

- uint32\_t running: set to > 0 to run the system clock and produce data. Set to 0 to stop the system clock and therefore stop data flow. Results in no other configuration changes.
- uint32\_t reset: set to > 0 to trigger a hardware reset and send a fresh device map to the host and reset hardware to its default state. Set to 0 by host firmware upon entering the reset state.
- uint32\_t sys\_clock\_hz: A read-only register specifying the master (clock domain 0) hardware clock frequency in Hz. The clock counter in the read frame header is incremented at this frequency.

### Data read channel (32-bit, asynchronous, read-only)

The data read channel provides high bandwidth communication from the FPGA firmware to the host computer using direct memory access (DMA). From the host's perspective, its behavior is equivalent to a read-only, blocking UNIX named pipe with the exception that data can only be read on 32-bit, instead of 8-bit, boundaries. The data input channel communicates with the host using frames with a read-header ("read-frames") .Read-frames are pushed into the data input channel at a rate dictated by the FPGA firmware. It is incumbent on the host to read this stream fast enough to prevent buffer overflow. At the time of this writing, a typical implementation will allocate an input buffer that occupies a 512 MB segment of kernal RAM. Increased bandwidth demands will necessitate the creation of a user-space buffer. This change shall have no effect on the API.

### Data write channel (32-bit, asynchronous, write-only)

The data write channel provides high bandwidth communication from the host computer to the FPGA firmware using DMA via calls. From the host's perspective, its behavior is equivalent to a write-only, blocking UNIX named pipe with the exception that data can only be written on 32-bit, instead of 8-bit, boundaries. Its performance characteristics are largely identical to the data input channel.

Device Driver Specification

165 TODO

## 166 Required API Types and Behavior

In the following sections we define required API datatypes and how they are used by the API to communicate with hardware. An implementation of this API, liboepcie, follows.

#### 169 Context

A context shall hold all state required to manage single FPGA/Host communication system. This includes a device map (simple list of devices) being acquired from, data buffering elements, etc. API calls will typically take a context handle as the first argument and use it to reference required state information to enable communication and/or to mutate the context to reflect some function side effect (e.g. add device map information):

```
int api_function(context *ctx, ...);
```

#### Device

180

181

182

183

184

185

186

187

191

192

193

194

195

197

198 199

201

202

203

204

206

207

209

210

A device is defined as configurable piece of hardware with its own register address space (e.g. an integrated circuit)
or something programmed within the firmware to emulate this (e.g. an electrical stimulation sub-circuit made to
behave like a Master-8). Host interaction with a device is facilitated using a device description, which holds the
following elements:

- device\_id: Device ID number
- slot: The index of the physical interface that this device uses for host communication
- clock\_dom: Device clock domain (0 is master, 1 or greater are slaves synchronized to master)
- clock\_hz: Clock rate in Hz of clock converning clock\_dom
- read\_size: Device data read size per frame in bytes
  - num\_reads: Number of frames that must be read to construct a full sample (e.g., for row reads from camera)
  - write\_size: Device data write size per frame in bytes
- num\_writes: Number of frames that must be written to construct a full output sample

An array of structures holding each of these entries forms a *device map*. A context is responsible for managing a single device map, which keep track of where to send and receive streaming data and configuration information during API calls. A detailed description of each of each value comprising a device instance is as follows:

- 1. device\_id: Device identification number which is globally enumerated for the entire project
  - There is a single enum for the entire library which enumerates all possible devices that are controlled across context configurations. This enumeration will grow with the number of devices supported by the library.
  - e.g. A host board GPIO subcircuit is 0, Intan RHD2132 is 1, Intan RHD2164 is 2, etc.
  - Device IDs up to 9999 are reserved. Device ID 10000 and greater are free to use for custom hardware projects.
  - The use of device IDs less than 10000 not specified within this enumeration will result in OE\_EDEVID errors.
  - Device numbers greater than 9999 are allowed for general purpose use and will not be verified by the API.
  - Incorporation into the official device enum (device IDs < 10000) can be achieved via pull-request to the ONI repository.
- 2. slot: The index of the physical interface that this device uses for host communication
  - e.g. the PCIe slot index
  - e.g. the USB port index
  - Typically, host hardware will be assigned an index in non-volatile memory or via dip switch configuration.
- 3. clock\_dom: The clock domain that the device is sychronized to.
  - All devices exist in a single clock domain
  - There are one or more clock domains per device map

- 4. clock\_hz: The clock rate in Hz of the clock governing clock\_dom
- 5. read size: Number of bytes of data transmitted by this device during a single read.
  - 0 indicates that it does not send data.
- 6. num\_reads: Number of reads required to construct a full device read sample (e.g., number of columns when read\_size corresponds to a single row of pixels from a camera sensor)
- 7. write\_size: Number of bytes accepted by the device during a single write
  - 0 indicates that it does not send data.
  - 8. num\_writes: Number of writes required to construct a full device output sample.

#### 219 Frame

211

213

217

227

228

229

230

231

232

233

234

235

236

237

238

239

240

242

244

245

246

A frame is a flat byte array containing a single sample's worth of data for a set (one to all) of devices within a device map. Data within frames is arranged into three memory sectors as follows:

225 Each frame memory sector is described below:

- 1. Header
  - Each frame starts with a 32-byte header
  - For reading (firmware to host) operations, the header contains
    - bytes 0-7: unsigned 64-bit integer holding system clock counter
    - bytes 8-9: unsigned 16-bit integer indicating number of devices that the frame contains data for
    - byte 10: 8-bit integer speficying fame error state. frame error. 0 = OK. 1 = data may be corrupt.
    - bytes 11-32: reserved
  - For writing (host to firmware) operations, the header contains
    - bytes 0-32: reserved
- 2. Device map indices
  - An array of unsigned 32-bit keys corresponding the device map captured by the host during context initialization
  - The offset, size, and type information of the \_i\_th data block within the data section of each frame is determined by examining the \_i\_th member of the device map.
- 3. Data
  - Raw data blocks from each device in the device map.
  - The ordering of device-specific blocks is the same as the device index within the *device map index* portion of the frame
  - The read/write size for each device-specific block is provided in the device map
  - If timing information is passed in the data block, it should be specified how to interpret it (using plain text).

## liboepcie: An Open Ephys ++ API Implementation

### Scope and External Dependencies

liboepcie is a C library that implements the Open Ephys++ API Specification. It is written in C to facilitate cross platform and cross-language use. It is composed of two mutually exclusive file pairs: 250

- 1. oepcie.h and oepcie.c: main API implementation
- 2. oedevice.h and oedevices.c: officially supported device and register definitions. This file can be ignored for project that do not wish to conform to the official device specification.

liboepcie is a low level library used by high-level language binding and/or software plugin developers. It is not 254 meant to be used by neuroscientists directly. The only external dependency aside from the C standard library is is 255 a hardware communication backend that fulfills the requirements of the FPGA/Host Communication Specification. 256 An example of such a backend is Xillybus, which provides proprietary FPGA IP cores and free and open source device drivers to allow the communication channels to be implemented using the PCIe bus. From the API's 258 perspective, hardware communication abstracted to IO system calls (open, read, write, etc.) on file descriptors. File descriptor semantics and behavior are identical to either normal files (configuration channel) or named pipes 260 (signal, data input, and data output channels). Because of this, a drop in replacement for the Xillybus IP Core 261 can be used without any API changes. The development of a free and open-source FPGA cores that emulate the 262 functionality of Xillybus would be a major benefit to the systems neuroscience community. 263

Importantly, the low-level synchronization, resource allocation, and logic required to use the hardware communication backend is implicit to liboepcie API function calls. Orchestration of the communication backend is not 265 directly managed by the library user.

#### License

typedef struct oe\_ctx\_impl {

251

```
MIT
   Types
   Integer types
      • oe size t: Fixed width size integer type.
271
      • oe_dev_id_t: Fixed width device identity integer type.
272
      • oe reg addr t: Fixed width device register address integer type.
273
      • oe_reg_value_t: Fixed width device register value integer type.
   oe_ctx
   Context implementation. oe_ctx is an opaque handle to a context structure which contains hardware and device
276
   state information.
277
    // oepcie.h
278
   typedef struct oe_ctx_impl *oe_ctx;
   Context details are hidden in implementation file (oepcie.c):
   typedef struct stream_fid {
        char *path;
        int fid;
   } stream_fid_t;
```

```
// Communication channels
    stream_fid_t config;
    stream_fid_t read;
    stream_fid_t write;
    stream_fid_t signal;
    // Devices
    oe_size_t num_dev;
    oe_device_t* dev_map;
    // Maximum frame sizes (bytes)
    oe size t max read frame size;
    oe_size_t write_frame_size;
    // Data buffer
    uint8 t *buffer;
    uint8 t *buff read pos;
    uint8 t *buff end pos;
    // Acqusition state
    enum run_state {
        CTXNULL = 0,
        UNINITIALIZED,
        IDLE,
        RUNNING
    } run_state;
} oe_ctx_impl_t;
Each context manages a single device map. Following a hardware reset, which is triggered either by a call to
oe init ctx or to oe set opt using the OE RESET option, the context run state is set to UNINTIALIZED and
the device map is pushed onto the signal stream by the FPGA as COBS encode packets. On the signal stream, the
device map is organized as follows,
... | DEVICEMAPACK, uint32 t num devices | DEVICEINST oe device t dev 0 | DEVICEINST oe device t
dev 1 | ... | DEVICEINST oe device t dev n | ...
where | represents '0' packet delimiters. During a call to oe init ctx, the device map is decoded from the signal
stream. It can then be examined using calls to oe_get_opt using the OE_DEVICEMAP option. After the map is
received, the context run_state becomes IDLE. A call to oe_set_ctx with the OE_RUNNING option can then be
used to start acquisition by transitioning the context run_state to RUNNING.
oe_device_t
Device implementation. An oe_device_t describes one of potentially many pieces of hardware within a context.
Examples include Intan chips, IMUs, optical stimulator's, camera sensors, etc. Each valid device type has a unique
ID which is enumerated in the auxiliary oedevices.h file or some use-specific header. A map of available devices
is read from hardware and stored in the current context via a call to oe_init_ctx. This map can be examined via
calls to oe_get_opt.
typedef struct {
                              // Device ID number
    oe_dev_id_t id;
                               // Device slot
    oe_size_t slot;
    oe_size_t clock_dom;
                              // Device clock domain
                              // Clock rate in Hz of clock_dom
    oe_size_t clock_hz;
                              // Device data read size per frame in bytes
    oe_size_t read_size;
    oe size t num reads;
                              // Number of read frames to construct a full sample
    oe_size_t write_size;
                              // Device data write size per frame in bytes
```

282

283

284

286

287

289

292

293

```
oe size t num writes;
                             // Number of written frames comprising a full sample
} oe_device_t;
Officially supported device IDs and configuration register definitions are provided in oedevices.h as a set of enumer-
ations. A portion of the official device ID enumeration is defined as follows:
typedef enum device_id {
    OE_IMMEDIATEIO = 0,
    OE_RHD2132,
    OE_RHD2164,
    OE_MPU9250,
    OE_ESTIM,
    . . .
    OE MAXDEVICEID = 9999
} oe_device_id_t
An example of a device register (for the OE_ESTIM device ID) enumeration is:
enum oe estim regs {
                          = 0, // No command
    OE ESTIM NULLPARM
    OE_ESTIM_BIPHASIC
                          = 1, // Biphasic pulse (0 = monophasic, 1 = biphasic;
                          = 2,
                                // Phase 1 current, (0 to 255 = -1.5 \text{ mA} to +1.5 \text{mA})
    OE_ESTIM_CURRENT1
                          = 3, // Phase 2 voltage, (0 to 255 = -1.5 mA to +1.5mA)
    OE_ESTIM_CURRENT2
    OE_ESTIM_PULSEDUR1
                          = 4, // Phase 1 duration, 10 microsecond steps
                          = 5, // Inter-phase interval, 10 microsecond steps
    OE_ESTIM_IPI
    OE_ESTIM_PULSEDUR2
                          = 6, // Phase 2 duration, 10 microsecond steps
    OE_ESTIM_PULSEPERIOD = 7, // Inter-pulse interval, 10 microsecond steps
    OE_ESTIM_BURSTCOUNT = 8, // Burst duration, number of pulses in burst
                          = 9, // Inter-burst interval, microseconds
    OE_ESTIM_IBI
    OE_ESTIM_TRAINCOUNT = 10, // Pulse train duration, number of bursts in train
    OE ESTIM TRAINDELAY = 11, // Pulse train delay, microseconds
    OE_ESTIM_TRIGGER
                          = 12, // Trigger stimulation (1 = deliver)
    OE ESTIM POWERON
                          = 13, // Control estim sub-circuit power (0 = off, 1 = on)
                          = 14, // Control null switch (0 = stim output shorted to ground, 1 = enabled)
    OE_ESTIM_ENABLE
    OE ESTIM RESTCURR
                          = 15, // Current between pulse phases, (0 to 255 = -1.5 \text{ mA} to +1.5 \text{mA})
                          = 16, // Reset all parameters to default
    OE_ESTIM_RESET
};
These registers may be familiar to those who have used a Master-8 or pulse-pal stimulus sequencer.
oe_frame_t
Frame implementation. Frames are produced by calls oe_read_frame. Frames are actually zero-copy views into an
external, RAII-capable circular buffer (the buffer handle). When implementing language bindings, simply ignore
this member's existence.
typedef struct oe_frame {
    // Header
                             // Base clock counter
    uint64_t clock;
    uint16_t num_dev;
                             // Number of devices in frame
    uint8_t corrupt;
                             // Is this frame corrupt?
    // Data
    oe_size_t *dev_idxs;
                             // Array of device indices in frame
    oe_size_t *dev_offs;
                             // Device data offsets within data block
                             // Multi-device raw data block
    uint8_t *data;
                             // Size in bytes of data buffer
    oe_size_t data_sz;
    // External buffer, don't touch
```

```
oe buffer buffer;
                                // Handle to external buffer
   } oe_frame_t;
   oe_opt_t
   Context option enumeration. See the description of oe_set_opt and oe_get_opt for valid values.
   oe_error_t
   Error code enumeration.
   typedef enum oe_error {
                                   // Success
       OE_ESUCCESS
                               0.
                            = -1, // Invalid stream path, fail on open
       OE_EPATHINVALID
       OE_EDEVID
                            = -2,
                                   // Invalid device ID on init or reg op
                            = -3, // Invalid device index
       OE_EDEVIDX
       OE_EWRITESIZE
                            = -4, // Data write size is incorrect for designated device
                            = -5,
                                   // Failure to read from a stream/register
       OE_EREADFAILURE
                            = -6
                                   // Failure to write to a stream/register
       OE_EWRITEFAILURE
                            = -7, // Attempt to call function w null ctx
       OE_ENULLCTX
       OE_ESEEKFAILURE
                            = -8, // Failure to seek on stream
       OE EINVALSTATE
                            = -9.
                                   // Invalid operation for the current context run state
       OE_EINVALOPT
                            = -10, // Invalid context option
                            = -11, // Invalid function arguments
       OE EINVALARG
                            = -12, // Invalid COBS packet
       OE_ECOBSPACK
       OE ERETRIG
                            = -13, // Attempt to trigger an already triggered operation
                            = -14, // Supplied buffer is too small
       OE EBUFFERSIZE
                            = -15, // Badly formated device map supplied by firmware
       OE_EBADDEVMAP
                            = -16, // Bad dynamic memory allocation
       OE_EBADALLOC
       OE_ECLOSEFAIL
                            = -17, // File descriptor close failure, check errno
                            = -18, // Attempted write to read only object (register, context option, etc)
       OE_EREADONLY
       OE_EUNIMPL
                            = -19, // Specified, but unimplemented, feature
                            = -20, // Block read size is smaller than the maximal frame size
       OE_EINVALREADSIZE
   } oe_error_t;
   oe create ctx
   Create a hardware context. A context is an opaque handle to a structure which contains hardware and device state
310
   information, configuration capabilities, and data format information. It can be modified via calls to oe set opt.
311
   Its state can be examined by oe_get_opt.
312
   oe_ctx oe_create_ctx()
   Returns oe_ctx
   An opaque handle to the newly created context if successful. Otherwise it shall return NULL and set errno to
   EAGAIN.
315
   Description
316
```

On success a context struct is allocated and created, and its handle is passed to the user. The context holds all state used by the library function calls for reflection and hardware communication. It holds paths to FIFOs and

configuration communication channels and knowledge of the hardware's parameters and run state. It is configured through calls to oe\_set\_opt. It can be examined through calls to oe\_get\_opt.

```
oe_init_ctx
```

Initialize a context, opening all file streams etc.

```
int oe_init_ctx(oe_ctx ctx)
```

#### 323 Arguments

324

327

330

331

ctx context

#### 325 Returns int

- 0: success
  - Less than 0: oe\_error\_t

#### 328 Description

Upon a call to oe\_init\_ctx, the following actions take place

- 1. All required data streams are opened.
- 2. A device map is read from the firmware. It can be examined via calls t oe\_get\_opt.
- 33. The data transmission packet size is calculated and stored. It can be examined via calls t oe\_get\_opt.

Following a successful call to oe\_init\_ctx, the hardware's acquisition parameters and run state can be manipulated using calls to oe\_get\_opt.

### 335 oe\_destroy\_ctx

Terminate a context and free bound resources.

```
int oe_destroy_ctx(oe_ctx ctx)
```

### 337 Arguments

ctx context

#### 339 Returns int

- 0: success
- Less than 0: oe\_error\_t

### 42 Description

During context destruction, all resources allocated by oe\_create\_ctx are freed. This function can be called from any context run state. When called, an interrupt signal (TODO: Which?) is raised and any blocking operations will return immediately. Attached resources (e.g. file descriptors and allocated memory) are closed and their resources freed.

### $oe_get_opt$

Get context options.

```
int oe_get_opt(const oe_ctx ctx, int option, void* value, size_t *size);
```

### 349 Arguments

351

353

- ctx context to read from
  - option option to read
- value buffer to store value of option
  - size pointer to the size of value (including terminating null character, if applicable) in bytes

#### 354 Returns int

- 0: success
- Less than 0: oe\_error\_t

#### 357 Description

The oe\_get\_opt function sets the option specified by the option argument to the value pointed to by the value argument for the context pointed to by the ctx argument. The size provides a pointer to the size of the option value in bytes. Upon successful completion oe\_get\_opt shall modify the value pointed to by size to indicate the actual size of the option value stored in the buffer.

Following a successful call to oe\_init\_ctx, the following socket options can be read:

#### 363 OE\_CONFIGSTREAMPATH\*

Obtain path specifying config data stream.

| option value type  | char *                                                          |
|--------------------|-----------------------------------------------------------------|
| option description | A character string specifying the configuration stream path     |
| default value      | /dev/xillybus_oe_config_32, \\.\xillybus_oe_config_32 (Windows) |

#### 365 OE\_READSTREAMPATH\*

Obtain path specifying input data stream.

| option value type  | char *                                                                                |
|--------------------|---------------------------------------------------------------------------------------|
| option description | A character string specifying the input stream path                                   |
| default value      | $/dev/xillybus\_oe\_input\_32 \ \backslash . \ \ villybus\_oe\_input\_32 \ (Windows)$ |

### 367 OE\_WRITESTREAMPATH\*

Obtain path specifying input data stream.

| option value type  | char *                                                                                   |
|--------------------|------------------------------------------------------------------------------------------|
| option description | A character string specifying the output stream path                                     |
| default value      | $/dev/xillybus\_oe\_output\_32, \ \backslash . \ \ villybus\_oe\_output\_32 \ (Windows)$ |

### 369 OE\_SIGNALSTREAMPATH\*

### 370 Obtain path specifying hardware signal data stream

| option value type  | char *                                                                                   |
|--------------------|------------------------------------------------------------------------------------------|
| option description | A character string specifying the signal stream path                                     |
| default value      | $/dev/xillybus\_oe\_signal\_8, \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\$ |

#### 371 OE\_DEVICEMAP

The device map.

| option value type  | oe_device_t *                                           |
|--------------------|---------------------------------------------------------|
| option description | Pointer to a pre-allocated array of oe_device_t structs |
| default value      | N/A                                                     |

### 373 OE\_NUMDEVICES

The number of devices in the device map.

| option value type  | oe_reg_val_t                                    |
|--------------------|-------------------------------------------------|
| option description | The number of devices supported by the firmware |
| default value      | N/A                                             |

### 375 OE\_MAXREADFRAMESIZE

The maximal size of a frame produced by a call to oe\_read\_frame in bytes. This number is the size of the frame produced by every device within the device map that generates read data.

| option value type  | oe_reg_val_t                     |
|--------------------|----------------------------------|
| option description | Maximal read frame size in bytes |
| default value      | N/A                              |

### 378 OE\_WRITEFRAMESIZE

The maximal size of a frame accepted by a call to oe\_write\_frame in bytes. This number is the size of the frame provided to oe\_write\_frame to update all output devices synchronously.

| option value type  | oe_reg_val_t                      |
|--------------------|-----------------------------------|
| option description | Maximal write frame size in bytes |
| default value      | N/A                               |

### 381 OE\_RUNNING

Hardware acquisition run state. Any value greater than 0 indicates that acquisition is running.

| option value type  | oe_reg_val_t                                    |
|--------------------|-------------------------------------------------|
| option description | Any value greater than 0 will start acquisition |
| default value      | False                                           |

### 383 OE\_SYSCLKHZ

System clock frequency in Hz. The PCIe bus is operated at this rate. Read-frame clock values are incremented at this rate.

| option value type  | oe_reg_val_t                 |
|--------------------|------------------------------|
| option description | System clock frequency in Hz |
| default value      | N/A                          |

#### 386 OE\_ACQCLKHZ

Acquisition clock frequency in Hz. Reads from devices are synchronized to this clock. Clock values within frame data are incremented at this rate.

| option value type  | oe_reg_val_t                      |
|--------------------|-----------------------------------|
| option description | Acquisition clock frequency in Hz |
| default value      | 42000000                          |

### 389 OE\_BLOCKREADSIZE

Number of bytes read during each read() syscall to the data read stream. This option allows control over a fundamental trade-off between closed-loop response time and overall performance. The minimum (default) value will provide the lowest response latency. Larger values will reduce syscall frequency and may improve processing performance for high-bandwidth data sources.

| option value type  | size_t                             |
|--------------------|------------------------------------|
| option description | Size, in bytes, of read() syscalls |
| default value      | value of OE_MAXREADFRAMESIZE       |

### oe\_set\_opt

Set context options.

```
int oe_set_opt(oe_ctx ctx, int option, const void* value, size_t size);
```

#### 396 Arguments

398

400

- ctx context
  - option option to set
- value value to set option to
  - size length of value in bytes

#### 401 Returns int

- 0: success
- Less than 0: oe\_error\_t

#### 404 Description

- The oe\_set\_opt function sets the option specified by the option argument to the value pointed to by the value argument within ctx. The size indicates the size of the value in bytes.
- The following context options can be set:

- 408 OE\_CONFIGSTREAMPATH\*
- $_{\rm 409}$  Set path specifying configuration data stream.

| option value type  | char *                                                          |
|--------------------|-----------------------------------------------------------------|
| option description | A character string specifying the configuration stream path     |
| default value      | /dev/xillybus_oe_config_32, \\.\xillybus_oe_config_32 (Windows) |

#### 410 OE\_READSTREAMPATH\*

Set path specifying input data stream.

| option value type  | char *                                                                                   |
|--------------------|------------------------------------------------------------------------------------------|
| option description | A character string specifying the input stream path                                      |
| default value      | $/dev/xillybus\_oe\_input\_32, \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\$ |

#### 412 OE WRITESTREAMPATH\*

Set path specifying input data stream.

| option value type  | char *                                                          |
|--------------------|-----------------------------------------------------------------|
| option description | A character string specifying the output stream path            |
| default value      | /dev/xillybus_oe_output_32, \\.\xillybus_oe_output_32 (Windows) |

### 414 OE\_SIGNALSTREAMPATH\*

Set path specifying hardware signal data stream

| option value type  | char *                                                                                   |
|--------------------|------------------------------------------------------------------------------------------|
| option description | A character string specifying the signal stream path                                     |
| default value      | $/dev/xillybus\_oe\_signal\_8, \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\ \\$ |

### 416 OE\_RUNNING\*\*

Set/clear data input gate. Any value greater than 0 will start acquisition. Writing 0 to this option will stop acquisition, but will not reset context options or the sample counter. All data not shifted out of hardware will be cleared. To obtain the very first samples produced by high-bandwidth devices, set OE\_RUNNING before a call to OE\_RESET.

| option value type  | oe_reg_val_t                                    |
|--------------------|-------------------------------------------------|
| option description | Any value greater than 0 will start acquisition |
| default value      | 0                                               |

### 421 OE\_RESET\*\*

Trigger global hardware reset. Any value great than 0 will trigger a hardware reset. In this case, acquisition is stopped and all global hardware state (e.g. sample counters, etc) is defaulted.

| option value type  | oe_reg_val_t                                  |
|--------------------|-----------------------------------------------|
| option description | Any value greater than 0 will trigger a reset |
| default value      | Untriggered                                   |

#### 424 OE BLOCKREADSIZE\*\*\*

Number of bytes read during each read() syscall to the data read stream. This option allows control over a fundamental trade-off between closed-loop response time and overall performance. The minimum (default) value

will provide the lowest response latency. Larger values will reduce syscall frequency and may improve processing performance for high-bandwidth data sources.

| option value type  | size_t                             |  |
|--------------------|------------------------------------|--|
| option description | Size, in bytes, of read() syscalls |  |
| default value      | value of OE_MAXREADFRAMESIZE       |  |

- \* Invalid following a successful call to oe\_init\_ctx. Before this, will return with error code OE\_EINVALSTATE.
- \*\* Invalid until a successful call to oe\_init\_ctx. After this, will return with error code OE\_EINVALSTATE.
- \*\*\* Invalid until a successful call to oe\_init\_ctx and before acquisition is started by setting the OE\_RUNNING context option. In other states, will return with error code OE\_EINVALSTATE.

### 433 oe\_read\_reg

Read a configuration register on a specific device.

```
int oe_read_reg(const oe_ctx ctx, size_t dev_idx, oe_reg_addr_t addr, oe_reg_val_t *value);
```

#### 435 Arguments

437

438

442

- ctx context
  - dev\_idx physical index number
  - addr The address of register to write to
- value pointer to an int that will store the value of the register at addr on dev\_idx

### 440 Returns int

- 0: success
  - Less than 0: oe\_error\_t

### 443 Description

oe\_read\_reg is used to read the value of configuration registers from devices within the current device map. This
can be used to verify the success of calls to oe\_read\_reg or to obtain state information about devices managed by
the current context.

#### 447 oe\_write\_reg

448 Set a configuration register on a specific device.

```
int oe_write_reg(const oe_ctx ctx, size_t dev_idx, oe_reg_addr_t addr, oe_reg_val_t value);
```

#### 449 Arguments

451

452

- ctx context
- dev\_idx the device index to read from
- addr register address within the device specified by dev\_idx to write to
- value value with which to set the register at addr on the device specified by dev\_idx

```
• 0: success
455
       • Less than 0: oe_error_t
456
   Description
457
    oe_write_reg is used to write the value of configuration registers from devices within the current device map. This
458
    can be used to set configuration registers for devices managed by the current context. For example, this is used to
459
   perform configuration of ADCs that exist in a device map. Note that successful return from this function does not
460
   guarantee that the register has been properly set. Confirmation of the register value can be made using a call to
461
   oe_read_reg.
    oe read frame
   Read high-bandwidth data from the read channel.
    int oe_read_frame(const oe_ctx ctx, oe_frame_t **frame)
    Arguments

    ctx context

         frame Pointer to a oe_frame_t pointer
467
    Returns int
468
       • 0: success
469
       • Less than 0: oe_error_t
470
   Description
471
   oe_read_frame allocates host memory and populates it with an oe_frame_t struct corresponding to a single frame,
472
   with a read header, from the data input channel. This call will block until either enough data to construct a frame
473
   is available on the data input stream or oe_destroy_ctx is called. It is the user's repsonisbility to free the resources
474
   allocated by this call by passing the resulting frame pointer to oe_destroy_frame.
    oe_destroy_frame
   Free heap-allocated frame.
    void oe_destroy_frame(oe_frame_t *frame);
    Arguments
       • frame pointer to an oe_frame_t
479
    Returns void
   There is no return value.
```

Returns int

### 482 Description

oe\_destroy\_frame frees a heap-allocated frame. It is generally used to clean up the resources allocated by oe\_read\_frame.

### 485 oe\_write

Write data to the data write channel.

```
int oe_write_frame(const oe_ctx ctx, size_t dev_idx, void *data, size_t data_sz)
```

#### 487 Arguments

490

- ctx context
  - dev idx device to write to
  - data pointer to data to write
  - data\_sz number of bytes to write

#### 492 Returns int

- 0: success
- Less than 0: oe\_error\_t

#### 495 Description

oe\_write\_frame writes block data to a particular device from the device map using the asynchronous data write
channel. If dev\_idx is not a writable device, or if data\_sz does not equal to write\_size the of the device, this
function will return OE\_EDEVIDX and OE\_EWRITESIZE, respectively.

### oe\_version

500 Report the oepcie library version.

```
void oe_version(int major, int minor, int patch)
```

#### 501 Arguments

502

504

- major major library version
- minor minor library version
  - patch patch number

#### 505 Returns void

There is no return value.

### Description Description

This library uses semantic versioning. Briefly, the major revision is for incompatible API changes. Minor version is for backwards compatible changes. The patch number is for backwards-compatible bug fixes.

```
oe\_error\_st
510
   Convert an error number into a human readable string.
    const char *oe_error_str(int err)
    arguments
512
       • err error code
   returns const char *
   Pointer to an error message string
515
    oe\_device\_str
    Convert a device ID into human readable string. Note: This is an extension function available in oedevices.h.
517
    const char *oe_device(ind dev_id)
    Arguments
518
      • dev_id device id
519
   Returns const char *
520
   Pointer to a device id string
```

522 Example Headstage Serialization Protocol

523 TODO