# cronologic Ndigo6G-12 User Guide # **Contents** | In | Introduction 4 | | | | | |----|---------------------------------------------------------------|------------|--|--|--| | | Features | 5 | | | | | | Board overview | 5 | | | | | | | | | | | | 1 | Hardware | 7 | | | | | | 1.1 Installation | 7 | | | | | | 1.2 Cooling | 7 | | | | | | 1.3 External Inputs and Connectors | 8 | | | | | | · | | | | | | | 1.3.1 Front bracket inputs | 8 | | | | | | 1.3.2 Clock connections | 8 | | | | | | 1.3.3 Analog Inputs | 9 | | | | | | Analog Offsets | 9 | | | | | | AC-Coupling and Baseline Drift | 10 | | | | | | 1.3.4 Digital TDC Inputs | 10 | | | | | | 1.3.5 Digital Control Inputs | 11 | | | | | | Use Control Inputs as TDCs | | | | | | | 25c control impacts as 12cs | | | | | | 2 | Functionality 1 | 12 | | | | | | 2.1 ADC Modes | 12 | | | | | | 2.1.1 1-Channel Modes A and D | | | | | | | 2.1.2 2-Channel Mode AD | | | | | | | | | | | | | | 2.1.3 4-Channel Mode ABCD | | | | | | | 2.1.4 Multiple Sampling Modes | | | | | | | Modes AA and DD | 14 | | | | | | Mode AADD | 14 | | | | | | Modes AAAA, DDDD | 15 | | | | | | 2.2 Zero Suppression | 15 | | | | | | 2.3 Trigger Setup | 16 | | | | | | | 16 | | | | | | Analog Inputs | 16 | | | | | | Digital Inputs | | | | | | | 2.3.2 Trigger inputs | | | | | | | | | | | | | | 2.3.3 Gating trigger events | | | | | | | 2.4 Gating Blocks | | | | | | | 2.4.1 Examples | 22 | | | | | | Example 1: Suppression of Noise After Starting an Acquisition | 22 | | | | | | 2.4.2 Example 2: Delayed Trigger | 23 | | | | | | 2.5 Auto Triggering Function Generator | 23 | | | | | | 2.6 Averaging Mode | 24 | | | | | | 2.7 Timing Generator (TiGer) | 24 | | | | | | 2.8 Performing a firmware update | | | | | | | 2.8.1 Procedure | | | | | | | | | | | | | | 2.9 Calibrating the TDC | | | | | | | 2.9.1 Re-calibrating the Ndigo6G-12 2 | 26 | | | | | 7 | Driver Programming API | 27 | | | | | 9 | 3.1 Constants | | | | | | | | | | | | | | 3.1.1 General | <i>∠ /</i> | | | | | | | 3.1.2 Trigger and Gating Block Sources | 28 | | | | |---|------------------|----------------------------------------|----|--|--|--| | | | 3.1.3 Function return values | 29 | | | | | | | 3.1.4 PCle Information | 30 | | | | | | 3.2 | Initialization | | | | | | | | ndigo6g12_get_default_init_parameters | 32 | | | | | | | ndigo6g12_init | | | | | | | | ndigo6g12_close | | | | | | | | ndigo6g12_device | | | | | | | | ndigo6g12_init_parameters | | | | | | | 3.3 | Status information | | | | | | | 0.0 | ndigo6g12_get_driver_revision | | | | | | | | ndigo6g12_get_driver_revision_str | | | | | | | | ndigo6g12_count_devices | | | | | | | | ndigo6g12_get_static_info | | | | | | | | ndigo6g12_get_param_info | | | | | | | | | | | | | | | | ndigo6g12_get_fast_info | | | | | | | | ndigo6g12_get_pcie_info | | | | | | | | ndigo6g12_param_info | | | | | | | | ndigo6g12_static_info | | | | | | | | ndigo6g12_fast_info | | | | | | | | ndigo6g12_pcie_info | | | | | | | 3.4 | Configuration | | | | | | | | ndigo6g12_get_default_configuration | | | | | | | | ndigo6g12_configure | | | | | | | | ndigo6g12_configuration | | | | | | | | ndigo6g12_trigger | | | | | | | | ndigo6g12_trigger_block | 54 | | | | | | | ndigo6g12_gating_block | 56 | | | | | | | ndigo6g12_tdc_configuration | 57 | | | | | | | ndigo6g12_averager_configuration | 58 | | | | | | | ndigo6g12_tdc_channel | 59 | | | | | | | ndigo6g12_tdc_gating_block | 59 | | | | | | | ndigo6g12_tdc_tiger_block | 60 | | | | | | 3.5 | Runtime control | 62 | | | | | | | ndigo6g12_start_capture | | | | | | | | ndigo6g12_stop_capture | | | | | | | | ndigo6g12_manual_trigger | | | | | | | | ndigo6g12_single_shot | | | | | | | | ndigo6g12_clear_pcie_errors | | | | | | | 3.6 | Readout | | | | | | | 0.0 | ndigo6g12_read | | | | | | | | ndigo6g12_get_last_error_message | | | | | | | | ndigo6g12_device_state_to_str | | | | | | | | ndigo6g12_read_in | | | | | | | | | | | | | | | | ndigo6g12_read_out | 04 | | | | | 4 | Packet Format 66 | | | | | | | | 4.1 | Output Structure crono_packet | 66 | | | | | | | Utility macros | | | | | | | | Data encoding for ADC hits | | | | | | | | 4.3.1 NDIGO6G12_OUTPUT_MODE_SIGNED16 | | | | | | | | 4.3.2 NDIGO6G12_OUTPUT_MODE_RAW | | | | | | | | NOIZ NOIGOUTE_COTTOT_NODE_TWIW | | | | | # Introduction The Ndigo6G-12 offers 6400 Msps sample rate, 12 bit resolution and a greatly improved readout rate of up to 5200 MB/s. The unit is a combined ADC/TDC board for the acquisition of pulses in time-of-flight applications. It builds on the established platform of the Ndigo5G-10 but takes it to the next level both in performance and flexibility. The Ndigo6G-12 was specifically designed for time-of-flight applications like LIDAR or TOF mass spectrometry. A measurement precision of 5 ps (RMS) is achievable for unipolar pulses. In addition, information on the pulse shape, such as area or amplitude, is recorded. Four channels with 1600 Msps at 12 bit resolution can be acquired independently. Alternatively, the four channels can be combined into two channels or into a single channel. This way, either a higher temporal resolution up to 6400 Msps or a larger dynamic range can be achieved via multiple-sampling modes. This User Guide documents the hardware and functionality of the Ndigo6G-12 board, as well as the driver programming API provided by the Ndigo6G-12 driver. This User Guide is also available online at docs.cronologic.de/ndigo6g. #### **Features** - · 12 bit dynamic range - · Up to **6400 Msps** sample rate (in 1-channel mode) for increased resolution in time domain. - · Up to four ADC channels for your individual measurement setups. - Four TDC channels with a resolution of 13 ps. - · Two digital control inputs for effective gating and triggering. - · PCIe3 x8 interface for simple and fast data transfer to most PCs. - · Unlimited multihit capabilities. - Common start and common stop capabilities. - Continuous ADC readout rate of approx. 5200 MB/s. - · Zero suppression, significantly reducing PCIe load. - Internal 10 MHz clock with a time base of 10 ppb or the ability to use an external 10 MHz clock. # **Board overview** | Optimized for | TOF applications | |------------------------------------|------------------------------------------------------------------------------------------| | ADC channels | 4 | | TDC channels | 4 | | Digital control channels | 2 | | Connectors | 10 × LEMO 00 | | Sample rate | 6400 Msps (1-Channel Mode)<br>3200 Msps (2-Channel Mode)<br>1600 Msps (4-Channel Mode) | | Resolution | 12 bit | | Maximum bandwidth | TBD | | TDC bin size | 13 ps | | TDC double pulse resolution | typically 5 ns | | Multihit | unlimited | | Dead time between groups | none | | Readout rate | 5200 MByte/s (ADC)<br>30 MHits/s (all TDC channels)<br>11.6 MHits/s (single TDC channel) | | Timestamp range | 106 d | | Readout interface | PCle3 x8 | | Time base | 10 ppb (internal) or external 10 MHz clock | | On-board calibration data storage | yes | | Adjustable trigger windows | yes | | Possibility for overlapping events | yes | | Easy-to-use Windows C-API | yes | | In-system firmware updates | yes | # **Hardware** #### 1.1 Installation The Ndigo6G-12 board can be installed in any PCle x8 (or higher amount of lanes) PCle slot. If the slot electrically supports less than eight lanes, the board will operate at lower data throughput rates. Connect a 6-pin PCIe power cable to the connector at the rear of the board (see Figure 1.1). # 1 Note The Ndigo6G-12 does not operate without a 6-pin PCle power connector. Figure 1.1: Overview of an Ndigo6G-12 board. Note the PCIe power connector at the rear of the board. #### 1.2 Cooling The Ndigo6G-12 board is equipped with an active cooling system, ensuring proper cooling of the device. If, however, the temperature of the ADC chip exceeds 90 °C (for instance, if the device is operated in inappropriate environmental conditions, see Section 6.3.1), a warning is issued to the device driver. When the temperature exceeds 95 °C, the ADC chip is disabled to avoid damaging the device. #### 1.3 **External Inputs and Connectors** #### 1.3.1 Front bracket inputs The inputs of the Ndigo6G-12 board are located on the slot bracket. Figure 1.2 shows the location of the four analog inputs A to D (see Section 1.3.3), the four digital TDC inputs 0 to 3 (see Section 1.3.4), and the two digital control inputs TRG and GATE (see Section 1.3.5). Figure 1.2: Input connectors of an Ndigo6G-12 board located on the PCI bracket. #### 1.3.2 Clock connections Connectors to connect an external clock or to access the internal clock signal are located at the top of the board (see Figure 1.1). #### Clk in (SMA) Connect your external 10 MHz clock signal here. Make sure to set ndigo6g12\_init\_parameters::clock\_source to NDIGO6G12\_CLOCK\_SOURCE\_SMA. #### Clk out (SMA) 10 MHz output. This is either the internal clock signal, or an external clock 10 MHz clock if one is used. ## Clk diff (LEMO00) Same as Clk out, but as a differential signal and with a LEMO00 connector. ## 1.3.3 Analog Inputs Figure 1.3: Input circuit for each of the four analog channels. The analog inputs of the ADC are single ended LEMO00 coax connectors. The inputs have a 50 $\Omega$ impedance and are AC coupled. The inputs are converted to a differential signal using a balun. #### **Analog Offsets** AC coupling removes the DC voltage offset from the input signal. However, users can shift the DC base-line voltage before sampling to a value of their choice (using the <u>analog offset</u> parameter). This feature is useful for highly asymmetric signals, such as pulses from TOF spectrometers or LIDAR systems. Without analog offset compensation, the pulses would begin in the middle of the ADC range, effectively cutting the dynamic range in half (see Figure 1.5). By shifting the DC baseline to one end of the ADC range, the input range can be used fully, providing the maximum dynamic range. The analog offset can be set between $\pm$ 0.5 V. Figure 1.4: Users can add an analog offset to the input before sampling. **Figure 1.5:** Asymmetric signal shifted to increase dynamic range. #### **AC-Coupling and Baseline Drift** Due to the AC-coupling of the analog and digital inputs, a baseline drift may occur over time (see Figure 1.6). To avoid this problem, make sure your input signal fulfills the requirements laid out in Section 6.3.4. **Figure 1.6:** Baseline drift due to AC-coupling. A second pulse close to a first may be influenced by a shifted baseline, as sketched in the lower graph. #### 1.3.4 Digital TDC Inputs The Ndigo6G-12 board includes four TDC channels with 13 ps timing resolution. The inputs are AC coupled (see Figure 1.7). **Figure 1.7:** Principal input circuit for each of the digital TDC and control inputs. The following members of the $ndigo6g12\_configuration$ struct configure, respectively, TDC channels 0 to 3: ## tdc\_trigger\_offsets[0:3] Configure the DC offset. #### trigger[NDIGO6G12\_TRIGGER\_TDC0:NDIGO6G12\_TRIGGER\_TDC3] Configure if an edge or level trigger is used (relevant, if the TDCs are used in $trigger\_blocks$ or $gating\_blocks$ ) and if the rising or falling edge of the input signal triggers. ## tdc\_configuration.channel[0:3] Configure if (channel [0:3] . enable) and when (channel [0:3] . gating\_block) timestamps are recorded on the TDC channel. The trigger unit input logic is summarized, as well, in Figure 2.14. # 1.3.5 Digital Control Inputs There are two digital control inputs on the front slot cover called TRG and GATE. Input-signals on the inputs TRG and GATE are digitized and routed to the Trigger Matrix. They can be used to trigger any of the trigger state machines and gating blocks with maximum sampling rate. The digital control inputs are optimally suited to be used as digital triggers and gates, and we recommend using them instead of the digital TDC inputs for these purposes. TRG and GATE are configured analogously to the TDC inputs (see Section 1.3.4 and Figure 2.14), where indices 4 (5) and NDIGO6G12 TRIGGER TRG (NDIGO6G12 TRIGGER GATE) correspond to input TRG (GATE). The input circuit and trigger logic is identical to the TDC inputs (see Figures 1.7 and 2.14). #### **Use Control Inputs as TDCs** The control inputs TRG and GATE can be used as low-resolution TDCs. The dead-time is 5 ns. Pulses should have a width of at least 300 ps to reliably be detected. To record timestamps with the TRG or GATE input, set config.tdc\_configuration. channel [4/15].enable to true. #### Note The digital control inputs TRG and GATE are best suited for triggering and controlling gates. The digital TDC inputs are best suited for measuring precise time stamps. # **Functionality** #### 2.1 ADC Modes The ADC quantizes the input signal using 12 bits. By default, these are mapped to signed 16 bit (for more details, see Section 4.3). Data processing such as trigger detection or packet building are always performed at 5 ns intervals. Depending on the ADC mode, this interval may contain 32 (1-Channel Mode @ 6.4 Gsps), 16 (2-Channel Mode @ 3.2 Gsps) or 8 (4-Channel Mode @ 1.6 Gsps) samples. The ADC mode is configured using ndigo6g12 configuration::adc mode. The board supports using one, two or four channels. This is configured when the board is initialized, see ndigo6g12\_init\_parameters::application\_type. During interleaving, the Ndigo6G-12 firmware reorders and groups the data into a linear sample stream. The process is fully transparent. For users, the only difference is that a 5 ns cycle can contain 8, 16 or 32 samples, depending on the mode. Depending on the application type, the minimal length of the output packets changes. The minimal lengths are: - $\cdot$ 3 $\times$ 32 Samples (15 ns) @ 6.4 Gsps (1-Channel Mode) - $\cdot$ 3 × 16 Samples (15 ns) @ 3.2 Gsps (2-Channel Mode) - $\cdot$ 4 × 8 Samples (20 ns) @ 1.6 Gsps (4-Channel Mode) #### 2.1.1 1-Channel Modes A and D In these modes, only a single channel is used. The analog signal on that channel is digitized at 6.4 Gsps. Packet size is always a multiple of 32 samples per 5 ns (See Figures 2.1 and 2.12). For this mode, ndigo6g12 static info::application type needs to be either NDIGO6G12 APP TYPE 1CH or NDIGO6G12 APP TYPE AVRG. Figure 2.1: ADCs in 1-channel-mode A, B, C or D interleaved for 6.4 Gsps. #### 2.1.2 2-Channel Mode AD In this mode, two channels are used simultaneously. The analog signals on these channels are digitized at 3.2 Gsps each. Packet size is always a multiple of 16 samples per 5 ns (See Figures 2.2 and 2.11). For this mode, $ndigo6g12\_static\_info::application\_type$ needs to be $NDIGO6G12\_APP\_TYPE\_2CH$ . Figure 2.2: ADCs in 2-channel-mode AD, interleaved for 3.2 Gsps. #### 2.1.3 4-Channel Mode ABCD In this mode, all four channels are digitized independently at 1.6 Gsps each. The packet size is always a multiple of 16 samples per 10 ns. (See Figures 2.3 and 2.10). For this mode, $ndigo6g12\_static\_info::application\_type$ needs to be $NDIGO6G12\_APP\_TYPE\_4CH$ . Figure 2.3: ADCs in 4-channel-mode ABCD at 1.6 Gsps. #### 2.1.4 Multiple Sampling Modes In these modes, only the specified input channels are used, but the channels are sampled independently by the ADC cores. The output of the board depends on $ndigo6g12\_configuration::sample\_averaging$ . • sample\_averaging == false: The digitized samples are output as separate packets (the number of which depends on the selected mode). • sample\_averaging == true: The average of the digitized samples is calculated and output as one single packet. Using the same trigger settings on all ADCs can be used to reduce noise by averaging the four channels. To deal with complex triggering conditions, different trigger settings on each of the ADCs can be used. The Ndigo6G-12 provides four ADCs sampling at 1.6 Gsps each. Higher speed modes are implemented by interleaving two or four of these ADCs. #### **Modes AA and DD** In this mode, input channel A (or D) is sampled at 3.2 Gsps two times and independently by the internal ADC cores, see Figure 2.4. For this mode, $ndigo6g12\_static\_info::application\_type$ needs to be NDIGO6G12\_APP\_TYPE\_2CH. Figure 2.4: ADCs in 2-channel-mode AA or DD at 3.2 Gsps. #### **Mode AADD** In this mode, input channel A and D are sampled at 1.6 Gsps two times and independently by the internal ADC cores, see Figure 2.5. For this mode, $ndigo6g12\_static\_info::application\_type$ needs to be NDIGO6G12 APP TYPE 4CH. Figure 2.5: ADCs in 4-channel-mode AADD at 1.6 Gsps. #### Modes AAAA, DDDD In this mode, input channel A (or D) are sampled at 1.6 Gsps four times and independently by the internal ADC cores, see Figure 2.6. For this mode, $ndigo6g12\_static\_info::application\_type$ needs to be NDIGO6G12 APP TYPE 4CH. Figure 2.6: ADCs in 4-channel-mode AAAA or DDDD at 1.6 Gsps. # 2.2 Zero Suppression One of the Ndigo6G-12's key features is on-board zero suppression to reduce PCIe bus load. Only data that passes specifications predefined by the user is transmitted. Data is transmitted as so-called "packets." For the ADC channels, the packet contains the waveform data and a timestamp giving the absolute time (i.e., the time since the start of the data acquisition) of the packet's first sample. Figure 2.7 shows a simple example: Data is only written to the PC if the sample values exceed a specific threshold. Expanding on that, the Ndigo6G-12's zero suppression can be used to realize much more complex scenarios using the Trigger and Gating Blocks (see Sections 2.3 and 2.4). Figure 2.7: Simple zero suppression: Only data with values above a threshold are written to the PC. #### **Trigger Setup** 2.3 The Ndigo6G-12 records analog waveforms using zero suppression. Whenever a relevant waveform is detected, data is written to an internal FIFO memory. Each ADC channel has two trigger units. These can be configured independently (e.g., one unit could trigger on rising edges, the other on falling). They are configured with config. trigger. Each ADC channel has a corresponding trigger block that determines whether data is written to the internal FIFOs. The trigger blocks are configured with config. trigger\_block. Each trigger block can take any amount of trigger units as a source (for details, see ndigo6g12\_trigger\_block::sources or Section 2.3.2), thus, enabling sophisticated trigger setups. #### 2.3.1 Trigger configuration #### **Analog Inputs** Users can specify a *threshold* and can choose whether triggering is used whenever incoming data is below or above the threshold (level triggering, see Figure 2.8) or only if data exceeds the threshold (edge triggering, see Figure 2.9). A gate length can be set to extend the recording window by multiples of 5 ns. Furthermore, a precursor window can be specified, causing the trigger unit to write data to the FIFO (precursor $\times$ 5 ns) before the trigger event. When edge triggering is used, all packets have the same length of (precursor + length + 1)-cycles of 5 ns. For level triggering, packet length is data dependent. If retrigger is enabled and the trigger conditions are fulfilled during the recording of the postcursor, the recording window is extended (see Figure 2.7). Figure 2.8: Example for level triggering. Figure 2.9: Example for edge triggering. Figure 2.10: Triggering in 4-channel mode at 8 samples per clock cycle. Figure 2.11: Triggering in 2-channel mode at 16 samples per clock cycle. Figure 2.12: Triggering in 1-channel mode at 32 samples per clock cycle. #### **Digital Inputs** For all digital inputs, the configuration value ndigo6g12 trigger::threshold is ignored. Their trigger threshold is configured by ndigo6g12\_configuration::tdc\_trigger\_offsets. Equivalently to the analog inputs, edge- or level-trigger functionality can be enabled using ndigo6g12\_trigger::edge. The duration of a level trigger is solely limited by the AC-coupling (see Figure 1.6 for the effects of AC-coupling on a signal). # 2.3.2 Trigger inputs A trigger block can use several input sources: - The eight trigger decision units of all four ADC channels (Figure 2.13) - The four TDC and the two digital control inputs (Figure 2.14) - · A function trigger providing random or periodic triggering (see Auto Triggering Function Generator). Trigger inputs from the above sources can be concatenated using a logical OR by setting the appropriate bits in the bitmask (see <a href="mailto:ndigo6g12\_trigger\_block::sources">ndigo6g12\_trigger\_block::sources</a>). See also Figure 2.15. Figure 2.13: From the ADC inputs, a trigger unit creates an input flag for the trigger matrix. Each digitizer channel (A, B, C, D) has two trigger units. Figure 2.14: The digital inputs TDC0, TDC1, TDC2, TDC3, TRG, and GATE have simpler trigger units. Figure 2.15: Trigger Matrix. The eight trigger signals from the four analog channels and the trigger signals from the six digital channels (four TDC channels, TRG, GATE) can be combined to create a trigger input for each trigger block. Additionally, four gate signals (see Figure 2.16) can be used to suppress trigger during configurable time frames. ## 2.3.3 Gating trigger events Triggers can be fed into the $gating\_blocks$ as outlined in Chapter 2.4 and Figure 2.16. In return, the gating\_blocks can be used to block writing data to the FIFO. That way, only zero-suppressed data occurring when the selected gate is active is transmitted. This procedure reduces PCIe bus load even further. Which gating block is used to block a particular trigger block is configured with ndigo6g12\_trigger\_block::gates. # 2.4 Gating Blocks In order to decrease the amount of data transmitted to the PC, the Ndigo6G-12 includes four independent gate and delay units. They are configured using $ndigo6g12\_configuration::gating\_block$ and (specifically for the TDC channels) ndigo6g12\_tdc\_channel::gating\_block. A gate and delay unit creates a gate window starting and closing at specified times after a trigger event (as configured by the user with $ndigo6g12\_gating\_block::start$ and stop). Concretely, if a trigger event is detected, a timer starts. After the timer reaches the time corresponding to start, the gate will activate. After the timer reaches the time corresponding to stop, it will inactivate. This behavior may be influenced by the retrigger feature. With this feature enabled, another trigger signal will reset the timer to zero. That means, if a second trigger is detected before the gate is activated, Figure 2.16: Gating Blocks: Each gating block can use an arbitrary combination of inputs to trigger its state machine. The outputs can be individually inverted and routed to the AND-gate feeding the trigger blocks. the time until it activates is extended. If, however, the gate was already active, the time until it inactivates will be extended. #### Attention A bug in Firmware Rev. $\leq 1.24120$ causes the retrigger feature to reset the gate logic entirely (i.e, the state of the gate will inactivate after a retrigger event). Depending on $ndigo6g12\_gating\_block::negate$ , an active gate will be open (signal detection enabled) or closed (signal detection disabled). Each gating block can use an arbitrary combination of inputs which trigger it. This is configured using ndigo6g12 gating block::sources. trigger blocks can use the gate signal to suppress data acquisition, that is, only data that fulfills zero suppression specifications occurring in an open gate window is written to the PC. Figure 2.17: Gate and delay functionality: When a trigger occurs, the gate opens after a set period of time "Gate Start" and closes when it reaches "Gate Stop". A second trigger event may influence this behavior if retriggering is enabled. #### 2.4.1 Examples #### **Example 1: Suppression of Noise After Starting an Acquisition** In mass spectrometer and other experiments, noise while starting data acquisition can result in undesired trigger events during start-up time. To prevent noise in the output data, a gating block could be used to suppress all triggers during start-up. The following example illustrates the use of a gating block (in the following, $gating \ block[0]$ ) to prevent recording noise: · Set up the GATE input to trigger on each acquisition start, that is, $trigger[NDIG06G12\_TRIG-$ GER\_GATE] is configured corresponding to the input signal (e.g., configuring the polarity). - NDIG06G12 TRIGGER SOURCE GATE is selected as input source of gating block[0].source and the $gating \ block[0]$ . start parameter is set to 0. - The gating block[0]. stop parameter is set to the desired length (in multiples of 5 ns). - gating\_block[0].negate is set to true. Now, gating\_block[0] will output a LOW pulse of the desired length (that is, the gate is closed during start-up time) whenever there is a pulse on the GATE input. Now, select the above gate for the trigger bock you want to use for triggering data acquisition, e.g., trigger block[0]: Set trigger block[0].sources e.g., ``` config.trigger_block[0].sources = NDIGO6G12_TRIGGER_SOURCE_AO | NDIGO6G12_ TRIGGER SOURCE DO ``` uses the ADC input channels A and D as sources. Set NDIGO6G12\_TRIGGER\_GATE\_0 as trigger\_block[0].gates. ``` config.trigger block[0].gates = NDIGO6G12 TRIGGER GATE 0 ``` Now, recording of data is suppressed for an initial start-up time. #### 2.4.2 Example 2: Delayed Trigger To sample a short window at a specified time after a trigger event on a channel, a gating block can be used to create a delayed trigger. To do this, one of the triggers of the channel of interest is configured to the desired parameters by selecting the threshold, setting the edge polarity and enabling edge triggering. Instead of directly using this trigger as an input to the trigger block's input matrix, the trigger is selected as an input to a gating block. The block is configured with start = delay (in multiples 5 ns) and stop = start+1, negate = false. This causes the gating block to produce a one clock cycle pulse on its output after the specified delay. To send this pulse to the trigger block, the gating block must be enabled in the trigger block's AND matrix and the ONE trigger source must be selected. #### 2.5 **Auto Triggering Function Generator** Some applications require periodic or random triggering. The Ndigo6G-12's function generator provides this functionality. The delay between two trigger pulses of this trigger generator is the sum of two components: A fixed value M and a pseudo-random value given by the exponent N. The period is $$T = M + [1...2^N] - 1$$ clock cycles with a duration of 5 ns per cycle, where $6 \le M < 2^{32}$ and $0 \le M < 32$ . This allows to monitor input signals at times the current trigger configuration does not trigger, e.g., to get baseline information in mass spectrometry applications. It can also be used to determine a suitable threshold level for the trigger by first getting random statistics on the input signal. This functionality is enabled and configured using ndigo6g12 configuration::auto trigger period and auto trigger random exponent. # 2.6 Averaging Mode Instead of streaming each recorded trigger event as packets, it is possible to average over multiple trigger events. By initializing the Ndigo6G-12 board with NDIGO6G12\_APP\_TYPE\_AVRG, Averaging Mode is enabled. Then, a number of ndigo6g12\_averager\_configuration::iterations are averaged before output is written. Averaging Mode can be used only with ADC modes A and D (see Section 2.1). #### Attention Be aware that in averaging mode, the first two 64-bit words in data are an extended header. See Section 4.5 for more information. # **Timing Generator (TiGer)** The LEMO connectors of all TDC channels, the TRG channel, and the GATE channel can be used as an AC-coupled trigger output. The TiGer functionality can be configured independently for each connector. Each TiGer is configured using the ndigo6g12\_tdc\_tiger\_block struct. The tiger blocks can be triggered by any combination of inputs, including the auto-trigger and the ADC channels. #### Note The TiGer configuration is similar to the gating blocks. The TiGer can be used in different output modes. For an overview of the different modes, see the documentation in the API section. With restrictions, the respective LEMO connectors can be used simultaneously as a TiGer output and as an input. # Performing a firmware update The Ndigo6G-12 device driver includes the tool FirmwareGUI\_64.exe. It can be used to perform a firmware update. The tool is located in your device driver installation path under apps x64. The tool is shown in Figure 2.18. Figure 2.18: Firmware flash tool for the Ndigo6G-12. #### 2.8.1 Procedure - 1. If you have multiple Ndigo6G-12 installed, choose a card in the "Card" dropdown menu. - 2. It is advisable to perform a backup of the current firmware and calibration data. Click the respective "Backup" button in the tool. - 3. Optionally select the application type that you wish to use. This step is not required and can also be performed when configuring the Ndigo6G-12 in your user software. - 4. Browse to the new firmware file: In the "Status" section, click on "Browse". The firmware that is delivered with the device driver is located at firmware\Ndigo6G\_Firmware\_YYYYMMDD.cronorom. - 5. Optionally compare the currently installed firmware with the selected firmware by pressing "Verify". - 6. Click "Flash" to perform the firmware update. #### Attention The new firmware will only be used after a *complete* power cycle. A simple reboot may not be sufficient. ### Attention After a firmware update the TDCs have to be re-calibrated. See *Calibrating the TDC* for the procedure. #### 2.9 Calibrating the TDC You can backup or restore the calibration from a previous backup using FirmwareGUI\_64.exe (see Performing a firmware update and Figure 2.18). In the "Calibration" section, click "Backup" and choose a location for the .ndigo6gcal calibration file. You can compare the currently used calibration with the calibration from a backup file by clicking "Verify" button after browsing to your calibration file. You can flash the calibration from a \*.ndigo6gcal file onto the Ndigo6G-12 by clicking "Flash" after browsing to your calibration file. #### Attention After performing a *firmware update*, it is always necessary to re-calibrate your Ndigo6G-12. Restoring a previous calibration from a backup is not sufficient. ## 2.9.1 Re-calibrating the Ndigo6G-12 Calibration is performed with the command-line tool ndigo6g12\_tdc\_alignment.exe. It is located in the installation directory of the Ndigo6G-12 driver under apps\x64 (by default C:\Program Files\ cronologic\Ndigo6G-12\apps\x64). Navigate to the folder and open it in a terminal, then start the tool from that terminal. The tool takes command line arguments. Run .\ndigo6g12 tdc alignment.exe -help for an overview. Calibration is performed by simply starting the tool: .\ndigo6g12 tdc alignment.exe. If multiple Ndigo6G-12 are installed in your system, you can perform calibration for each one of them by calling .\ndigo6g12\_tdc\_alignment.exe -index <device\_index>, where the <device\_index> starts at 0 and increments for each Ndigo6G-12. # **Driver Programming API** #### Attention The API requires *driver versions* >2.0.0 and *firmware* 1.24120. The API is a DLL with C linkage. Declarations of the interface are found in $ndigo6g12\_interface.h$ , provided by the Ndigo6G-12 driver. This chapter provides an overview of the provided API functionality. #### 3.1 Constants #### 3.1.1 General ## NDIGO6G12\_API\_VERSION The current API version. #### NDIGO6G12\_TRIGGER\_COUNT The number of ADC and TDC triggers, including AUTO and ONE. #### NDIGO6G12\_ADC\_CHANNEL\_COUNT The number of analog input channels. ## NDIGO6G12\_GATE\_COUNT The number of gating blocks. #### NDIGO6G12\_TDC\_CHANNEL\_COUNT The number of high (TDC0-3) and low (TRG, GATE) resolution TDC input channels. #### NDIGO6G12\_BITSTREAM\_DATE\_LEN Bitstream date format: YYYY-MM-DD hh:mm:ss #### NDIGO6G12\_CALIBRATION\_DATE\_LEN Calibration date format: YYYY-MM-DD hh:mm #### NDIGO6G12\_FLASH\_SIG\_LEN Length of Ndigo6G-12 flash signature #### NDIGO6G12\_FIFO\_DEPTH ADC sample FIFO depth. It is the maximum recording length in multiples of 5 ns. #### NDIGO6G12\_MAX\_PRECURSOR Maximum for ndigo6g12\_trigger\_block::precursor. #### NDIGO6G12\_MAX\_MULTISHOT Maximum for ndigo6g12\_trigger\_block::multi\_shot\_count. ## 3.1.2 Trigger and Gating Block Sources Bitmasks for trigger sources. Used for ndigo6g12\_trigger\_block::sources, ndigo6g12\_gating\_block::sources, ndigo6g12\_tdc\_gating\_block::sources, and ndigo6g12\_tdc\_tiger\_block::sources. #### **Defines** #### NDIGO6G12\_TRIGGER\_SOURCE\_NONE All trigger sources disabled. NDIGO6G12\_TRIGGER\_SOURCE\_AO NDIGO6G12\_TRIGGER\_SOURCE\_A1 NDIGO6G12\_TRIGGER\_SOURCE\_BO NDIGO6G12\_TRIGGER\_SOURCE\_B1 NDIGO6G12\_TRIGGER\_SOURCE\_CO NDIGO6G12\_TRIGGER\_SOURCE\_C1 NDIGO6G12\_TRIGGER\_SOURCE\_DO NDIGO6G12\_TRIGGER\_SOURCE\_D1 NDIGO6G12\_TRIGGER\_SOURCE\_TDCO NDIGO6G12\_TRIGGER\_SOURCE\_TDC1 NDIGO6G12\_TRIGGER\_SOURCE\_TDC2 NDIGO6G12\_TRIGGER\_SOURCE\_TDC3 NDIGO6G12\_TRIGGER\_SOURCE\_TRG NDIGO6G12\_TRIGGER\_SOURCE\_GATE NDIGO6G12\_TRIGGER\_SOURCE\_AUTO NDIGO6G12\_TRIGGER\_SOURCE\_ONE Trigger signal is active each clock cycle. NDIGO6G12\_TRIGGER\_SOURCE\_FPGAO Deprecated. Alias for NDIGO6G12\_TRIGGER\_SOURCE\_TRG. NDIGO6G12\_TRIGGER\_SOURCE\_FPGA1 Deprecated. Alias for NDIGO6G12\_TRIGGER\_SOURCE\_GATE. #### 3.1.3 Function return values Return codes of various functions. All ERRORS must be positive integers, because the upper byte is used by crono\_tools **Defines** CRONO\_OK CRONO\_WINDRIVER\_NOT\_FOUND CRONO\_DEVICE\_NOT\_FOUND CRONO\_NOT\_INITIALIZED CRONO\_WRONG\_STATE CRONO\_INVALID\_DEVICE CRONO\_BUFFER\_ALLOC\_FAILED CRONO\_TDC\_NO\_EDGE\_FOUND CRONO\_INVALID\_BUFFER\_PARAMETERS CRONO\_INVALID\_CONFIG\_PARAMETERS CRONO\_WINDOW\_CALIBRATION\_FAILED CRONO\_HARDWARE\_FAILURE CRONO\_INVALID\_ADC\_MODE CRONO\_SYNCHRONIZATION\_FAILED CRONO\_DEVICE\_OPEN\_FAILED CRONO\_INTERNAL\_ERROR CRONO\_CALIBRATION\_FAILURE CRONO\_INVALID\_ARGUMENTS CRONO\_INSUFFICIENT\_DATA #### 3.1.4 PCle Information PCIe correctable error flags. Only relevant when troubleshooting. #### **Defines** CRONO\_PCIE\_RX\_ERROR CRONO\_PCIE\_BAD\_TLP CRONO\_PCIE\_BAD\_DLLP CRONO\_PCIE\_REPLAY\_NUM\_ROLLOVER CRONO\_PCIE\_REPLAY\_TIMER\_TIMEOUT CRONO\_PCIE\_ADVISORY\_NON\_FATAL CRONO\_PCIE\_CORRECTED\_INTERNAL\_ERROR CRONO\_PCIE\_HEADER\_LOG\_OVERFLOW PCIe uncorrectable error flags. Only relevant when troubleshooting. #### **Defines** ``` CRONO_PCIE_UNC_UNDEFINED CRONO_PCIE_UNC_DATA_LINK_PROTOCOL_ERROR CRONO_PCIE_UNC_SURPRISE_DOWN_ERROR CRONO_PCIE_UNC_POISONED_TLP CRONO_PCIE_UNC_FLOW_CONTROL_PROTOCOL_ERROR CRONO_PCIE_UNC_COMPLETION_TIMEOUT CRONO_PCIE_UNC_COMPLETER_ABORT CRONO_PCIE_UNC_UNEXPECTED_COMPLETION CRONO_PCIE_UNC_RECEIVER_OVERFLOW_ERROR CRONO_PCIE_UNC_MALFORMED_TLP CRONO_PCIE_UNC_ECRC_ERROR CRONO_PCIE_UNC_UNSUPPORED_REQUEST_ERROR ``` #### 3.2 Initialization To use a Ndigo6G-12 board, it first needs to be initialized. This is done by calling $ndigo6g12\_init()$ . The initialization parameters necessary for ndigo6g12 init() are provided in the ndigo6g12 init parameters struct. The general procedure for initialization is as follows: - 1. Load a default set of initialization parameters using <a href="mailto:ndigo6g12\_get\_default\_init\_parame-">ndigo6g12\_get\_default\_init\_parame-</a> ters. - 2. If necessary, adjust default parameters to your specific needs. - 3. Initialize the Ndigo6G-12 board using ndigo6g12\_init(). - 4. Check that the initialization was successful. If so, the return value of $ndigo6g12\_init()$ is CRONO OK. Information on the current device will be stored as type ndigo6g12 device. ``` ndigo6g12_get_default_init_parameters(init) ``` Macro that calls <code>ndigo6g12\_get\_default\_init\_parameters\_version</code> with the correct API version. int ndigo6g12\_get\_default\_init\_parameters\_version(ndigo6g12\_init\_parameters\*init, int client\_api\_version) Sets up the standard parameters. Gets a set of default parameters for ndigo6g12\_init(). This must always be used to initialize the ndigo6g12\_init\_parameters structure. For convinience, the macro ndigo6g12\_get\_default\_init\_parameters is provided, which automatically sets the correct client api version. #### Default values: - card\_index = 0 - board\_id = 0 - buffer\_size[0] = 64 (MiB) - buffer\_size[1-7] = 0 (unused) - dma\_read\_delay = 1000 - perf\_derating = 0 - led\_flashing\_mode = 1 - clock\_source = NDIGO6G12\_CLOCK\_SOURCE\_INTERNAL - application\_type = NDIGO6G12\_APP\_TYPE\_CURRENT - force\_bitstream\_update = false - partial\_bitstream\_size = 0 - partial\_bitstream = nullptr - firmware\_locations = nullptr #### **Parameters** - init [in] Pointer to a structure in which to store the initialization values. - · client\_api\_version [in] NDIGO6G12\_API\_VERSION #### Returns See Function return values. int ndigo6g12\_init(ndigo6g12\_device \*device, ndigo6g12\_init\_parameters \*params, const char \*\*error\_message) Open and initialize an Ndigo6G-12 board. Which Ndigo6G-12 board will be initialized is determined by ndigo6g12\_init\_parameters::card\_index. #### **Parameters** device – [out] Pointer to the device struct. - · params [in] Pointer to the structure that contains the initialization parame- - error\_message [out] Location in which to store the error message as plain #### **Returns** See Function return values. ## int ndigo6g12\_close(ndigo6g12\_device \*device) Finalize the driver for this device. #### **Parameters** device - [in] Pointer to the device that should be finalized. #### **Returns** See Function return values. #### struct ndigo6g12\_device Contains information of the Ndigo6G-12 device in use. #### **Public Members** bool is\_valid void \*ndigo6g12 #### struct ndigo6g12\_init\_parameters Struct for the initialization of the Ndigo6G-12. This structure MUST be completely initialized. #### **Public Members** #### int version The version number. It is increased when the definition of the structure is changed. The increment can be larger than 1 to match driver version numbers or similar. Set to 0 for all versions up to first release. Must be set to NDIGO6G12\_API\_VERSION. #### int card\_index The index in the list of Ndigo6G-12 boards that should be initialized. There might be multiple boards installed in the system that are handled by this driver as reported by ndigo6g12\_count\_devices(). This index selects one of them. Boards are enumerated depending on the PCIe slot. The lower the bus number and the lower the slot number the lower the card index. #### int board\_id The global index in the list of all cronologic devices. This 8-bit number is filled into each packet created by the board and is useful if data-streams of multiple boards will be merged. If only Ndigo6G-12 boards are used, this number can be set to card\_index. If boards of different types that use a compatible data format are used in a system, each board should get a unique ID. ## int64\_t buffer\_size[8] The minimum size of the DMA buffer. If set to 0, the default size of 64 MiBytes is used. For the Ndigo6G-12 only the first entry is used. #### int dma\_read\_delay The update delay of the writing pointer after a packet has been send over PCIe. Default is 1000. Do not change. #### int perf\_derating Default 0, corresponding to 1.6, 3.2, or 6.4 Gsps (depending on application\_type). For internal use only. Do not change. #### int led\_flashing\_mode Controls the LED flashing mode. Define what LEDs do during initialization: - · 0: LEDs are off - · 1: LEDs light up once #### int clock\_source Defines which clock source is used (internal, SMA, AUX2). Must be one of the following: #### NDIGO6G12\_CLOCK\_SOURCE\_INTERNAL Device is using the internal 10 MHz clock. #### NDIGO6G12\_CLOCK\_SOURCE\_SMA Use an external 10 MHz clock as reference. The input is the SMA socket located on the board. ### NDIGO6G12\_CLOCK\_SOURCE\_AUX2 Use an external 10 MHz clock as reference. The input is the TRG LEMO connector located on the slot bracket. #### uint32\_t application\_type Select the application type. Note that ndigo6g12\_configuration::adc\_mode must match the application type chosen here. Must be one of the following: #### NDIGO6G12\_APP\_TYPE\_AVRG Averaging mode at 6.4 Gsps. For more information, see Section 2.6. #### NDIGO6G12\_APP\_TYPE\_4CH Four ADC channels at 1.6 Gsps. #### NDIGO6G12\_APP\_TYPE\_2CH Two ADC channels at 3.2 Gsps. #### NDIGO6G12\_APP\_TYPE\_1CH One ADC channel at 6.4 Gsps. ## NDIGO6G12\_APP\_TYPE\_CURRENT Use currently installed application type. #### crono\_bool\_t force\_bitstream\_update Force a bitstream update that configures the FPGA. During the initialization of the board, a bitstream configures the FPGA of the Ndigo6G-12. This is only done if during the initialization of the Ndigo6G-12, application\_type is different from the application\_type that the Ndigo6G-12 is currently configured in. That is, the FPGA is only reconfigured, if application type changes. By setting force bitstream update to true, one can force a reconfiguration of the FPGA. #### int partial\_bitstream\_size Size of partial\_bitstream. Reserved for future expandability. ## uint32\_t \*partial\_bitstream Pointer to a buffer with partial bitstream data. Can be nullptr if application\_type matches application\_type of currently installed firmware. Reserved for future expandability. #### const char \*firmware\_locations Location where firmware is installed. Pointer to a list of paths (separated by ;) Can be nullptr if application\_type matches application type of currently installed firmware. #### 3.3 Status information The driver provides functions to retrieve detailed information on the type of board, it's configuration, settings and state. The information is split according to its scope and the computational requirements to query the information from the board. int ndigo6g12\_get\_driver\_revision() Get the driver version in integer format. #### **Returns** The driver version in the same format as ndigo6g12\_static\_info::driver\_revision. const char \*ndigo6g12\_get\_driver\_revision\_str() Get the driver version in string format. #### **Returns** The Driver version including SVN build revision as a string with format x.y.z.svn. int ndigo6g12\_count\_devices(int \*error\_code, const char \*\*error\_message) Get the number of Ndigo6G-12 boards that are installed in the system. #### **Parameters** - error\_code [out] Pointer to an integer in which to store the error code. - · error\_message [out] Location in which to store the error message as plain text. #### **Returns** The number. int ndigo6g12\_get\_static\_info(ndigo6g12\_device \*device, ndigo6g12\_static\_info \*static\_info) Get the static information. The static information does not change after the device initialization. #### **Parameters** - device [in] Pointer to the device from which to get the information. - static\_info [out] Pointer to a structure in which to store the information. #### **Returns** See Function return values. int ndigo6g12\_get\_param\_info(ndigo6g12\_device \*device, ndigo6g12\_param\_info \*param\_info) Get parametric information. The parametric information may change due to the configuration. #### **Parameters** - · device [in] Pointer to the device from which to get the information. - param\_info [out] Pointer to a structure in which to store the information. #### **Returns** See Function return values. int ndigo6g12\_get\_fast\_info(ndigo6g12\_device \*device, ndigo6g12\_fast\_info \*fast\_info) Get fast status information. The information can be retrieved within a few microseconds. #### **Parameters** - device [in] Pointer to the device from which to get the information. - fast\_info [out] Pointer to a structure in which to store the information. #### **Returns** See Function return values. int ndigo6g12\_get\_pcie\_info(ndigo6g12\_device \*device, crono\_pcie\_info \*pcie\_info) Reads the PCIe info like correctable and uncorrectable errors. #### **Parameters** - · device [in] Pointer to the device. - pcie\_info [out] Pointer to the structure in which to store the information. #### **Returns** See Function return values. ### struct ndigo6g12\_param\_info Contains configuration changes. Structure filled by *ndigo6g12\_get\_param\_info()*. This structure contains information that may change indirectly due to configuration changes. #### **Public Members** ### double bandwidth Bandwidth. 4.5 or 6.5 GHz depending on ndigo6g12\_configuration::extended\_bandwidth. #### int resolution ADC sample resolution. Always 12 bit. ### double sample\_rate Actual ADC sample rate of currently sampled data. Depending on *ndigo6g12\_configuration::adc\_mode*, that is, sample\_rate = 6.4 GHz/*chan-nels*. ### double sample\_period The period that one sample in the data represents in picoseconds. ### double tdc\_period The period that one TDC bin in the data represents in picoseconds. ### double packet\_ts\_period The period that one tick of the packet timestamp represents in picoseconds. ### uint64\_t tdc\_packet\_timestamp\_offset The TDC packet timestamp offset. Since TDC packets carry the timestamp of the end of the packet, to calculate the start, tdc packet timestamp offset has to be subtracted. ### uint32\_t tdc\_rollover\_period Time span of one TDC timestamp rollover period in units of the TDC binsize. All TDC hits within this period are written to one crono\_packet. ### double adc\_sample\_delay The delay of the ADC samples relative to TDC timestamps in picoseconds. Note: For driver release 2.2.0 with firmware 1.25086, this value is bugged. ### int board\_id The ID the board uses to identify itself in the output data stream. Takes values 0 to 255. #### int channels Number of ADC channels in the current mode. See ndigo6g12\_configuration::adc\_mode. ### int channel\_mask Mask with a set bit for each enabled input channel. ### int tdc\_channels Number of TDC channels in the current mode. ### int64\_t total\_buffer The total amount of the DMA buffer in bytes. ### int samples\_per\_clock The number of samples in one clock cycle in the current mode. ### struct ndigo6g12\_static\_info Structure contains static information. This structure contains information about the board that does not change during run time. It is provided by ndigo6g12\_get\_static\_info(). #### **Public Members** ### char bitstream\_date[NDIGO6G12\_BITSTREAM\_DATE\_LEN] Bitstream creation date. DIN EN ISO 8601 string YYYY-MM-DD HH:DD:SS describing the time when the bitstream was created. ### int board\_configuration Describes the schematic configuration of the board. The same board schematic can be populated in multiple variants. This is a 8-bit code that can be read from a register. ### int board\_revision Board revision number. The board revision number can be read from a register. It is a four bit number that changes when the schematic of the board is changed. - · 0: Experimental version of the first board. Labeled "Rev. 1". - · 2: First commercial version. Labeled "Rev. 2" ### int board\_serial The board's serial number. With year and running number in 8.24 format (yy.nnn; 8 bits are used to encode the year, 24 bits to encode the number). The number is identical to the one printed on the silvery sticker on the board. ### char calibration\_date[NDIGO6G12\_CALIBRATION\_DATE\_LEN] Calibration date. DIN EN ISO 8601 string YYYY-MM-DD HH:DD describing the time when the card was calibrated. ### int chip\_id 16-bit factory ID of the ADC chip. This is the chipID as read from the 16-bit ADC chip-ID register. ### crono\_bool\_t dc\_coupled Shows if the inputs are DC-coupled. Default is false, that is, AC-coupled. ### int driver\_revision Encoded version number for the driver. The lower three bytes contain a triple-level hierarchy of version numbers. E.g., 0x010103 encodes version 1.1.3. A change in the first digit generally requires a recompilation of user applications. Change in the second digit denote significant improvements or changes that don't break compatibility and the third digit changes with minor bugfixes and the like (see https://semver.org/). ### int driver\_build\_revision The build number of the driver according to cronologic's internal versioning system. ### crono\_bool\_t flash\_valid Calibration data read from flash is valid. If not false, the driver found valid calibration data in the flash on the board and is using it. ### int fw\_revision Revision number of the FPGA configuration. ### int fw\_type Type of firmware, always 5 -> Ndigo6G-12. ### int pcb\_serial Trenz serial number. #### int svn\_revision Subversion revision ID of the FPGA configuration. A number to track builds of the firmware in more detail than the firmware revision. It changes with every change in the firmware, even if there is no visible effect for the user. The subversion revision number can be read from a register. ### int application\_type Shows the initialized mode. See NDIGO6G12\_APP\_TYPE\_\* constants. ### char config\_flash\_signature\_primary[NDIGO6G12\_FLASH\_SIG\_LEN] Shows the signature of the primary flash. ### char config\_flash\_signature\_secondary[NDIGO6G12\_FLASH\_SIG\_LEN] Shows the signature of the secondary flash. ### double auto\_trigger\_ref\_clock Auto trigger clock frequency. The clock frequency of the auto trigger in Hz used for the calculations of ndigo6g12\_configuration::auto\_trigger\_period. Fixed at 200 MHz. ### struct ndigo6g12\_fast\_info Contains fast dynamic information. This structure is filled by *ndigo6g12\_get\_fast\_info()*. This information can be obtained within a few microseconds. #### **Public Members** #### int **state** The current state of the device. Is one of the following: ### NDIGO6G12\_DEVICE\_STATE\_INITIALIZED Device is initialized but not yet configured for data capture. ### NDIGO6G12\_DEVICE\_STATE\_CONFIGURED Device is ready for data capture. ### NDIGO6G12\_DEVICE\_STATE\_CAPTURING Device has started data capture. ### int fan\_speed Speed of the FPGA fan in rounds per minute. Reports 0 if no fan is present. ### double fpga\_temperature Temperature of the FPGA in °C. ### double fpga\_vccint Internal Voltage of the FPGA in V. Useful debugging information. ### double fpga\_vccaux Auxillary Voltage of the FPGA in V. Useful debugging information. ### double fpga\_vccbram BRAM Voltage of the FPGA in V. Useful debugging information. ### double mgt\_0v9 Shows measured voltage for the mgt\_0v9 power supply in V. Useful debugging information. ### double mgt\_1v2 Shows measured Voltage for the mgt\_1v2 power supply in V. Useful debugging information. ### double adc\_2v5 Shows measured voltage for the 2v5 power supply in V. Useful debugging information. ### double clk\_3v3 Shows measured voltage for the clk\_3v3 power supply in V. Useful debugging information. ### double adc 3v3 Shows measured voltage for the adc\_3v3 power supply in V. Useful debugging information. ### double pcie\_3v3 Shows measured voltage for the pcie\_3v3 power supply in V. Useful debugging information. ### double opamp\_5v2 Shows measured voltage for the opamp\_5v2 power supply in V. Useful debugging information. ### double temp4633\_1 Shows temperature of voltage regulartor U3\_1 in °C. ### double temp4633\_2 Shows temperature of voltage regulator U3\_2 in °C. ### double temp4644 Shows temperature of voltage regulator U4 in °C. ### double tdc1\_temp Temperature of the TDC-chip in °C. ### double ev12\_cmiref Shows voltage for differential ADC input common mode voltage in V. Measured or calibration target depending on board revision and assembly variant. ### double ev12\_temp Temperature of the ADC in °C. #### int alerts Alert bits from temperature sensor and the system monitor. Bit 0 is set if the TDC temperature exceeds 140°C. In this case the TDC shut down and the device needs to be reinitialized. Is one of the following: ### NDIGO6G12\_ALERT\_FPGA\_TEMPERATURE FPGA temperature alert (> 70°C) ### NDIGO6G12\_ALERT\_VCCINT Internal FPGA voltage out of range (< 0.83 V or > 0.88 V). ### NDIGO6G12\_ALERT\_VCCAUX FPGA auxiliary voltage out of range (< 1.75 V or > 1.89 V). ### NDIGO6G12 ALERT FPGA TEMPERATURE CRITICAL FPGA temperature critical (> 80°C) ### NDIGO6G12\_ALERT\_THS\_TEMPERATURE\_CRITICAL THS temperature critical (> 140°C) ### int pcie\_link\_width Number of PCIe lanes the card uses. Should always be 8 for the Ndigo6G-12. ### int pcie\_link\_speed Data rate of the PCIe card. Should always be 3 for the Ndigo6G-12. ### int pcie\_max\_payload Maximum size for a single PCIe transaction in bytes. Depends on the system configuration. ### crono\_bool\_t adc\_data\_pll\_locked ADC data clock is PLL locked. ### crono\_bool\_t adc\_data\_pll\_lost\_lock ADC data clock PLL lost lock (Sticky Bit). ### int adc\_lanes\_synced Shows the synced ADC lanes. Each bit corresponds to one lane. Useful debugging information. ### int adc\_lanes\_lost\_sync Shows the ADC lanes that lost sync. Each bit corresponds to one lane. Useful debugging information. ### int adc\_lanes\_fifo\_empty Shows which ADC lanes have an empty FIFO. Each bit corresponds to one lane. Useful debugging information. ### int adc\_lanes\_fifo\_full Shows which ADC lanes have a full FIFO. Each bit corresponds to one lane. Useful debugging information. ### int adc\_lanes\_running Shows which ADC lanes are running. Each bit corresponds to one lane. Useful debugging information. ### int adc\_lanes\_sync\_timeout Shows which ADC lanes were unable to sync before a timeout. Each bit corresponds to one lane. Useful debugging information. ### int adc\_sync\_retry\_count The number of ADC lane synchronization retries. Default is set to 0. Useful debugging information. ### int adc\_sync\_strobe\_retry\_count The number of ADC strobe synchronization retries. Default is set to 0. Useful debugging information. ### int adc\_sync\_delay\_count 16 Bit number showing when the last ADC lane synchronization was achieved. Useful debugging information. ### crono\_bool\_t adc\_mgt\_power\_good Shows if the supplied mgt power is sufficient. Useful debugging information. ### ${\tt crono\_bool\_t~lmk\_pll1\_locked}$ Shows if lmk\_pll1 is locked. Useful debugging information. #### crono\_bool\_t lmk\_pl12\_locked Shows if Imk\_pll2 is locked. Useful debugging information. ### crono\_bool\_t lmk\_lost\_lock Shows if lmk lost lock. Useful debugging information. ### int lmk\_lock\_wait\_count Wait count of the lmk. Useful debugging information. ### int lmk\_ctrl\_vcxo Usefull for hardware debugging. ### crono\_bool\_t lmx\_locked Imx locked. Useful debugging information. ### crono\_bool\_t lmx\_lost\_lock lmx lost lock. Useful debugging information. Imx lock wait count. Useful debugging information. ### struct crono\_pcie\_info Structure containing PCle information. ### **Public Members** ### uint32\_t pwr\_mgmt Organizes power supply of PCIe lanes. ### uint32\_t link\_width Number of PCIe lanes that the card uses. Should be 1, 2, or 4 for Ndigo5G and 1, 2, 4, or 8 for the Ndigo6G-12. Ideally, should be the respective maximum. ### uint32\_t max\_payload Maximum size in bytes for one PCIe transaction. Depends on the system configuration. ### uint32\_t link\_speed Data rate of the PCIe card. Depends on the system configuration. ### uint32\_t error\_status\_supported Different from 0 if the PCIe error status is supported for this device. ### uint32\_t correctable\_error\_status Correctable error status flags, directly from the PCIe config register. Useful for debugging PCIe problems. 0, if no error is present, otherwise one of CRONO\_PCIE\_\* ### uint32\_t uncorrectable\_error\_status Uncorrectable error status flags, directly from the PCIe config register. Useful for debugging PCIe problems. 0, if no error is present, otherwise one of CRONO\_PCIE\_UNC\_\*. ### uint32\_t reserved For future extension. ## 3.4 Configuration The Ndigo6G-12 board is configured with a configuration structure (ndigo6g12\_configuration). The user should first obtain a standard set of configuration parameters using ndigo6g12 get default configuration(), then modify only the necessary parameters to their specific needs. The configuration itself is done by calling <a href="mailto:ndigo6g12\_configure">ndigo6g12\_configure</a> (). int ndigo6g12\_get\_default\_configuration(ndigo6g12\_device \*device, ndigo6g12\_configuration \*config) Copies the default configuration to the specified config pointer. Default values of ndigo6g12\_configuration: - · adc mode = - NDIGO6G12\_ADC\_MODE\_A (if application\_type = NDIGO6G12\_APP\_TYPE\_1CH) - NDIGO6G12\_ADC\_MODE\_AD (if application\_type = NDIGO6G12\_APP\_TYPE\_2CH) - NDIGO6G12\_ADC\_MODE\_ABCD (if application\_type = NDIGO6G12\_APP\_TYPE\_4CH) - NDIGO6G12\_ADC\_MODE\_A (if application\_type = NDIGO6G12\_APP\_TYPE\_AVRG) - adc cal set = 3 - analog\_offsets[i] = 0 - tdc\_trigger\_offsets[i] = NDIGO6G12\_DC\_OFFSET\_N\_NIM - trigger[i]: - edge = true - rising = false - threshold = 512 - trigger\_block[i]: - enabled = false - retrigger = false - multi shot count = 1 - precursor = 0 - *length* = 16 - sources = NDIGO6G12\_TRIGGER\_SOURCE\_0 - gates = NDIGO6G12\_TRIGGER\_GATE\_NONE - minimum\_free\_packets = 0 - gating\_block[i]: - negate = false - retrigger = false - start = 0 - stop = 1000 - sources = NDIGO6G12\_TRIGGER\_SOURCE\_0 - tdc\_configuration: - channel[i]: - \* enable = false - \* gating\_block: - · enable = false - · negate = false - · retrigger = false - retrigger = NDIGO6G12\_TRIGGER\_SOURCE\_AUTO - start = 0 - · *stop* = 1000 - sources = NDIGO6G12\_TRIGGER\_SOURCE\_0 - \* tiger\_block: - · mode = NDIGO6G12\_TIGER\_OFF - · negate = true - · retrigger = false - retrigger = NDIGO6G12\_TRIGGER\_SOURCE\_AUTO - start = 0 - $\cdot$ stop = 1 - sources = NDIGO6G12\_TRIGGER\_SOURCE\_0 - skip\_alignment = false - alignment\_mode = false - alignment\_pin\_high\_z = false - alignment\_pin\_invert = false - alignment\_phase\_steps = 6 - send\_empty\_packets = false - auto\_trigger\_period = 200000 - auto\_trigger\_random\_exponent = 0 - output\_mode = - NDIGO6G12\_OUTPUT\_MODE\_SIGNED32 (if application\_type = NDIGO6G12\_APP\_TYPE\_AVRG - NDIGO6G12\_OUTPUT\_MODE\_SIGNED16 (otherwise) - extended\_bandwidth = false ramp\_test\_mode = false #### **Parameters** - device [in] Pointer to the device from which to get the information. - · config [out] Pointer to a structure in which to store the configuration values. #### **Returns** See Function return values. int ndigo6g12\_configure(ndigo6g12\_device \*device, ndigo6g12\_configuration \*config) Configures the Ndigo6G-12 device. The config information is copied such that it can be changed after the call to ndigo6g12\_configure. #### **Parameters** - device [in] Pointer to the device from which to get the information. - config [out] Pointer to the configuration structure. #### Returns See Function return values. ### struct ndigo6g12\_configuration Structure that contains the configuration values for the Ndigo6G-12. This structure contains the configuration information. It is used in conjunction with ndigo6g12\_get\_default\_configuration() and ndigo6g12\_configure(). ### **Public Members** ### int adc\_mode Configure ADC mode. The chosen ADC mode has to be supported by the current NDIGO6G12\_APP\_TYPE. For example, if NDIG06G12 APP TYPE 1CH is used, one cannot choose, e.g., adc mode = NDIGO6G12 ADC MODE AA, but one has to either choose NDIGO6G12 ADC MODE A or NDIGO6G12\_ADC\_MODE\_D. Default value depends on ndigo6g12\_init\_parameters::application\_type. - NDIGO6G12\_APP\_TYPE\_4CH: NDIGO6G12\_ADC\_MODE\_A - NDIGO6G12\_APP\_TYPE\_2CH: NDIGO6G12\_ADC\_MODE\_AD - NDIGO6G12\_APP\_TYPE\_1CH: NDIGO6G12\_ADC\_MODE\_ABCD For more information, see Section 2.1. Must be one of the following: ### NDIGO6G12\_ADC\_MODE\_ABCD 4-channel mode at 1600 Msps sample rate ### NDIGO6G12\_ADC\_MODE\_AADD 4-channel mode at 1600 Msps sample rate ### NDIGO6G12\_ADC\_MODE\_AAAA 4-channel mode at 1600 Msps sample rate ### NDIGO6G12\_ADC\_MODE\_DDDD 4-channel mode at 1600 Msps sample rate ### NDIGO6G12\_ADC\_MODE\_AD 2-channel mode at 3200 Msps sample rate ### NDIGO6G12\_ADC\_MODE\_AA 2-channel mode at 3200 Msps sample rate ### NDIGO6G12\_ADC\_MODE\_DD 2-channel mode at 3200 Msps sample rate ### NDIGO6G12\_ADC\_MODE\_A 1-channel mode at 6400 Msps sample rate ### NDIGO6G12\_ADC\_MODE\_D 1-channel mode at 6400 Msps sample rate ### int adc\_cal\_set Select ADC calibration set. Default is 3. Do not change. ### double analog\_offsets[NDIGO6G12\_ADC\_CHANNEL\_COUNT] Set the offsets of the ADC inputs in V. The indices 0 to 3 of the array correspond to ADC channels A to D. Must be between $\pm$ 0.5 V. Defaults are 0 V for each ADC channel. ### double tdc\_trigger\_offsets[NDIGO6G12\_TDC\_CHANNEL\_COUNT] Set DAC for trigger threshold of the TDC inputs in V. Channel assignment: - 0 to 3: high-resolution TDC, inputs E to H - · 4 and 5: inputs TRG and GATE Set to a value between -1.32 V and +2.0 V. This should be close to 50% of the height of your pulses on the inputs. Examples for various signaling standards are defined below. The inputs are AC coupled. This means that for pulse inputs the absolute voltage is not important. Only the relative pulse amplitude causes the input circuits to switch. tdc\_trigger\_offset for an input must be set to the relative switching voltage for the input standard in use. If the pulses are negative, a negative switching threshold must be set and vice versa. Defaults are NDIGO6G12\_DC\_OFFSET\_N\_NIM for each TDC channel. Defines for various signal standards: NDIGO6G12\_DC\_OFFSET\_P\_NIM NDIGO6G12\_DC\_OFFSET\_P\_CMOS NDIGO6G12\_DC\_OFFSET\_P\_LVCMOS\_33 NDIGO6G12\_DC\_OFFSET\_P\_LVCMOS\_25 NDIGO6G12\_DC\_OFFSET\_P\_LVCMOS\_18 NDIGO6G12\_DC\_OFFSET\_P\_TTL NDIGO6G12\_DC\_OFFSET\_P\_LVTTL\_33 NDIGO6G12\_DC\_OFFSET\_P\_LVTTL\_25 NDIGO6G12\_DC\_OFFSET\_P\_SSTL\_3 NDIGO6G12\_DC\_OFFSET\_P\_SSTL\_2 NDIGO6G12\_DC\_OFFSET\_N\_NIM NDIGO6G12\_DC\_OFFSET\_N\_CMOS NDIGO6G12\_DC\_OFFSET\_N\_LVCMOS\_33 NDIGO6G12\_DC\_OFFSET\_N\_LVCMOS\_25 NDIGO6G12\_DC\_OFFSET\_N\_LVCMOS\_18 NDIGO6G12\_DC\_OFFSET\_N\_TTL NDIGO6G12\_DC\_OFFSET\_N\_LVTTL\_33 NDIGO6G12\_DC\_OFFSET\_N\_LVTTL\_25 ``` NDIGO6G12_DC_OFFSET_N_SSTL_3 ``` NDIGO6G12\_DC\_OFFSET\_N\_SSTL\_2 ### ndigo6g12\_trigger trigger[NDIGO6G12\_TRIGGER\_COUNT] Configuration of the external trigger sources. The entries in the array correspond to the following defines. ndigo6g12\_trigger::threshold is ignored for index NDIGO6G12\_TRIGGER\_TDC0 and above. ndigo6g12\_trigger::edge and ndigo6g12\_trigger::rising are ignored for indeces NDIGO6G12\_TRIGGER\_AUTO and NDIGO6G12\_TRIGGER\_ONE. NDIGO6G12\_TRIGGER\_AO NDIGO6G12\_TRIGGER\_A1 NDIGO6G12\_TRIGGER\_BO NDIGO6G12\_TRIGGER\_B1 NDIGO6G12\_TRIGGER\_CO NDIGO6G12\_TRIGGER\_C1 NDIGO6G12\_TRIGGER\_DO NDIGO6G12\_TRIGGER\_D1 NDIGO6G12\_TRIGGER\_TDC0 NDIGO6G12\_TRIGGER\_TDC1 NDIGO6G12\_TRIGGER\_TDC2 NDIGO6G12\_TRIGGER\_TDC3 NDIGO6G12\_TRIGGER\_TRG NDIGO6G12\_TRIGGER\_GATE NDIGO6G12\_TRIGGER\_AUTO ### NDIGO6G12 TRIGGER ONE ### NDIGO6G12\_TRIGGER\_FPGAO Deprecated. Alias for NDIGO6G12\_TRIGGER\_TRG. ### NDIGO6G12\_TRIGGER\_FPGA1 Deprecated. Alias for NDIGO6G12\_TRIGGER\_GATE. ### ndigo6g12\_trigger\_block trigger\_block[NDIGO6G12\_ADC\_CHANNEL\_COUNT] Trigger settings of ADC inputs. The number of input channels depends on ADC mode. ### ndigo6g12\_gating\_block gating\_block[NDIGO6G12\_GATE\_COUNT] Configuration of gating blocks. Gating blocks are used to filter trigger. ### ndigo6g12\_tdc\_configuration tdc\_configuration Configuration of TDC channels. ### ndigo6g12\_averager\_configuration average\_configuration Configuration of the Averager. ### int auto\_trigger\_period Component to create a trigger either periodically or randomly. To be exact, there are two parameters $M = auto\_trigger\_period$ and $N = auto\_trigger\_ran$ dom\_exponent that result in a distance between triggers of $T = M + [1...2^N] - 1$ clock cycles, where $6 \le M < 2^{32}$ and $0 \le N < 32$ . There is no enable or reset as the usage of this trigger can be configured in the channels. Each clock cycle is 5 ns. Default is 200000, corresponding to a 1 kHz auto trigger. ### int auto\_trigger\_random\_exponent Component to create a trigger either periodically or randomly. See auto\_trigger\_period. Default is 0. ### int output\_mode Output mode of the ADC data. Default value depends on ndigo6g12\_init\_parameters::application\_type. - NDIGO6G12\_APP\_TYPE\_AVRG: NDIGO6G12\_OUTPUT\_MODE\_SIGNED32 - otherwise: NDIGO6G12\_OUTPUT\_MODE\_SIGNED16. Must be one of the following: ### NDIGO6G12 OUTPUT MODE RAW Return the native range (0 to 4095) augmented by two ADC control bits per sample. Not supported for user applications. ### NDIGO6G12\_OUTPUT\_MODE\_SIGNED16 Return a signed 16 integer. The range is -32768 to 32767. ### NDIGO6G12\_OUTPUT\_MODE\_SIGNED32 Output in signed32 integer format. Must be used in (and only in) averaging mode. The range is $-2^{31}$ to $2^{31} - 1$ . ### NDIGO6G12\_OUTPUT\_MODE\_RAW\_NO\_CB Return the native range (0 to 4095). For more information, see Section 4.3. ### crono\_bool\_t extended\_bandwidth Extended bandwidth. If true, the input bandwidth is 6.5 GHz instead of the default 4.5 GHz. Since the extended input bandwidth of the ADC influences the total bandwidth of the Ndigo6G-12 board only in a minimal manner, we recommend using the non-extended input bandwidth of 4.5 GHz. This ensures the best signal-to-noise ratio. Default is false. ### crono\_bool\_t ramp\_test\_mode Default is false. Do not change. ### crono\_bool\_t sample\_averaging Calculate sample average for multi-sampling modes AAAA, DDDD, AADD, AA, and DD. Manipulate the output in multi-sampling modes. - true: Average all samples and combine them to a single output. - · false: Output all samples in their own package. For more information, see Multiple Sampling Modes in Section 2.1. ### struct ndigo6g12\_trigger Structure that contains trigger settings. #### **Public Members** #### short threshold Threshold controlling when the ADC channel is active. Sets the threshold for the trigger block within the range of the ADC data. The range depends on ndigo6g12\_configuration::output\_mode: - NDIGO6G12\_OUTPUT\_MODE\_RAW: 0 to 4095 - NDIGO6G12\_OUTPUT\_MODE\_SIGNED16 and NDIGO6G12\_OUTPUT\_MODE\_SIGNED32: -32768 to 32767 For trigger indices NDIGO6G12\_TRIGGER\_TDC to NDIGO6G12\_TRIGGER\_ONE the threshold is ignored. For the TDC channels, the trigger threshold is controlled by ndigo6g12\_configuration::tdc\_trigger\_offsets. #### Note NDIGO6G12\_OUTPUT\_MODE\_SIGNED32 is only used for NDIGO6G12\_APP\_TYPE\_AVRG. ### crono\_bool\_t edge Enables edge-trigger functionality. For trigger indices NDIGO6G12\_TRIGGER\_AUTO and NDIGO6G12\_TRIGGER\_ONE this is ignored. - · false: Use a level trigger. The level trigger triggers as long as the signal is above or below (depending on rising) the set threshold. Followingly, the trigger gives the sign of the signal in reference to the threshold. - · true: Use an edge trigger. The edge trigger triggers as soon as its set threshold is crossed by the signal. Thus, the roots in reference to the threshold are recorded. Default is true. ### crono\_bool\_t rising Sets rising-edge trigger functionality. For trigger indices NDIGO6G12\_TRIGGER\_AUTO and NDIGO6G12\_TRIGGER\_ONE, this is ignored. - · If edge is true (i.e., an edge trigger is used): - false: Trigger when the signal crosses from above to below the threshold. - true: Trigger when the signal crosses from below to above the threshold. - If edge is false (i.e., a level trigger is used): - false: Triggers the part of the signal below the threshold. - true: Triggers the part of the signal above the threshold. Default is false. ### struct ndigo6g12\_trigger\_block Configuration of the trigger block. #### **Public Members** #### crono\_bool\_t enabled Activates triggers on this channel. ### crono\_bool\_t retrigger Enable retrigger functionality. If a new trigger condition occurs while the postcursor is acquired (i.e., within the time frame controlled by length), the packet is extended by starting a new postcursor. Otherwise the new trigger is ignored and the packet ends after the postcursor of the first trigger. #### int multi\_shot\_count Number of packets created in single-shot mode (i.e., ndigo6g12\_single\_shot() was called) before packet generation stops. This value is ignored if enabled is true. Maximum is NDIGO6G12\_MAX\_MULTISHOT. Note: Up to firmware revision 1.24120, this feature is bugged in 4-channel mode while multi\_shot\_count > 1. ### int precursor Precursor in multiples of 5 ns. The amount of data preceding a trigger that is captured. The maximum is NDIGO6G12 MAX PRECURSOR. ### int length Length of the postcursor in multiples of 5 ns. The total amount of data that is recorded in addition to the trigger window is controlled by length and precursor. precursor determines the amount of data before the trigger window, length the amount of data after the trigger condition was false the first time. In edge-trigger mode, the trigger window is always 1 (i.e., 5 ns long). Otherwise, (level-trigger mode) the trigger window is as long as the trigger condition was fulfilled. The maximum value is NDIGO6G12\_FIFO\_DEPTH minus ndigo6g12\_trigger\_block::precursor minus trigger window. #### int sources A bit mask with a bit set for all trigger sources that can trigger this channel. Default NDIGO6G12\_TRIGGER\_SOURCE\_0 (NDIGO6G12\_TRIGGER\_SOURCE\_A0 for ADC channel A, NDIGO6G12\_TRIGGER\_SOURCE\_B0 for ADC channel B, etc). ### int gates A bit mask with a bit set for all trigger gates. Mask which selects the gates that have to be open for the trigger block to use. Default NDIGO6G12\_TRIGGER\_GATE\_NONE. The following defines can be used to create the bit mask: NDIGO6G12\_TRIGGER\_GATE\_NONE NDIGO6G12\_TRIGGER\_GATE\_O NDIGO6G12\_TRIGGER\_GATE\_1 NDIGO6G12\_TRIGGER\_GATE\_2 NDIGO6G12\_TRIGGER\_GATE\_3 ### double minimum\_free\_packets Number of packets that fit into the FIFO. This parameter sets how many packets are supposed to fit into the on-board FIFO before a new packet is recorded after the FIFO was full, i.e., a certain amount of free space in the FIFO is demanded before a new packet is written after the FIFO was full. As a measure for the packet length, the recording window as defined by precursor and length is used. The on-board algorithm checks the free FIFO space only in case the FIFO is full. Therefore, if this number is 1.0 or more, at least every second packet in the host buffer is guaranteed to have the full length set by the precursor and length. In many cases smaller values will also result in full length packets. But below a certain value multiple packets that are cut off at the end will show up. Default is 0. ### struct ndigo6g12\_gating\_block Contains settings of the gating block. After a signal at one of the sources is detected, a timer starts running. Once the timer reaches the value specified by start, a gate is opened (or closed, depending on negate) until the timer reaches the time specified by stop. What happens in the event that another signal before stop is detected is controlled by retrigger. See also Section 2.4. #### **Public Members** ### crono\_bool\_t negate Invert output polarity. If false (true) the gate is opened (closed) inbetween the times specified by start and stop. Default is false. ### crono\_bool\_t retrigger Enable retriggering. If enabled and a second trigger event is detected before the timer reaches stop, the timer is restarted. Otherwise signals at the input sources are ignored until stop is reached. Default is false. #### int start The time from the first input signal seen in the idle state until the gating output is set. In multiples of 5 ns. $0 \le \text{start} < 2^{16}$ , while start $\le \text{stop}$ . Default is 0. ### int stop The number of samples from leaving the idle state until the gating output is reset. In multiples of 5 ns. $0 \le \text{stop} < 2^{16}$ , while stop $\ge \text{start}$ . Default is 1000. #### int sources Bit mask with a bit set for all trigger sources that can trigger this channel. Default NDIGO6G12\_TRIGGER\_SOURCE\_0 (NDIGO6G12\_TRIGGER\_SOURCE\_A0 for ADC channel A, NDIGO6G12\_TRIGGER\_SOURCE\_B0 for ADC channel B, etc). ### struct ndigo6g12\_tdc\_configuration Contains configuration information of the TDC channels. ### **Public Members** ### ndigo6g12\_tdc\_channel channel[NDIGO6G12\_TDC\_CHANNEL\_COUNT] Configure polarity, type and threshold for the TDC channels. ### crono\_bool\_t skip\_alignment Configure THS788 calibration. - · true: Skip THS788 calibration. - false: Do THS788 calibration (default). Default is false. ### crono\_bool\_t alignment\_mode Align TDC channels. Default is false. ### crono\_bool\_t alignment\_pin\_high\_z Default is false. ### crono\_bool\_t alignment\_pin\_invert Default is false. ### int alignment\_phase\_steps Default is 6. ### crono\_bool\_t send\_empty\_packets Default is false. ### struct ndigo6g12\_averager\_configuration Contains averaging settings. #### **Public Members** #### int iterations Set the number of trigger events that are averaged. Must be 0 if no averaging application is installed on the Ndigo6G-12 (see ndigo6g12\_init\_parameters::application\_type). Default is 0. ### crono\_bool\_t stop\_on\_overflow Stops averaging before an overflow can happen. Stops the averaging once averaging\_value ≥ max\_averaging\_value - max\_ADC\_value or $averaging\_value \le min\_averaging\_value - min\_ADC\_value$ to prevent overflow. - max(min)\_averaging\_value is 2097151 (-2097152) - max(min)\_ADC\_value is 32768 (-32767) Default is false. ### crono\_bool\_t stop\_manual Stops the averaging manually. Software stop for averaging. If an averaging iteration has already started it is finished before the averaging will stop. Default is false. ### crono\_bool\_t use\_saturation Determines if saturation arithmetic is used by the averager. • true: Instead of averaging\_value over(under)flowing once max(min)\_averaging\_value is reached, the maximum (minimum) value is kept. · false: Once averaging\_value reaches max(min)\_averaging\_value, averaging\_value will over(under)flow and wrap around. See stop\_on\_overflow for the values of averaging\_value and max(min)\_averaging\_value. Default is true. ### crono\_bool\_t stop\_on\_timeout Determine if the averager stops on timeout. The timeout time is configured by timeout\_threshold. Default is false. ### int timeout\_threshold Set the number of microseconds until timeout. Must be 0 if no averaging application is installed on the Ndigo6G-12 board. Default is 0. ### struct ndigo6g12\_tdc\_channel Contains TDC channel settings. #### **Public Members** ### crono\_bool\_t enable Enable TDC channel. Default is false. ### crono\_bool\_t reserved3 Reserved for future extension. ### crono\_bool\_t reserved2 Reserved for future extension. #### crono\_bool\_t reserved1 Reserved for future extension. ### ndigo6g12\_tdc\_gating\_block gating\_block Configuration of the gating blocks. ### ndigo6g12\_tdc\_tiger\_block tiger\_block Configuration of the TiGer blocks. ### struct ndigo6g12\_tdc\_gating\_block Contains settings of the gating blocks specifically for the TDCs. The functionality is similar to ndigo6g12\_gating\_block. #### **Public Members** ### crono\_bool\_t enable Activates gating block. #### crono\_bool\_t **negate** Inverts output polarity. Default is false. ### crono\_bool\_t retrigger Enable retriggering. If enabled and a second trigger event is detected before the timer reaches stop, the timer is restarted. Otherwise signals at the input sources are ignored until stop is reached. Defaults to false. #### int start The time from the first input signal seen in the idle state until the gating output is set. In multiples of 5 ns. $0 \le \text{start} < 2^{16}$ , while start $\le \text{stop}$ . Default is 0. ### int stop The number of samples from leaving the idle state until the gating output is reset. In multiples of 5 ns. $0 \le \text{stop} < 2^{16}$ , while stop $\ge \text{start}$ . Default is 1000. ### int sources Bit mask with a bit set for all trigger sources that can trigger this channel. Default NDIGO6G12\_TRIGGER\_SOURCE\_0 (NDIGO6G12\_TRIGGER\_SOURCE\_A0 for ADC channel A, NDIGO6G12\_TRIGGER\_SOURCE\_B0 for ADC channel B, etc). ### struct ndigo6g12\_tdc\_tiger\_block Contains settings of TiGer block. The configuration is similiar to *ndigo6g12\_gating\_block*. #### **Public Members** #### int mode Enables the desired mode of operation for the TiGer. Default is NDIGO6G12\_TIGER\_OFF. Must be one of the following: ### NDIGO6G12\_TIGER\_OFF TiGer deactivated. ### NDIGO6G12\_TIGER\_OUTPUT Pulse height is approximately 2 V. Connected hardware must not drive any signals to the connectors used as outputs, as doing so could damage both the Ndigo6G-12 and the external hardware. We recommend to only use short pulses to avoid undesirable baseline shift due to the AC coupling, but the device does not pose any restrictions on the duty cycle. This mode can be used as a clock output with a frequency of 75/N MHz (for integer N). ### NDIGO6G12\_TIGER\_BIDI Pulse height is approximately 1 V. The LEMO connector may be used as input with OR function. Use short pulses to keep the probability of collision and the effect on the baseline low. ### NDIGO6G12\_TIGER\_BIPOLAR TiGer pulses are bipolar. Not supported for inputs TRG and GATE. In this mode, the connector creates bipolar pulses with 1 V amplitude. The connector can still be used as an input. Pulses have no effect on the baseline offset. The TiGer should be configured with start= stop + 1 for minimium-width bipolar pulses. The maximum bipolar pulse width is NDIGO6G12\_TIGER\_MAX\_BIPO-LAR\_PULSE\_LENGTH. ### crono\_bool\_t negate Set pulse polarity. The TiGer creates a high pulse from start to stop unless negated. Default is true. ### crono\_bool\_t retrigger Enable retriggering. If enabled and a second trigger event is detected before the timer reaches stop, the timer is restarted. Otherwise signals at the input sources are ignored until stop is reached. Defaults to false. #### int start The time from the first input signal seen in the idle state until the TiGer outputs a signal. In multiples of 5 ns. $0 \le \text{start} < 2^{16}$ , while start $\le \text{stop}$ . Default is 0. ### int stop The number of samples from leaving the idle state until the TiGer output is reset. In multiples of 5 ns. $0 \le \text{stop} < 2^{16}$ , while stop $\ge \text{start}$ . Note that the maximum length for bipolar pulses is given by NDIGO6G12\_TIGER\_MAX\_BIPO-LAR\_PULSE\_LENGTH. Default is 1. #### int sources Bit mask with a bit set for all trigger sources that can trigger this channel. Default NDIGO6G12\_TRIGGER\_SOURCE\_0 (NDIGO6G12\_TRIGGER\_SOURCE\_A0 for ADC channel A, NDIGO6G12\_TRIGGER\_SOURCE\_B0 for ADC channel B, etc). ### 3.5 Runtime control int ndigo6g12\_start\_capture(ndigo6g12\_device \*device) Start data acquisition. #### **Parameters** device - [in] Pointer to the device. #### **Returns** See Function return values. int ndigo6g12\_stop\_capture(ndigo6g12\_device \*device) Stop data acquisition. #### **Parameters** device - [in] Pointer to the device. #### **Returns** See Function return values. int ndigo6g12\_manual\_trigger(ndigo6g12\_device \*device, int channel\_mask) Enables manual triggering of the ADC channels. #### **Parameters** - · device [in] Pointer to the device. - · channel\_mask [in] A bit mask that chooses which channels to trigger. ### Returns See Function return values. int ndigo6g12\_single\_shot (ndigo6g12\_device \*device, int channel\_mask) Enables single-shot recording of the ADC channels. Instead of continously triggering on input signals, only trigger and record a ndigo6g12\_trigger\_block::multi\_shot\_count number of events. Note: Up to firmware revision 1.24120, this feature is bugged in 4-channel mode while ndigo6g12\_trigger\_block::multi\_shot\_count >1. Requires that ndigo6g12\_trigger\_block::enabled is false. #### **Parameters** device – [in] Pointer to the device. · channel\_mask - [in] A bit mask that chooses which channels to trigger. #### **Returns** See Function return values. int ndigo6g12\_clear\_pcie\_errors(ndigo6g12\_device \*device, int flags) Clear PCIe errors. Only useful for PCIe problem debugging flags. #### **Parameters** - · device [in] Pointer to the device. - flags [in] Specify which flags to clear. - CRONO\_PCIE\_CORRECTABLE\_FLAG: clear all correctable errors - CRONO\_PCIE\_UNCORRECTABLE\_FLAG: clear all uncorrectable errors #### **Returns** char array containing the plain text error message. Relevant defines: CRONO\_PCIE\_CORRECTABLE\_FLAG CRONO\_PCIE\_UNCORRECTABLE\_FLAG ### 3.6 Readout After an Ndigo6G-12 board is initialized and capturing, the captured events can be read from the board with ndigo6g12\_read(). The read-out data is packaged in packets (see Chapter 4). int ndigo6g12\_read(ndigo6g12\_device \*device, ndigo6g12\_read\_in \*in, ndigo6g12\_read\_out \*out) Reads packets from the board. If ndigo6g12\_read\_in::acknowledge\_last\_read is true, automatically acknowledges the last read packets. ### **Parameters** - device [in] Pointer to the device that should be read. - in [in] Pointer to the structure that configures the read call. - out [out] Pointer to a structure in which the read-out should be stored. #### **Returns** See Function return values. const char \*ndigo6g12\_get\_last\_error\_message(ndigo6g12\_device \*device) Gets latest error message of device. #### **Parameters** device - [in] Pointer to the device. char array containing the plain text error message. ### const char \*ndigo6g12\_device\_state\_to\_str(int state) Convert state to plain text. #### **Parameters** **state** – **[in]** The device state as stored in *ndigo6g12\_fast\_info::state*. #### Returns char array containing the state as plain text. ### struct ndigo6g12\_read\_in The parameters of the read commands. #### **Public Members** ### crono\_bool\_t acknowledge\_last\_read Automatically acknowledge packets from the previous call of ndigo6g12\_read. Only acknowledged packets will release the memory of the DMA buffer. ### struct ndigo6g12\_read\_out Struct for the read-out of the Ndigo6G-12 packets. #### **Public Members** ### volatile crono\_packet \*first\_packet Pointer to the first packet. That is, the pointer that was captured by the call of ndigo6g12\_read. ### volatile crono\_packet \*last\_packet Pointer to the last packet. ### int error\_code Error code. Is one of the following: ### CRONO\_READ\_OK Reading packets from the device was successful. ### CRONO\_READ\_NO\_DATA Trying to read packets does not yield data. ### CRONO\_READ\_INTERNAL\_ERROR Some unhandled error occured. A device reinit is required. ### CRONO\_READ\_TIMEOUT Trying to read packets does not yield data in the given amount of time. | const char *error_message | | |------------------------------------------------------------|---| | Plain text error message. | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | cronologic GmbH & Co. KG Ndigo6G-12 User Guide, Rev. 1.4.0 | ) | ## 4 Packet Format Packets are retrieved by $ndigo6g12\_read()$ . They are of type $crono\_packet$ . - · Each hit on an ADC channel is stored in one packet. The format of the payload data (see crono\_packet::data) is explained in Section 4.3. - · All TDC hits within the time given by $ndigo6g12\_param\_info::tdc\_rollover\_period$ are stored in a single packet (stored in the payload data). The memory layout thereof is shown in Section 4.4. ## Note The minimum packet length depends on the application type. See Section 2.1 for details. ## Output Structure crono\_packet ``` struct crono_packet ``` ``` uint8_t channel ``` Source channel of the data. Values correspond to the following: ``` 0x0 - 0x3: ADC A - D ``` ≥ 0x4: TDC channels. Which specific TDC channel is encoded in crono\_packet::data. #### uint8\_t card ID of the card. ### uint8\_t type Type of the packet. Different packet types correspond to different encodings of crono\_packet::data. Is one of the following: ### CRONO\_PACKET\_TYPE\_16\_BIT\_SIGNED Used for ADC data. crono\_packet::data must be cast to int16\_t and crono\_packet::length must be multiplied by 4. ### CRONO PACKET TYPE TDC DATA Used for TDC data. crono\_packet::data must be cast to uint32\_t and crono\_packet::length must be multiplied by 2. CRONO\_PACKET\_TYPE\_AVRG\_DATA Used for averaged ADC data. crono\_packet::data must be cast to uint32\_t and crono\_packet::length must be multiplied by 2. ### uint8\_t flags Bit field of the following flags: ### CRONO\_PACKET\_FLAG\_SHORTENED Packet was truncated because internal FIFO was full. This means that less than the requested number of samples have been written. ### CRONO\_PACKET\_FLAG\_PACKETS\_LOST Lost triggers preceded this packet due to insufficient DMA buffers. The DMA controller has discarded packets due to the full host buffer. ### CRONO\_PACKET\_FLAG\_OVERFLOW The packet contains ADC sample overflows. ### CRONO\_PACKET\_FLAG\_TRIGGER\_MISSED Lost triggers preceded this packet due to insufficient buffers. The trigger unit has discarded packets due to a full FIFO. ### CRONO\_PACKET\_FLAG\_DMA\_FIFO\_FULL The internal DMA FIFO was full. Triggers only got lost if a subsequent package has crono\_packet::flags with a bit weight CRONO\_PACKET\_FLAG\_TRIGGER\_MISSED set. ### CRONO\_PACKET\_FLAG\_HOST\_BUFFER\_FULL The host buffer was full. Triggers only got lost if a subsequent package has crono\_packet::flags with a bit weight CRONO\_PACKET\_FLAG\_TRIGGER\_MISSED set. ### CRONO\_PACKET\_FLAG\_TDC\_NO\_EDGE The packet from a TDC does not contain valid data. Hence, the timestamp is not corrected. No valid edge was found for the TDC. For TDC data, can also be one of the following: ### NDIGO6G12\_TDC\_PACKET\_FLAG\_RESERVED ### NDIGO6G12\_TDC\_PACKET\_FLAG\_CONTAINS\_DATA Packet contains at least one TDC event. At least one packet was lost due to full FIFO. ### NDIGO6G12\_TDC\_PACKET\_FLAG\_SHORTENED The trigger unit has shortend the current packet due to full FIFO. ### NDIGO6G12\_TDC\_PACKET\_FLAG\_DMA\_FIFO\_FULL The DMA FIFO was full. Trigger only got lost if a subsequent package has crono\_packet::flags with a bit weight NDIGO6G12\_TDC\_PACKET\_FLAG\_LOST set. ### NDIGO6G12\_TDC\_PACKET\_FLAG\_HOST\_BUFFER\_FULL The host buffer was full. Trigger only got lost if a subsequent package has crono\_packet::flags with a bit weight NDIGO6G12\_TDC\_PACKET\_FLAG\_LOST set. #### uint32\_t length Length of crono packet::data in multiples of 64 bits. The actual length of crono\_packet::data depends on crono\_packet::type. ### uint64\_t timestamp Timestamp of the packet. For the Ndigo6G-12, this corresponds to the beginning of the packet data. ### uint64\_t data[1] Payload data of the packet. The length of data corresponds to crono packet::length. The data type must be cast according to crono packet::type, and the data encoding also depends on crono packet::type. See Section 4.3 for the data encoding of ADC data. See Section 4.4 for the data encoding of TDC data. See Section 4.5 for the data encoding of averaged ADC data. ## 4.2 Utility macros The following macros can be used to navigate through the packets obtained by $ndigo 6g12\_read()$ . ### crono\_packet\_data\_length(current) Returns the length of crono\_packet::data in multiples of 8 bytes. ### crono\_packet\_bytes(current) Returns the length of crono\_packet::data including its header in bytes. ### crono\_next\_packet(current) Returns a crono\_packet pointer pointing to the next packet in the host buffer. Must be checked before use to not point beyond the last packet of the readout data, e.g., crono next packet(current packet) <= readout data.last packet.</pre> ## 4.3 Data encoding for ADC hits data, that is, the packet-data payload, depends on ndigo6g12\_configuration::output\_mode. The length of the data array is encoded in length. Be aware that length is in multiples of 64 bit, while the size of the fields of data depends on type. Thus, reading packet data requires the following steps: - Depending on crono\_packet::type, multiply length appropriately. E.g., if type is CRONO PACKET TYPE 16 BIT SIGNED, length has to be multiplied by 4 (since 4 × 16 bit = 64 bit). - · Cast data according to type. E.g., if type is CRONO PACKET TYPE 16 BIT SIGNED, cast data to int16 t. ### 4.3.1 NDIGO6G12\_OUTPUT\_MODE\_SIGNED16 Raw data of the ADC is mapped to the range of a signed 16 integer (-32768 to 32767). Packet data must be cast to int16 t. ### 4.3.2 NDIGO6G12 OUTPUT MODE RAW Packet data is returned in the native range of the ADC (0 to 4095). It must be cast to int16 t. #### Data layout: | Bit | 15 | 14 | 13 | 12 | 11 | 10 | | 0 | | |------|----|----|--------------|----|-------------|----|--|---|--| | Data | 0 | 0 | control bits | | sample data | | | | | ### 4.3.3 NDIGO6G12\_OUTPUT\_MODE\_RAW\_NO\_CB Packet data is returned in the native range of the ADC (0 to 4095). It must be cast to int16 t. Unlike NDIGO6G12 OUTPUT MODE RAW, it does not contain control bits. ### Attention NDIGO6G12 OUTPUT MODE RAW and NDIGO6G12 OUTPUT MODE RAW NO CB are useful for debugging purposes. They are not supported for user applications. Use NDIGO6G12 OUT-PUT\_MODE\_SIGNED16 instead. ### 4.3.4 NDIGO6G12\_OUTPUT\_MODE\_SIGNED32 Only used if ndigo6g12\_init\_parameters::application\_type is NDIG06G12\_APP\_TYPE\_AVRG. See Section 4.5 for more information. ## 4.4 Data encoding for TDC hits The following bit table shows the encoding of the payload data (crono packet::data) of all recorded TDC hits within the time-frame given by ndigo6g12 param info::tdc rollover period. | Bit | 31 | 30 | | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 | |------|-----------|----|--|---------------|---|---|----------------|---|---|---|---|---|---| | Data | Timestamp | | | TDC hit flags | | | Channel number | | | r | | | | #### Details: - · The timestamp is relative to crono\_packet::timestamp and is given in units of ndigo6g12 param info::tdc period. - · The channel numbers are: 0x0: TDC channel 1 0x1: TDC channel 2 0x2: TDC channel 3 0x3: TDC channel 4 0x4: TRG 0x5: GATE 0xD: Dummy data 0xF: Rollover marker The TDC hit flags are one of the following: ### NDIGO6G12\_TDC\_HIT\_FLAG\_LOST At least one preceding event was lost due to full FIFO. ### NDIGO6G12\_TDC\_HIT\_FLAG\_ROLLOVER\_LOST Rollover has been lost due to full FIFO. Results in a fatal error. ### NDIGO6G12\_TDC\_HIT\_FLAG\_VALID Timestamp is a valid TDC event. ### NDIGO6G12\_TDC\_HIT\_FLAG\_GROUP\_TIME\_ROLLOVER Timestamp is a rollover marker. Add ndigo6g12\_param\_info::tdc\_rollover\_period to all subsequent timestamps in the packet. ### NDIGO6G12\_TDC\_HIT\_ERROR\_MASK TDC hit flag mask for error reporting. ### NDIGO6G12\_TDC\_HIT\_TYPE\_MASK TDC hit flags mask for timestamp type. ### NDIGO6G12\_TDC\_PADDING\_DATA\_CHANNEL TDC hit channel number for padding-data. Padding-data can be ignored. Does not contain any usefull information. Padding-data has NDIGO6G12\_TDC\_HIT\_FLAG\_GROUP\_TIME\_ROLLOVER and NDIGO6G12\_TDC\_HIT\_FLAG\_VALID always cleared. ### NDIGO6G12\_TDC\_ROLLOVER\_CHANNEL TDC hit channel number for rollover marker. Rollover marker has NDIGO6G12\_TDC\_HIT\_FLAG\_GROUP\_TIME\_ROLLOVER always set. # Data encoding for averaged ADC hits When using NDIGO6G12\_APP\_TYPE\_AVRG, the first two 64-bit words of crono\_packet::data are the extended header containing more information, as shown in the following bit table. | Bits | 127 – 38 | 27 – 32 | 31 – 12 | 11 - 0 | |------|----------|---------|----------|----------------------| | Data | reserved | flags | reserved | iterations performed | ### Flags: - 0x01: stopped iterations prematurely - · 0x02: overflow detected - · 0x04: stopped by timeout - 0x08: stopped by software - 0x10: stopped by overflow The following data words contain the raw ADC data mapped to the range of a signed 32 integer ( $-2^{31}$ to 2<sup>31</sup> -1). Thus, crono\_packet::data must be cast to int32\_t and crono\_packet::length must be multiplied by 2 taking into account the extended header. That is, ``` uint32_t extended_header_length = 2; uint32_t sample_count = ((pkt->length - extended_header_length) * 2); int32_t* adc_data = (int32_t*)(pkt->data + extended_header_length); for (uint32_t i = 0; i < sample_count; i++) {</pre> /* work with adc_data[i] */ } ``` ## 5 C++-Example The following source code is an example of an Ndigo6G-12 application written in C++. The source code is also available on our GitHub. | Source file | Description | |---------------------------------|-----------------------------------------------------------------------------------| | ndigo6g12_example.cpp | Main source-code file of the example application. | | ndigo6g12_app.h | Header file for classes for different Ndigo6G-12 application types and TDC setup. | | ndigo6g12_adc_sin-<br>gle.cpp | Implementation of application type NDIGO6G12_APP_TYPE_1CH. | | ndigo6g12_adc_dual.cpp | Implementation of application type NDIGO6G12_APP_TYPE_2CH. | | ndigo6g12_adc_quad.cpp | Implementation of application type NDIGO6G12_APP_TYPE_4CH. | | ndigo6g12_adc_aver-<br>ager.cpp | Implementation of application type NDIGO6G12_APP_TYPE_AVRG. | | ndigo6g12_tdc.cpp | Implementation of the TDC-class. | | delay.h | Implementation for measuring delays. | ## ndigo6g12\_example.cpp ``` // Example application for the Ndigo6G-12 #include "ndigo6g12_app.h" #include "ndigo6g12 interface.h" #include <map> #include <stdio.h> #include <stdlib.h> 8 std::map<int, std::string> appTypeMap = {{1, "One ADC channels @6.4 Gsps"}, 9 {2, "Two ADC channels @3.2 Gsps"}, {4, "Four ADC channels @1.6 Gsp"}, 11 {5, "Averaging mode @6.4 Gsps"}}; 12 13 std::map<int, std::string> requirementsMap = { 14 15 "Starts the test of the currently configured app type"}, 16 {1, "Measure time distance between passing of " 18 "threshold, calculates the frequency, requires NIM signal on channel A"}, 19 {2, "Dual-channel application that measures delay between start " 20 "pulse on channel A and stop pulse on channel D (NIM)"}, 21 (continues on next page) ``` ``` (continued from previous page) {4, "Quad-channel application that measures delay between start " 22 "pulse on channel A and stop pulses on channels B-D (NIM)"}, 23 {5, "Measure time distance between averaged start on TRG (NIM) and stop on " 24 "channel A (falling) by summing data of 16 runs " "to increase precision of measurement for signal with low amplitude"}}; Ndigo6GApp *adcApp; 28 ndigo6g12_param_info paramInfo; 29 30 // initialize Ndigo6G-12 device 31 ndigo6g12_device initialize_ndigo6g12(int bufferSize, int boardId, int cardIndex, int appType, int_ // prepare initialization 34 ndigo6g12_init_parameters params; 35 // fill initialization data structure with default values 36 // so that the data is valid and only parameters 37 // of interest have to be set explicitly ndigo6g12_get_default_init_parameters(&params); 39 params.application_type = appType; 40 41 params.buffer_size[0] = bufferSize; // size of the packet buffer 42 params.board_id = boardId; // value copied to "card" field of every packet, 43 // allowed range 0..255 params.card_index = cardIndex; // which of the Ndigo6G-12 board found in 45 // the system to be used // this specifies the directories or the specific .cronorom if dynamic 47 // switching of appType is required. If not specified, the example will 48 // return an error if the appType does not match the current appType in the 49 // firmware 50 params.firmware locations = "."; 51 // initialize card 53 int errorCode; 54 const char *errorMessage; 55 ndigo6g12 device device; 56 errorCode = ndigo6g12_init(&device, &params, &errorMessage); 57 if (errorCode != CRONO OK) { printf("Could not init Ndigo6G-12: %s\n", errorMessage); 60 printf("Please change path to the .cronorom in ndigo6g12_example.cpp\n"); 61 exit(1); 62 } 63 // check if firmware now supports the chosen application type ndigo6g12 static info si; ndigo6g12 get static info(&device, &si); 67 if (si.application_type != appType) { 68 printf("The switch to appType did not work, please make sure that " 69 "the firmware file is provided"); 70 ``` ``` (continued from previous page) ndigo6g12 close(&device); exit(1); } if (appType == 0) { appType = si.application_type; switch (appType) { case 1: adcApp = new Ndigo6GAppSingle(tdcChannels); break: case 2: adcApp = new Ndigo6GAppDual(tdcChannels); break; case 4: adcApp = new Ndigo6GAppQuad(tdcChannels); case 5: adcApp = new Ndigo6GAppAverager(tdcChannels); break; default: printf("Not supported appType %d'\n", appType); ndigo6g12_close(&device); exit(1); printf("Running in %s\n%s\n", appTypeMap[appType].c_str(), requirementsMap[appType].c_str()); return device; int configure_ndigo6g12(ndigo6g12_device *device, int adcThreshold) { // prepare configuration ndigo6g12_configuration config; // fill configuration data structure with default values // so that the configuration is valid and only parameters // of interest have to be set explicitly if (CRONO_OK != ndigo6g12_get_default_configuration(device, &config)) { printf("Could not get default configuration: %s\n", ndigo6g12_get_last_error_message(device)); ndigo6g12 close(device); return 1; } // configuration for the TDC channels adcApp->ConfigureTDC(&config); ``` 71 72 73 78 79 ജവ 83 84 85 86 87 90 91 92 93 94 97 98 99 100 101 103 104 105 106 107 108 110 111 112 113 114 115 117 118 119 120 // configuration for the ADC channels adcApp->ConfigureADC(&config, adcThreshold); ``` 121 // write configuration to board 122 int error_code = ndigo6g12_configure(device, &config); 123 if (error_code != CRONO_OK) { 124 printf("Could not configure Ndigo6G-12: %s\n", ndigo6g12 get last error message(device)); 126 return 1; 127 } 128 ndigo6g12 get param info(device, &paramInfo); 129 adcApp->SetParamInfo(&paramInfo); 130 return 0; 133 // print some basic information about the Ndigo6G-12 device 134 void print device information(ndigo6g12 device *device) { 135 ndigo6g12_static_info si; 136 ndigo6g12_get_static_info(device, &si); 137 printf("Firmware revision %d.%d - Type %d\n", si.fw_revision, si.svn_revision, si.application_type); 139 printf("Firmware Bitstream Timestamp : %s\n", si.bitstream_date); 140 printf("Calibration date : %s\n", si.calibration date); 141 printf("Board serial : %d.%d\n", si.board_serial >> 24, 142 si.board_serial & Oxffffff); 143 printf("Board revision : %d\n", si.board revision); 144 : %d\n", si.board_configuration); printf("Board configuration 145 printf("Driver Revision : %d.%d.%d\n'', 146 ((si.driver_revision >> 16) & 255), 147 ((si.driver revision \gg 8) & 255), (si.driver revision & 255)); 148 printf("Driver Build Revision : %d\n", si.driver_build_revision); 149 150 ndigo6g12 fast info fi; 151 ndigo6g12_get_fast_info(device, &fi); printf("TDC temperature : %.2f C\n", fi.tdc1_temp); 153 : %.2f C\n", fi.ev12_temp); printf("ADC temperature 154 printf("FPGA temperature : %.2f C\n", fi.fpga_temperature); 155 : Gen. %d\n", fi.pcie_link_speed); printf("PCIe link speed 156 : %d lanes\n", fi.pcie_link_width); printf("PCIe link width 157 : %d bytes\n", fi.pcie max payload); printf("PCIe payload ndigo6g12 param info pi; 160 ndigo6g12_get_param_info(device, &pi); 161 printf("Sample rate : %.0f Msps\n", 162 pi.sample rate / 1000000.0); 163 printf("Resolution : %d Bit\n", pi.resolution); 164 printf("Sample period : %.2f ps\n", pi.sample_period); : %.2f ps\n", pi.tdc period); printf("TDC bin size printf("Packet Timestamp period : %.2f ps\n", pi.packet ts period); 167 printf("ADC Sample delay : %.2f ps\n", pi.adc_sample_delay); 168 169 170 ``` ``` int main(int argc, char *argv[]) { 171 if (argc < 2) { 172 printf("Usage: ndigo6g12_example <appType> [<tdcMask>]\n"); 173 for (auto atPair : appTypeMap) { 174 int at = atPair.first; printf("AppType %d: %s\n %s\n", at, appTypeMap[at].c_str(), requirementsMap[at].c_str()); 177 } 178 printf("tdcMask: Bit flag for TDC channels E-H\n"); 179 exit(1); 180 } 181 182 int appType = atoi(argv[1]); 183 int tdcChannelMask = 0; 184 if (argc > 2) { 185 tdcChannelMask = atoi(argv[2]); 186 187 // use 128 MiByte to buffer incoming data // largest ADC data packet has about 500 KiByte 189 const int64_t BUFFER SIZE = 128 * 1024 * 1024; 190 191 // use the first Ndigo6G-12 device found in the system 192 const int CARD_INDEX = 0; 193 194 // set board ID in all packets to 0 195 // can be used to distinguish packets of multiple devices 196 const int BOARD ID = 0; 197 198 printf("cronologic ndigo6g12_example using driver: %s\n", 199 ndigo6g12_get_driver_revision_str()); 200 201 // create and initialize the device // may fail if the device is already in use by another process 203 // or the device driver is not installed 204 ndigo6g12 device device = 205 initialize_ndigo6g12(BUFFER_SIZE, BOARD_ID, CARD_INDEX, appType, 206 tdcChannelMask); 207 208 print_device_information(&device); 210 // set the configuration required for capturing data 211 // the base line is shifted by +350mV, as the target is to trigger at the 212 // middle of the NIM pulse edge 213 int adcThreshold = 0; 214 int status = configure_ndigo6g12(&device, adcThreshold); 215 if (status != 0) { exit(1); 217 } 218 219 // configure readout behaviour 220 ``` ``` // automatically acknowledge all data as processed 221 // on the next call to ndigo6g12_read() 222 // old packet pointers are invalid after calling ndigo6g12_read() 223 ndigo6g12_read_in readConfig; readConfig.acknowledge_last_read = 1; // structure with packet pointers for read data 227 ndigo6g12_read_out readData; 228 229 // start data capture 230 status = ndigo6g12_start_capture(&device); if (status != CRONO OK) { printf("Could not start capturing: %s", 233 ndigo6g12_get_last_error_message(&device)); 234 ndigo6g12 close(&device); 235 exit(1); 236 } 237 // get current sample rate to calculate event timestamps 239 ndigo6g12 param info paramInfo; 240 ndigo6g12 get param info(&device, &paramInfo); 241 242 // ADC data is provided in packets, one packet per ADC channel and trigger 243 // TDC data is provided in a single packet for all TDC inputs in a certain 244 // timespan printf("\nReading packets:\n"); 246 247 const int MAX PACKET COUNT = 70; 248 int packetCount = 0; 249 bool noDataLastTime = false; 250 while ((packetCount < MAX PACKET COUNT)) {</pre> 251 // get pointers to acquired packets status = ndigo6g12_read(&device, &readConfig, &readData); 253 if (status != CRONO OK) { 254 if (!noDataLastTime) { 255 printf(" - No data! -\n"); 256 } 257 noDataLastTime = true; } else { 260 noDataLastTime = false; 261 // iterate over all packets received by the last read 262 volatile crono packet *p = readData.first packet; 263 while (p <= readData.last_packet) {</pre> if (p->channel < 4) { // packets with channel number < 4 are ADC data 267 double packet ts = 268 adcApp->ProcessADCPacket(const cast<crono packet *>(p)); 269 } else { 270 (continues on next page) ``` ``` // packets with channel number >= 4 are TDC data 271 adcApp->ProcessTDCPacket(const_cast<crono_packet *>(p)); 272 } 273 // go to next packet p = crono next packet(p); 277 packetCount++; 278 } // end: iterate over received packets 279 // end: Got any packets? 280 } // end: while 281 // shut down packet generation and DMA transfers 283 ndigo6g12_stop_capture(&device); 284 285 // deactivate Ndigo6G-12 286 ndigo6g12_close(&device); 287 return 0; 289 290 ``` ## ndigo6g12\_app.h ``` #pragma once #include "delay.h" #include "ndigo6g12_interface.h" #include <map> #include <string> #include <vector> // Base class for Ndigo6G applications 8 // contains common code for packet processing 9 class Ndigo6GApp { protected: const int PRECURSOR = 1; // contains the timing parameters of the current mode like sample period 13 ndigo6g12_param_info *pi; 14 int adcThreshold: 15 int tdcChannelMask; 16 // convenience method for adding the TDC channels to the channel map void AddTDCChannels(std::map<int, std::string> &channelMap) { for (int i = 0; i < 4; i++) { 19 if (tdcChannelMask & (1 << i)) {</pre> 20 channelMap[4 + i] = (char)'E' + (char)i; 21 } 22 } 23 } 24 25 public: 26 ``` (continues on next page) double feOffset = i; 27 28 29 32 33 34 35 36 39 40 41 42 46 47 48 49 50 51 52 53 54 55 56 57 59 60 61 62 63 66 67 68 69 71 74 75 76 ``` (continued from previous page) // linear interpolation of trigger threshold crossing 77 feOffset += (double)(adc_data[i] - adcThreshold) / 78 (adc_data[i] - adc_data[i + 1]); 79 // convert to picoseconds feOffset *= pi->sample_period; 82 // calculate timestamp of threshold crossing in picoseconds 83 double fallingEdgeTs = packetTs + feOffset; 84 // adjust for ADC pipeline delay 85 fallingEdgeTs -= pi->adc_sample_delay; 86 return fallingEdgeTs; } 89 90 return -1; 91 } 92 93 95 // maximum distance of two pulses, so that they are considered to be a cable _{f L} 96 static const double MAX_DELAY_PS = 500000.; 97 98 class Ndigo6GAppSingle : public Ndigo6GApp { 99 private: 100 // last falling edge to compute the difference to 101 double lastFallingEdgeTs = 0; 102 103 public: 104 Ndigo6GAppSingle(int tdcChannelMask) : Ndigo6GApp(tdcChannelMask) { 105 106 virtual void ConfigureADC(ndigo6g12_configuration *config, int adc_threshold); 108 virtual double ProcessADCPacket(crono packet *pkt); 109 virtual void ProcessTDCTimestamp(int tdcChannel, double timestamp) {} 110 }; 111 112 113 // Implementation of the different sample applications 115 class Ndigo6GAppDual : public Ndigo6GApp { 116 private: 117 DelayMeasurement delayMeasure; 118 119 public: Ndigo6GAppDual(int tdcChannelMask) : Ndigo6GApp(tdcChannelMask) { std::map<int, std::string> channelMap = {{0, "A"}, {3, "D"}}; 122 AddTDCChannels(channelMap); 123 delayMeasure.Init(0, MAX_DELAY_PS, channelMap); 124 } 125 (continues on next page) ``` ``` (continued from previous page) virtual void ConfigureADC(ndigo6g12_configuration *config, 126 int adc threshold); 127 128 virtual double ProcessADCPacket(crono_packet *pkt); virtual void ProcessTDCTimestamp(int tdcChannel, double timestamp); virtual void SetParamInfo(ndigo6g12_param_info *pi) { 132 Ndigo6GApp::SetParamInfo(pi); 133 // we have to wait for 3 TDC periods to make sure that the TDC data has 134 // arrived 135 delayMeasure.SetMaxWaitTime(pi->tdc_rollover_period * 3.5 * pi->tdc period); } 138 }; 139 140 141 private: 142 DelayMeasurement delayMeasure; public: 145 Ndigo6GAppQuad(int tdcChannelMask) : Ndigo6GApp(tdcChannelMask) { 146 147 std::map<int, std::string> channelMap = { 148 \{0, "A"\}, \{1, "B"\}, \{2, "C"\}, \{3, "D"\}\}; 149 AddTDCChannels(channelMap); 150 delayMeasure.Init(0, MAX_DELAY_PS, channelMap); 151 } 152 153 virtual void ConfigureADC(ndigo6g12_configuration *config, 154 int adc_threshold); 155 156 virtual void SetParamInfo(ndigo6g12 param info *pi) { Ndigo6GApp::SetParamInfo( pi); 158 // we have to wait for 3 TDC periods to make sure that the TDC data has 159 // arrived 160 delayMeasure.SetMaxWaitTime(pi->tdc rollover period * 3.5 * 161 pi->packet_ts_period); 162 } virtual double ProcessADCPacket(crono_packet *pkt); 165 166 virtual void ProcessTDCTimestamp(int tdcChannel, double timestamp); 167 }; 168 class Ndigo6GAppAverager : public Ndigo6GApp { 170 private: // last falling edge to compute the difference to 172 double lastFallingEdgeTs = 0; 173 174 public: 175 (continues on next page) ``` ``` Ndigo6GAppAverager(int tdcChannelMask) : Ndigo6GApp(tdcChannelMask) {} virtual void ConfigureADC(ndigo6g12_configuration *config, int adc_threshold); virtual double ProcessADCPacket(crono_packet *pkt); }; ``` ## 5.3 ndigo6g12\_adc\_single.cpp ``` #include "ndigo6g12 app.h" #include <stdio.h> /\!/ a simple application that measures the distance of two packets and computes // the frequency of the signal double Ndigo6GAppSingle::ProcessADCPacket(crono_packet *pkt) { double fallingEdgeTs = ComputeFallingEdge(pkt); 8 if (fallingEdgeTs > 0) { 10 if (lastFallingEdgeTs > 0) { 11 double packetRate = (1.0 / (fallingEdgeTs - lastFallingEdgeTs)); double packetRateKHz = packetRate * 1e9; printf("ADC packet rate: %.3f kHz\n", packetRateKHz); 14 } 15 lastFallingEdgeTs = fallingEdgeTs; 16 17 return fallingEdgeTs; 18 void Ndigo6GAppSingle::ConfigureADC(ndigo6g12_configuration *config, 21 int adcThreshold) { 22 this->adcThreshold = adcThreshold; 23 // single channel mode with 6.4 Gsps 24 config->adc_mode = NDIGO6G12_ADC_MODE_A; // ADC sample value range -32768 .. 32767 27 config->output mode = NDIGO6G12 OUTPUT MODE SIGNED16; 28 29 // enable ADC channel A and trigger on the falling edge of ADC data 30 // shift baseline of analog inputs to +350 mV 31 config->analog_offsets[0] = NDIGO6G12_DC_OFFSET_N_NIM * -1; // trigger on falling edge of ADC data 34 config->trigger[NDIGO6G12_TRIGGER_A0].edge = true; 35 config->trigger[NDIGO6G12_TRIGGER_AO].rising = false; 36 config->trigger[NDIGO6G12_TRIGGER_A0].threshold = adcThreshold; 37 38 // enable channel A 39 config->trigger_block[0].enabled = true; 40 // multiples of 32 ADC samples (5 ns recording time) ``` ``` config->trigger_block[0].length = 1; // multiples of 32 ADC samples, gets added to packet length config->trigger_block[0].precursor = PRECURSOR; // select ADC data as trigger source of the channel config->trigger_block[0].sources = NDIGO6G12_TRIGGER_SOURCE_AO; } ``` ## 5.4 ndigo6g12\_adc\_dual.cpp ``` #include "ndigo6g12 app.h" #include <stdio.h> #include <cmath> // an application that measures the delay between a start signal (A) and a // stop signal (D) double Ndigo6GAppDual::ProcessADCPacket(crono packet *pkt) { double falling_edge_ts = ComputeFallingEdge(pkt); 10 11 // gather data 12 if (falling_edge_ts > 0) { 13 delayMeasure.InsertTimestamp(pkt->channel, falling_edge_ts); Delays *delays = delayMeasure.MeasureDelays(); 17 delayMeasure.PrintDelays(delays); 18 19 return falling_edge_ts; 20 21 void Ndigo6GAppDual::ProcessTDCTimestamp(int tdcChannel, double timestamp) { 22 //TDC channels are mapped as 4-7 23 delayMeasure.InsertTimestamp(4 + tdcChannel, timestamp); 24 25 Delays *delays = delayMeasure.MeasureDelays(); 26 27 delayMeasure.PrintDelays(delays); 30 31 void Ndigo6GAppDual::ConfigureADC(ndigo6g12_configuration *config, 32 int adcThreshold) { 33 this->adcThreshold = adcThreshold; 34 // dual channel mode with 3.2 Gsps 35 config->adc_mode = NDIGO6G12_ADC_MODE_AD; 36 37 ``` ``` // ADC sample value range -32768 .. 32767 38 config->output_mode = NDIGO6G12_OUTPUT_MODE_SIGNED16; 39 40 // enable ADC channel A and trigger on the falling edge of ADC data // shift baseline of analog inputs to +350 mV // do the same for channel D config->analog offsets[0] = NDIGO6G12 DC OFFSET N NIM * -1; 44 config->analog_offsets[3] = NDIGO6G12_DC_OFFSET_N_NIM * -1; 45 46 // trigger on falling edge of ADC data 47 config->trigger[NDIGO6G12_TRIGGER_A0].edge = true; config->trigger[NDIGO6G12 TRIGGER A0].rising = false; config->trigger[NDIGO6G12 TRIGGER A0].threshold = adcThreshold; 50 config->trigger[NDIGO6G12_TRIGGER_DO].edge = true; 51 config->trigger[NDIGO6G12_TRIGGER_DO].rising = false; 52 config->trigger[NDIGO6G12 TRIGGER DO].threshold = adcThreshold; 53 54 // enable channel A config->trigger_block[0].enabled = true; 57 // in multiples of 16 ADC samples (5 ns recording time) 58 config->trigger_block[0].length = 1; 59 60 // in multiples of 16 ADC samples, gets added to packet length 61 config->trigger_block[0].precursor = PRECURSOR; 62 63 // select ADC data as trigger source of the channel 64 config->trigger block[0].sources = NDIGO6G12 TRIGGER SOURCE AO; 65 66 // enable channel D 67 config->trigger block[3].enabled = true; 68 // in multiples of 16 ADC samples (5 ns recording time) 70 config->trigger_block[3].length = 1; 71 72 // in multiples of 16 ADC samples, gets added to packet length 73 config->trigger_block[3].precursor = PRECURSOR; // select ADC data as trigger source of the channel config->trigger_block[3].sources = NDIGO6G12_TRIGGER_SOURCE_DO; 78 ``` # 5.5 ndigo6g12\_adc\_quad.cpp ``` #include "ndigo6g12_app.h" #include <stdio.h> #include <cmath> #include <array> ``` ``` 6 // an application that measures the delay between a start signal (A) \, // stop signals on channels B-D double Ndigo6GAppQuad::ProcessADCPacket(crono_packet *pkt) { double falling edge ts = ComputeFallingEdge(pkt); 11 12 // gather data 13 if (falling edge ts > 0) { 14 delayMeasure.InsertTimestamp(pkt->channel, falling_edge_ts); 15 Delays *delays = delayMeasure.MeasureDelays(); 18 19 delayMeasure.PrintDelays(delays); 20 21 return falling_edge_ts; 22 23 24 void Ndigo6GAppQuad::ProcessTDCTimestamp(int tdcChannel, double timestamp) { 25 // insert TDC as channel 4-7 26 delayMeasure.InsertTimestamp(4 + tdcChannel, timestamp); 27 28 Delays *delays = delayMeasure.MeasureDelays(); 29 30 delayMeasure.PrintDelays(delays); 31 32 33 void Ndigo6GAppQuad::ConfigureADC(ndigo6g12_configuration *config, 34 int adcThreshold) { 35 this->adcThreshold = adcThreshold; 36 // quad channel mode with 1.6 Gsps config->adc_mode = NDIGO6G12_ADC_MODE_ABCD; 38 39 // ADC sample value range -32768 .. 32767 40 config->output_mode = NDIGO6G12_OUTPUT_MODE_SIGNED16; 41 // trigger on falling edge of ADC data 42 for (int index : {NDIGO6G12_TRIGGER_AO, NDIGO6G12_TRIGGER_BO, NDIGO6G12_TRIGGER_CO, NDIGO6G12_TRIGGER_DO}) { config->trigger[index].edge = true; 45 config->trigger[index].rising = false; 46 config->trigger[index].threshold = adcThreshold; 47 } 48 // the sources of each channel (they should trigger on the input data // of the channel) std::array<int, 4> sources = { 52 NDIGO6G12_TRIGGER_SOURCE_AO, NDIGO6G12_TRIGGER_SOURCE_BO, 53 NDIGO6G12_TRIGGER_SOURCE_CO, NDIGO6G12_TRIGGER_SOURCE_DO}; 54 55 ``` ``` // enable ADC channels A-D and trigger on the falling edge of ADC data 56 // shift baseline of analog inputs to +350 mV 57 for (int c = 0; c < 4; c++) { 58 config->analog_offsets[c] = NDIGO6G12_DC_OFFSET_N_NIM * -1; // enable channel 61 config->trigger_block[c].enabled = true; 62 63 // in multiples of 8 ADC samples (5 ns recording time) after trigger 64 config->trigger_block[c].length = 1; 65 // in multiples of 8 ADC samples, gets added to packet length config->trigger block[c].precursor = PRECURSOR; 68 69 // select ADC data as trigger source of the channel 70 config->trigger block[c].sources = sources[c]; 71 } 72 73 ``` ## 5.6 ndigo6g12\_adc\_averager.cpp ``` #include <stdio.h> #include "ndigo6g12_app.h" const int AVERAGING_COUNT = 16; double Ndigo6GAppAverager::ProcessADCPacket(crono_packet* pkt) { // calculate packet timestamp in picoseconds 8 // not adjusted for ADC-data precursor double packet_ts = pkt->timestamp * pi->packet_ts_period; 10 11 printf("\nPacket timestamp: %.3f ns\n", (packet_ts / 1000.0)); // packet length is number of 64-bit words of data // the first two 64-bit packet data words are additional header 15 // information 16 uint32_t data offset = 2; 17 // only the first currently carries valid information 18 uint64_t averaging_header0 = *(pkt->data); // if bit is set, less than the requested number of iterations have been 21 // performed before writing the packet due to possible data overflow on 22 // the next iteration 23 bool stopped_due_to_overflow = (averaging_header0 >> 32) & 0x1; 24 25 // if bit is set, the averaged data contains saturated or overflowed 26 // samples does NOT indicate that the input signal has not exceeded the 27 // ADC range 28 ``` ``` (continued from previous page) bool averaging_overflow = (averaging_header0 >> 32) & 0x2; 29 30 // number of iterations; may be less than requested 31 int iterations_performed = (averaging_header0 & Oxfffffff); // 2 averaged ADC samples are stored in each 64-bit chunk of packet data uint32_t sample_count = ((pkt->length - data_offset) * 2); 35 36 // ADC data is a signed 32-bit integer 37 int32_t* adc_data = (int32_t*)(pkt->data + data_offset); 38 // find first falling edge in averaging data for (uint32_t i = 0; i < sample_count - 1; i++) {</pre> 41 if (adc_data[i] >= 0 && adc_data[i + 1] < 0) {</pre> 42 // calculate threshold crossing relative to start of 43 ⇔packet double fe_offset = i; // linear interpolation of trigger threshold crossing fe offset += (double)(adc data[i] - 0) / (adc data[i] - adc 47 data[i + 1]); // calculate timestamp of threshold crossing in 48 ⇔picoseconds fe offset *= pi->sample period; 49 50 printf("Falling edge event - offset to packet start: %. 51 43f ns n'', (fe offset / 1000.0)); 52 break; 53 return packet_ts; 57 58 void Ndigo6GAppAverager::ConfigureADC(ndigo6g12_configuration* config, 59 int 60 ⊶adcThreshold) { // adcThreshold not used here, 0 is used as threshold for the data config->adc_mode = NDIGO6G12_ADC_MODE_A; 63 // ADC sample value range -32768 .. 32767 64 // averaging data saturates at +/- 2^21 - 1 65 config->output_mode = NDIGO6G12_OUTPUT_MODE_SIGNED32; 66 // enable ADC channel A and trigger on the falling edge of TRG input // shift baseline of analog inputs to +350 mV config->analog offsets[0] = NDIGO6G12 DC OFFSET N NIM * -1; 70 71 // trigger on falling edge of TRG input 72 config->trigger[NDIGO6G12_TRIGGER_TRG].edge = true; 73 (continues on next page) ``` ``` config->trigger[NDIGO6G12_TRIGGER_TRG].rising = false; 74 75 // set trigger level on TRG input to -350 mV 76 config->tdc_trigger_offsets[4] = NDIGO6G12_DC_OFFSET_N_NIM; // enable channel config->trigger_block[0].enabled = true; 80 // multiples of 32 ADC samples (5 ns recording time) 81 config->trigger block[0].length = 32764; 82 83 // select TRG as trigger source of the channel config->trigger block[0].sources = NDIGO6G12 TRIGGER SOURCE TRG; 85 86 // configuration of the Averaging features 87 // number of events that are averaged/summed 88 config->average configuration.iterations = AVERAGING COUNT; 89 90 // saturate averaging data instead of overflow config->average_configuration.use_saturation = true; 93 // don't stop averaging if next iteration could lead to sample datau 94 ∽overflow config->average_configuration.stop_on_overflow = false; 95 96 ``` # 5.7 ndigo6g12\_tdc.cpp ``` #include <stdio.h> #include "ndigo6g12_app.h" 2 void Ndigo6GApp::ProcessTDCPacket(crono_packet* pkt) { // TDC packet timestamp relates to end of packet 5 // adjust for timespan covered double packetTs = (double)(pkt->timestamp - pi->tdc_packet_timestamp_offset); // calculate packet timestamp in picoseconds 10 packetTs *= pi->packet ts period; 11 12 // packet length is number of 64-bit words of data // 2 TDC events are stored in each 64-bit chunk of packet data uint32_t tdcEventCount = pkt->length * 2; 15 16 17 // event encoding: 18 // Bits 31 downto 8: event timestamp in TDC bins relative to packet 19 timestamp 20 // Bits 7 downto 4: event flags 21 // Bits 3 downto 0: channel number 22 ``` ``` (continued from previous page) uint32_t* tdcEventData = (uint32_t*)(pkt->data); // each TDC packet covers up to 3 coarse TDC periods // the end of one period is marked by an event on channel 15 // 15: internal marker: end of current TDC time frame uint32_t tdcChannel = tdcEventData[i] & Oxf; uint32_t flags = (tdcEventData[i] >> 4) & Oxf; uint32_t event_ts = tdcEventData[i] >> 8; // add accumulated rollovers since start of packet // calculate timestamp of TDC event in picoseconds double edgeTsPs = event_ts * pi->tdc_period; ProcessTDCTimestamp(tdcChannel, edgeTsPs); printf("TDC event on channel %d timestamp: packet_ "with shift \%.3f ns, edge \%.3f ns \n", tdcChannel, (double)(pkt->timestamp * pi->packet_ packetTs / 1000., edgeTsPs / 1000.); ``` ``` if (tdcChannel == 15) { rolloverEra += pi->tdc_rollover_period; } } void Ndigo6GApp::ConfigureTDC(ndigo6g12 configuration* config) { // enable TDC channels for (int i = 0; i < NDIGO6G12_TDC_CHANNEL_COUNT; i++) {</pre> // for NIM pulses: trigger at -350 mV config->tdc_trigger_offsets[i] = NDIGO6G12_DC_OFFSET_N_NIM; (continues on next page) ``` // dummy data, can be ignored 23 24 25 28 29 30 31 32 35 36 37 38 39 42 43 44 45 46 48 49 50 51 53 54 55 56 60 61 62 63 64 66 67 68 69 70 ⇔without " sts\_period) / 1000.0, } } uint32\_t rolloverEra = 0; // print all TDC timestamps of the packet // TDC channel number // 0 - 3: LEMO inputs // 24-bit timestamp // valid input channel? if (tdcChannel == 14) { // rollover marker event ts += rolloverEra; "shift %.3f ns, " edgeTsPs += packetTs; if (tdcChannel < 4) {</pre> // event flags for (uint32\_t i = 0; i < tdcEventCount; i++) {</pre> ``` // enable TDC channel config->tdc_configuration.channel[i].enable = (tdcChannelMask &_ (1 << i)) != 0; // enable falling edge trigger as input to trigger matrix for ⊶selected // TDC channel // only required if used as trigger source for Gating, TiGer // or ADC trigger blocks config->trigger[NDIGO6G12_TRIGGER_TDC0 + i].edge = true; config->trigger[NDIGO6G12 TRIGGER TDCO + i].rising = false; // threshold not applicable for TDC inputs // trigger threshold is set via tdc_trigger_offsets[i] config->trigger[NDIG06G12_TRIGGER_TDC0 + i].threshold = 0; } ``` #### delay.h 5.8 71 72 73 76 77 78 81 82 83 84 85 ``` #include <deque> #include <map> #include <string> #include <vector> #include <float.h> /\!/ this utility class manages the arrival of timestamps for // a number of channels and tries to group them to a start // signal on one channel and stop signals on the other channels 10 // delay status 11 enum DelayStatus { 12 NotEnoughData, // we do not know, if the following signal has already arrived StopsMissing, // some of the stops have arrived after a maximum wait time // start and all expected stops were processed correctly Complete, 15 StartMissing 16 }; 17 18 class ChannelInfo { 19 public: size_t index; 21 int channel; 22 std::string name; 23 // contains the timestamps of pulses 24 std::deque<double> timestamps; 25 bool early; 26 bool ok; 27 bool HasData() const { return timestamps.size() > 0; } 28 29 ``` (continued from previous page) 30 // per channel output of delay measurement 31 class ChannelDelay { 32 public: int channel; bool missing; 35 bool isStart; 36 std::string name; 37 double delayPs; 38 // number of events that were ignored because of missing start 39 int ignoredCount; }; 42 // output of delay measurement 43 class Delays { 44 public: 45 DelayStatus status; 46 std::vector<ChannelDelay> channelDelays; double startTimestamp; }; 49 50 // class for measurement of delays between given number of channels 51 class DelayMeasurement { 52 std::vector<ChannelInfo> channels; 53 // map from channel to the index in channels std::map<int, size\_t> channelIndexes; 55 56 Delays delays; 57 size\_t startIndex; 58 // maxDelay is the time that two timestamps are considered to be in the 59 // same group, e.g., the maximum delay for a simple cable delay time 60 double maxDelay; // maxWaitTime is the time to wait after a signal, to know that a following 62 // signal has been received; this allows deciding if a group is complete 63 double maxWaitTime; 64 65 public: 66 void Init(int startChannel, double maxDelay, std::map<int, std::string> channelMap) { channels.resize(channelMap.size()); 69 this->maxDelay = maxDelay; 70 maxWaitTime = 10 \* maxDelay; 71 72 delays.channelDelays.resize(channelMap.size()); 73 size\_t i = 0; 74 for (auto const &e : channelMap) { int channel = e.first; 76 std::string name = e.second; 77 (continues on next page) 78 79 channels[i].index = i; channels[i].channel = channel; ``` channels[i].name = name; 80 delays.channelDelays[i].channel = channel; 81 delays.channelDelays[i].name = name; 82 delays.channelDelays[i].isStart = startChannel == channel; channelIndexes[channel] = i; i++; 85 86 startIndex = channelIndexes[startChannel]; 87 } 88 89 void SetMaxWaitTime(double maxWaitTime) { this->maxWaitTime = maxWaitTime; 92 // write the current timestamp in ps to the structure 93 void InsertTimestamp(int channel, double timestamp) { 94 size_t index = channelIndexes[channel]; 95 channels[index].timestamps.push_back(timestamp); 96 } // the parameter is returned by pointer to avoid memory allocations 99 Delays *MeasureDelays() { 100 size_t maxSize = channels[0].timestamps.size(); 101 size_t earliestIndex = 0; 102 double earliestTimestamp = DBL MAX; 103 double latestTimestamp = 0; 104 delays.channelDelays.resize(channels.size()); 105 106 for (const ChannelInfo &ci : channels) { 107 maxSize = std::max(maxSize, ci.timestamps.size()); 108 if (ci.timestamps.size() > 0 && 109 ci.timestamps.front() < earliestTimestamp) {</pre> 110 earliestTimestamp = ci.timestamps.front(); earliestIndex = ci.index; 112 } 113 if (ci.timestamps.size() > 0 && 114 ci.timestamps.back() > latestTimestamp) { 115 latestTimestamp = ci.timestamps.back(); 116 117 delays.channelDelays[ci.index].missing = false; delays.channelDelays[ci.index].ignoredCount = 0; 119 } 120 121 // process the queues if enough data is expected 122 if (maxSize > 0 && latestTimestamp - earliestTimestamp > maxWaitTime) { 123 int channelsTooEarly = 0; 124 int channelsTooLate = 0; int channels0k = 0; 126 int channelsMissing = 0; 127 bool startPresent = channels[startIndex].HasData(); 128 double startTimestamp = startPresent 129 (continues on next page) ``` (continued from previous page) ? channels[startIndex].timestamps[0] 130 : latestTimestamp - 2 \* maxDelay; 131 132 for (ChannelInfo &ci : channels) { ci.early = false; ci.ok = false; if (ci.HasData()) { 136 double diffToStart = ci.timestamps[0] - startTimestamp; 137 delays.channelDelays[ci.index].delayPs = diffToStart; 138 if (diffToStart < -maxDelay) {</pre> 139 ci.early = true; channelsTooEarly++; } else if (diffToStart > maxDelay) { channelsTooLate++; 143 delays.channelDelays[ci.index].missing = true; 144 } else { 145 ci.ok = true; 146 channelsOk++; } 148 } else { 149 if (latestTimestamp > startTimestamp + maxWaitTime) { 150 // if there was data it should have arrived by now 151 delays.channelDelays[ci.index].missing = true; 152 channelsMissing++; 153 } 154 } 155 } 156 157 if (channelsOk + channelsTooLate + channelsMissing == 158 channels.size()) { 159 // best case, every stop and start is included; 160 // otherwise some channels are missing/too late for (ChannelInfo &ci : channels) { 162 **if** (ci.ok) { 163 ci.timestamps.pop\_front(); 164 165 } 166 delays.startTimestamp = startTimestamp; 167 delays.status = Complete; } else if (channelsTooEarly > 0 || !startPresent) { 169 // cut away 170 double cutOffTimestamp = startPresent 171 ? startTimestamp - maxDelay 172 : latestTimestamp - maxWaitTime; 173 174 bool removed = false; for (ChannelInfo &ci : channels) { 176 while (ci.timestamps.size() > 0 && 177 ci.timestamps[0] < cutOffTimestamp) {</pre> 178 ci.timestamps.pop\_front(); (continues on next page) 179 ``` delays.channelDelays[ci.index].ignoredCount++; 180 removed = true; 181 } 182 } 183 delays.startTimestamp = earliestTimestamp; delays.status = removed ? StartMissing : NotEnoughData; 186 } else { 187 delays.status = NotEnoughData; 188 189 } else { // else not enough data, process during next process packets delays.status = NotEnoughData; 192 193 return &delays; 194 } 195 196 // the parameter is passed by reference to avoid memory allocations void PrintDelays(Delays *delays) { 198 if (delays->status == NotEnoughData) { 199 return; 200 201 if (delays->status == Complete || delays->status == StopsMissing) { 202 for (const ChannelDelay &cd : delays->channelDelays) { 203 if (cd.isStart) { 204 printf("---\n%s: Start \%.3lf ns\n", cd.name.c_str(), 205 delays->startTimestamp / 1000.); 206 } else if (!cd.missing) { 207 printf("%s: Delay %.31f ns\n", cd.name.c_str(), 208 cd.delayPs / 1000.); 209 } else { 210 printf("%s: Missing\n", cd.name.c_str()); 212 } 213 214 (delays->status == StartMissing) { 215 216 printf("---\n Start missing at %.3lf ns\n", 217 delays->startTimestamp / 1000.); for (const ChannelDelay &cd : delays->channelDelays) { 219 if (cd.isStart) { 220 // ignore 221 } else if (cd.ignoredCount > 0) { 222 printf("%s: Ignored %d\n", cd.name.c_str(), cd.ignoredCount); 224 } } 226 } 227 } 228 229 ``` ## **Technical Data** · Input Passband: 1 MHz to 950 MHz · Power Requirements: **35 W** Mechanical Dimensions: 170 mm x 106 mm x 22 mm (fits in one PCle slot) · Throughput: **5200 MByte/s** on PCle x8 #### **Digitizer Characteristics** 6.1 Each board is tested against the values listed in the "Min" column. "Typical" is the mean value of the first 10 boards that were produced. ## 6.1.1 1-Channel-Mode (6.4 Gsps) | Symbol | Parameter | Min | Typical | Max | Units | |------------------------|-------------------------------------------------------------|-----|---------|-----|-------| | THD <sub>1</sub> | Total Harmonic Distortion | | -67 | -56 | dB | | SNR <sub>1</sub> | Signal-to-Noise Ratio | 53 | 54 | | dB | | SFDR <sub>incl,1</sub> | Spurious Free Dynamic Range (including Harmonics) | 58 | 75 | | dB | | SFDR <sub>excl,1</sub> | Spurious Free Dynamic Range (excluding Harmonics) | 71 | 75 | | dB | | SINAD <sub>1</sub> | Signal-to-Interference Ratio including Noise and Distortion | 49 | 54 | | dB | | ENOB <sub>1</sub> | Effective Number of Bits | 8.5 | 8.7 | | | ## 6.1.2 2-Channel-Mode (3.2 Gsps) | Symbol | Parameter | Min | Typical | Max | Units | |------------------------|-------------------------------------------------------------|-----|---------|-----|-------| | THD <sub>2</sub> | Total Harmonic Distortion | | -70 | -56 | dB | | SNR <sub>2</sub> | Signal-to-Noise Ratio | 54 | 54 | | dB | | SFDR <sub>incl,2</sub> | Spurious Free Dynamic Range (including Harmonics) | 58 | 75 | | dB | | SFDR <sub>excl,2</sub> | Spurious Free Dynamic Range (excluding Harmonics) | 71 | 77 | | dB | | SINAD <sub>2</sub> | Signal-to-Interference Ratio including Noise and Distortion | 49 | 54 | | dB | | ENOB <sub>2</sub> | Effective Number of Bits | 8.5 | 8.7 | | | ## 6.1.3 4-Channel-Mode (1.6 Gsps) | Symbol | Parameter | Min | Typical | Max | Units | |------------------------|-------------------------------------------------------------|-----|---------|-----|-------| | THD <sub>4</sub> | Total Harmonic Distortion | | -68 | -56 | dB | | SNR <sub>4</sub> | Signal-to-Noise Ratio | 53 | 55 | | dB | | SFDR <sub>incl,4</sub> | Spurious Free Dynamic Range (including Harmonics) | 58 | 74 | | dB | | SFDR <sub>excl,4</sub> | Spurious Free Dynamic Range (excluding Harmonics) | 71 | 75 | | dB | | SINAD <sub>4</sub> | Signal-to-Interference Ratio including Noise and Distortion | 49 | 54 | | dB | | ENOB <sub>4</sub> | Effective Number of Bits | 8.5 | 8.7 | | | ## 6.2 Oscillator Time Base | Symbol | Parameter | Min | Typical | Max | ppb | |-------------------|----------------------------------------|-----|---------|------|------| | ΔΤ | Temperature stability -20 °C to 70 °C¹ | | | 10 | ppb | | Fo | Initial calibration | | <300 | 500 | ppb | | $\Delta F/F_1$ | Aging first year | | | 100 | ppb | | $\Delta F/F_{20}$ | All inclusive aging 20 years | | | 1000 | ppb | | | Warm-up <sup>2</sup> | | | 3 | min. | $<sup>^1</sup>$ Over –40 °C to +85 °C; relative to stabilized frequency after 1 hour of continuous operation ## 6.3 Electrical Characteristics ## 6.3.1 Environmental Conditions for Operation The board is designed to be operated under the following conditions: | Symbol | Parameter | Min | Typical | Max | Units | |--------|------------------------------------------|-----|---------|-----|-------| | Т | ambient temperature | 5 | | 40 | °C | | RH | relative humidity at 31°C non condensing | 20 | | 75 | % | <sup>&</sup>lt;sup>2</sup>@+25 °C; within ±100 ppb of F, where F is the stabilized frequency reached after 1 hour of continuous operation ## 6.3.2 Environmental Conditions for Storage The board shall be stored between operation under the following conditions: | Symbol | Parameter | Min | Typical | Max | Units | |--------|------------------------------------------|-----|---------|-----|-------| | Т | ambient temperature | -30 | | 60 | °C | | RH | relative humidity at 31°C non condensing | 10 | | 70 | % | ## 6.3.3 Power Supply | Symbol | Parameter | Min | Typical | Max | Units | |--------------------|--------------------------------------------------|------|---------|------|-------| | I <sub>3.3</sub> | PCIe 3.3 V rail power consumption | | 0.42 | | W | | VCC <sub>3.3</sub> | PCIe 3.3 V rail power supply | 3.1 | 3.3 | 3.6 | V | | l <sub>12</sub> | PCIe 12 V rail power consumption <sup>1</sup> | | 31 | | W | | VCC <sub>12</sub> | PCIe 12 V rail power supply <sup>1</sup> | 11.1 | 12 | 12.9 | V | | l <sub>aux</sub> | PCIe 3.3 V <sub>aux</sub> rail power consumption | | 0 | | W | | VCC <sub>aux</sub> | PCIe 3.3 V <sub>aux</sub> rail power supply | | 3.3 | | V | <sup>&</sup>lt;sup>1</sup> The 12 V power is sourced solely from the PCle power connector located at the rear of the board. ## 6.3.4 Analog Inputs AC coupled single-ended analog inputs: | Symbol | Parameter | Min | Typical | Max | Units | |------------------|----------------------------|------|---------|-----|-------| | V <sub>p-p</sub> | Peak-to-peak input voltage | | | 1 | ٧ | | Z <sub>p</sub> | Input impedance | | 50 | | Ω | | $V_{\rm offs}$ | Adjustable offset | -0.5 | | 0.5 | V | ## 6.3.5 Digital Inputs AC coupled single-ended digital inputs: | Symbol | Parameter | Min | Typical | Max | Units | |--------------------|----------------------------|------|---------|-----|-------| | $V_{p-p}$ | Peak-to-peak input voltage | | | 1.3 | V | | $Z_p$ | Input impedance | | 50 | | Ω | | I <sub>Term</sub> | Termination Current | -50 | -20 | 50 | mA | | $V_{\rm offs}$ | Adjustable offset | -1.3 | | 1.3 | V | | t <sub>Pulse</sub> | Pulse length | 2 | 5 | 200 | ns | | t <sub>Rise:</sub> | Pulse Edge 20% to 80% | | | 10 | ns | | t <sub>Fall:</sub> | Pulse Edge 80% to 20% | | | 10 | ns | ## 6.3.6 Absolute Maximum Ratings The absolute ratings are the maximum amplitude that an input pulse can safely have before the board may be damaged. The maximum voltage of any input voltage may not exceed the values given by $V_{\text{max}}$ . The voltages relative to a constant DC offset (i.e., the pulse "height") may not exceed the values given by $V_{\text{AC,max}}.$ ### **Analog Inputs** | Symbol | Parameter | Min | Typical | Max | Units | |------------------|--------------------------------------------|------|---------|-----|-------| | V <sub>max</sub> | Maximum input voltage | -25 | | 25 | V | | $V_{AC,max}$ | Maximum pulse height relative to DC offset | -1.9 | | 2.0 | V | ## **Digital Inputs** | Symbol | Parameter | Min | Typical | Max | Units | |------------------|--------------------------------------------|-----|---------|-----|-------| | V <sub>max</sub> | Maximum input voltage | -16 | | 16 | V | | $V_{AC,max}$ | Maximum pulse height relative to DC offset | -5 | | 5 | V | # 6.4 Information Required by DIN EN 61010-1 ### 6.4.1 Manufacturer The Ndigo6G is a product of: cronologic GmbH & Co. KG Jahnstraße 49 60318 Frankfurt HRA 42869 beim Amtsgericht Frankfurt/M VAT-ID: DE235184378 ### 6.4.2 Intended Use and System Integration The devices are not ready to use as delivered by cronologic. It requires the development of specialized software to fulfill the application of the end user. The device is provided to system integrators to be built into measurement systems that are distributed to end users. These systems usually consist of a Ndigo6G, a main board, a case, application software and possible additional electronics to attach the system to some type of detector. They might also be integrated with the detector. The Ndigo6G is designed to comply with DIN EN 61326-1 when operated on a PCIe compliant main board housed in a properly shielded enclosure. When operated in a closed standard compliant PC enclosure the device does not pose any hazards as defined by EN 61010-1. Radiated emissions, noise immunity and safety highly depend on the quality of the enclosure. It is the responsibility of the system integrator to ensure that the assembled system is compliant to applicable standards of the country that the system is operated in, especially with regard to user safety and electromagnetic interference. Compliance was only tested for attached cables shorter than 3 m. When handling the board, adequate measures have to be taken to protect the circuits against electrostatic discharge (ESD). All power supplied to the system must be turned off before installing the board. ### 6.4.3 Environmental Conditions See Section 6.3.1 and Section 6.3.3. ### **6.4.4** Inputs All inputs are AC coupled. The inputs have very high input bandwidth requirements and therefore there are no circuits that provide overvoltage protection for these signals. ## **Danger** Applying high voltage on the inputs relative to the slot cover can result in permanent damage to the board. See Section 6.3.6 for the maximum ratings of the inputs. ### 6.4.5 Recycling cronologic is registered with the "Stiftung Elektro-Altgeräte Register" as a manufacturer of electronic systems with **Registration ID DE 77895909**. The Ndigo6G-12 belongs to category 9, "Überwachungs und Kontrollinstrumente für ausschließlich gewerbliche Nutzung". The last owner of an Ndigo6G-12 must recycle it, treat the board in compliance with §11 and §12 of the German ElektroG, or return it to the manufacturer's address listed in Section 6.4.1. ### 6.4.6 Export Control The Ndigo6G product line is a dual-use item under Council Regulation (EC) No 428/2009 of 5 May 2009 in section 3A002h. Similar regulations exist in many countries outside Europe. Regardless of the fact that we at cronologic exclude the use of our products for military purposes, the laws of the EU and many other countries restrict exports of dual-use items. Since we have to apply for a General Export Permit for these countries, delivery processes may be delayed or delivery to certain countries may become impossible. For the application of this export license we need the following documents from you: - Exporter declaration - · Company profile - · Import license (country dependent) There are countries for which a General Export License can be used for the export of dual-use goods. In this case we need the corresponding documents from you and there will be no further delay. Included countries are: - · Australia - Japan - · Canada - · Liechtenstein - · New Zealand - Norway - Switzerland - · Singapore - · USA Before re-exporting an Ndigo6G or any product containing an Ndigo6G as a component, please check you local regulations whether an export permit is required. It is not permitted to export an Ndigo6G to the Russian Federation or the Republic of Belarus. # **Revision History** #### 7.1 Firmware #### 1.25086 — 2025-04-03 Bugfix: Removed trigger dead time #### 1.24120 - 2024-04-30 Improved ADC/TDC synchronization Added sample averaging modes AA/DD, AAAA/DDDD, and AADD TiGer Updates Internal optimizations Bug fixes #### 5493 — 2023-10-30 Fixed bug related to level triggering Fixed first packet being empty Minor bug fixes #### 5467 — 2023-05-05 PCIe optimizations Minor bug fixes ### 7.2 Driver #### 2.2.0 — 2025-04-03 Bugfix: Removed trigger dead time Bugfix: Fixed NDIGO6G12 MAX PRECURSOR for Averaging Mode #### 2.0.1 - 2024-07-17 Extensive revision of the application programming interface Improved linux support Improved documentation Improved TDC and ADC synchronization ### 1.5.4 - 2024-07-13 Fixed 2 channel handling with trigger from opposite channel (trigger A on channel D) Fixed timestamp uncertainty in lower bits ### 1.5.3 - 2024-07-07 Dynamic reconfiguration with .cronorom support #### 1.4.5 - 2023-01-23 Crono kernel driver update to v1.4.2 Added support for revision 3 boards Minor bug fixes Support for 32-bit OS discontinued #### 1.4.0 - 2022-08-18 Added support for external 10 MHz reference on slot bracket #### 1.3.0 - 2022-05-25 Added support for Averager ### 7.3 User Guide #### 1.4.0 - 2025-06-12 Updated documentation of ndigo6g12\_configuration::output\_mode Updated ndigo6g12\_param\_info::adc\_sample\_delay docstring Documented possible baseline drift due to the AC coupling Updated various figures to also work in dark mode Improved Averager documentation Revised documentation of crono packet ## 1.3.0 - 2025-04-10 Documented firmware update procedure Documented TDC calibration procedure #### 1.2.0 - 2025-04-02 Documented minimal packet length Moved alert and device-state defines to corresponding documentation section Updated Erratum #### 1.1.0 - 2025-01-14 Added Section 6.3.6 ### 1.0.1 - 2024-10-22 Improved Figure 1.1 #### 1.0.0 - 2024-10-17 Added digitizer characteristics Added chapter on TiGer Added Erratum Fixed gating documentation Many corrections ## 0.2.1 - 2024-10-01 Corrections in Export Control #### 0.2.0 - 2024-10-01 Added gating examples **Updated Export Control** ### 0.1.4 - 2024-08-06 Added figures for the Trigger Matrix and Gating Blocks. #### 0.1.3 - 2024-08-01 Added documentation for clock connections Added link to current user guide example code Removed clutter from the APIs "ON THIS PAGE" sidebar Updated C++ example General improvements ### 0.1.2 — 2024-07-17 Renamed FPGA0/1 to TRG/GATE Restructured API documentation Expanded documentation on Packet Format ### 0.1.1 - 2024-07-16 Corrected values in introduction Improved phrasing throughout ### 0.1.0 - 2024-07-11 Initial release ## **Erratum** Up to firmware revision 1.24120, the retrigger feature of the *gating blocks* does not behave as intended. Instead of a gate being only extended by a retrigger event, the state of the gate is reset to inactive. Up to firmware revision 1.24120 and in 4-channel mode, $ndigo6g12\_single\_shot()$ only works properly if ndigo6g12\_trigger\_block::multi\_shot\_count == 1. Up to firmware revision 1.24120, a deadtime of up to 10 ns after a packet can occur.