



# Project Nheengatu: EPICS support for CompactRIO FPGA and LabVIEW-RT





### CompactRIO 9045

Has a Xilinx Kintex 7 FPGA and an atom processor running LinuxRT





### LabVIEW



National instruments general interface to configure the CRIO through VI development. VIs Can be implemented on the FPGA or LinuxRT (labviewRT).







### Objective

Input and output binary and analog data, fast trigger counting, fine control using EPICS and CRIO

### FPGA LabviewRT (runs in LinuxRT)







### Nheengatu architecture



- (1) LabviewRT : Developed extensions to open a shared memory and write/read from/to it FPGA : Use C API generator to generate addresses associated with variables
- (2) LibCrioLinux.so: abstraction layer to reads and writes to shared memory and FPGA addresses of the variables using a reference "name"
  - NiFpgaHelp.so: Contains all NI specific functions
- (3) Device support: EPICS library that contains 5 types of records: BI, BO, AI, AO, Scaler, Waveform, and contains functions to initialize CRIO from IOC
- (4) EPICS IOC : IOC (no custom code, just db templates) that has access to all device support types





## 1.1 Handling FPGA variables (1/6)

BIs: Concatenate and connect to U64 Indicator

BOs: As is (boolean indicator)

### Binary output: as is



### Binary input: Convert to 64-bit U64







## 1.1 Handling FPGA variables (2/6)

Al and AO handling: Convert to fixed-point or single precision floating point

Single precision floating point handling

**Analog input: Convert to float** 



**Analog output: Convert to FXP** 



**Fixed-point handling** 

Analog input: use cast to U64 polymorphic VI



Analog output: use cast to fixed-point polymorphic VI



15% if this optimization is applied



### 1.1 FPGA variables (3/6)







## 1.1 Handling FPGA variables (4/6)

**Reading arrays: instantiate indicator array** 

- 108 | 116, | 132, | 164
- U08, U16, U32, U64
- Single precision floating point





### 1.1 FPGA variables (5/6)

#### Generate bitstream and C API files

#### fpga target libs & examples examples Open scaler\_digitial\_64 Explore... scaler\_digital\_2c scaler\_analog\_12 Show in Files View Ctrl+E scaler analog 2d Print... scaler\_analog\_di fpga\_all\_exampl scaler.lvlib 40 MHz Onboard Cloc Find IP Builder Save Mod3 (Slot 3, NI 9403) Save As... Mod1 (Slot 1, NI 9401) Mod5 (Slot 5, NI 9263) Mod8 (Slot 8, NI 9215) Mod2 (Slot 2, NI 940 Launch C API Generator...

### **Generates header file with addresses**

```
NiFpga fpga all example IndicatorU32 Looptime = 0x18034,
     NiFpga fpga all example IndicatorU32;
38
39
40
      typedef enum
41
    ⊟{
42
         NiFpga fpga all example IndicatorU64 BI = 0x180A0,
     NiFpga fpga all example IndicatorU64;
43
44
45
      typedef enum
46
    ⊟ {
47
         NiFpga fpga all example IndicatorSgl Mod4AI0 = 0x180B0,
48
         NiFpga fpga all example IndicatorSgl Mod4AI1 = 0x180AC,
49
         NiFpga fpga all example IndicatorSgl Mod4AI2 = 0x180A8,
         NiFpga fpga all example IndicatorSgl Mod4AI3 = 0x180A4,
51
         NiFpga fpga all example IndicatorSgl Mod6TC0 = 0x180D0,
         NiFpga fpga all example IndicatorSgl Mod6TC1 = 0x180CC,
53
         NiFpga fpga all example IndicatorSgl Mod6TC2 = 0x180C8,
         NiFpga fpga all example IndicatorSgl Mod6TC3 = 0x18
```

The address of each variable must be noted for the cfg.ini file (or automation tools can be used).





### 1.1 FPGA variables (6/6)



Make sure that non of your indicators and controls are omitted by abiding with the data types defined above!





### 1.2 LabviewRT variables

Four Vis were developed to synchronize variable/array exchange with EPICS

- IOC Shared memory initialize.vi
  - Initializes shared memory. All input can be left as default
- IOC Shared memory de-initialize.vi
  - de-initializes shared memory.
- SM read and increment.vi (polymorphic VI)
  - Reads the variable (from EPICS) when setting the polymorphic VI
- SM write and increment.vi (polymorphic VI)
  - Writes the variable (to EPICS) when setting the polymorphic VI

All VI MUST be chained with <ptr in> and <ptr out>.

The index of the variable must be noted for the cfg.ini file.











# 2 & 3 libCrioLinux.so and EPICS device support

# NO alteration or re-compile is required!





### 4.1 EPICS IOC (1/12)

# As long as the associated DB files are used, the IOC does not need to be compiled either!





### 4.2 EPICS IOC - CFG.INI (2/12)

```
Destination Crio IP: The IP address of the target CRIO
                         For safety, our intention is to keep this
                         IP as the loopback address (127.0.0.1)
  - Path: is the path to the bitfile that will be used to configure
          the FPGA of the target CRIO.
 - Bitfile Name: Is the name of the bitfile
; - Signature: Is the signature of that specific bitfile
 - Use Shared Memory: Set to 1 if labviewRT will open a shared memory
 - Shared Memory Path: If Use Shared Memory is set to 1, then this path
                        will be used.
[Settings]
Shared Memory Path=/labview_linux_sm
Path=/home/ABTLUS/dawood.alnajjar/work/crio-linux-libs/bitfiles
Signature=071ABA139A0C89D5C7E4051E2DB7F220
Bitfile Name=NiFpga_waveform.lvbitx
Destination Crio IP=127.0.0.1
Use Shared Memory=1
Shared Memory Size=4096
```

If changed, must be changed in labview RT VI too!

Disabling shared memory disables all labview RT variable processing.





### 4.2 EPICS IOC - CFG.INI (3/12)

2 types of variables: labview RT variables, and FPGA variables

```
56 Mod5/A00=180B4
57 RT_DBL_A01=1
```

The library distinguishes the RT variables type and its size through its name

```
RT Variables:
   The keyword RT is reserved for variables that are defined
   in labview RT. Do not use this reserved word in your names
   unless it is an RT variable, otherwise it will be ignored!
   In case of AI, AO, BI, BO, WF, Keywords for realtime double, single,
   Signed 8, 16, 32, 64 and unSigned 8, 16, 32, 64 are defined as follows
                   : RT_DBL_<NAME>
    Double
   Single
                   : RT_SGL_<NAME>
   UnSigned 64 bit : RT_U64_<NAME>
   UnSigned 32 bit : RT_U32_<NAME>
   UnSigned 16 bit : RT_U16_<NAME>
   UnSigned 08 bit : RT_U08_<NAME>
   Signed 64 bit : RT I64 < NAME>
   Signed 32 bit : RT_I32_<NAME>
   Signed 16 bit : RT I16 <NAME>
   Signed 08 bit : RT_I08_<NAME>
```





### 4.2 EPICS IOC - CFG.INI (4/12)

- [BIAddresses]
  - Address/index of BI
- [BI0]: index -name relation of each bit in the 64 bits (FPGA)
- [AO]: Address/index of each available output analog variable
- [AI]: Address/index of each available input analog variable
- [BO]: Address/index of each available output digital variable

```
[BIAddresses]
    BI0=180A0
    RT_BOL_BITest=21
47
     ; This has the bit mapping of BIO.
     [BI0]
    0=Mod3/DI00
    1=Mod3/DI01
50
51
52
53
     ; This has the address of each AO peripheral.
     : Category must have name AO.
     [AO]
    Mod5/A00=180B4
    RT_DBL_A01=1
     ; This has the address of each AI peripheral. TCs are also
     ; Considered as AI. Category must have name AI.
     [AI]
61
    Mod4/AI0=180B0
62
    RT_DBL_AI0=2
     ; This has the address of each BO peripheral.
66
      Category must have name BO.
67
     [BO]
    Mod1/DI00=18092
    RT_BOL_BO0=0
70
```





### 4.2 EPICS IOC - CFG.INI (5/12)

- [WAVEFORM]
  - List pf waveforms (RT & FPGA)
- [WAVEFORMXX]:
  - details of WAVFFORMXX
- [FXP\_XX]:
  - Details of FXP\_XX

```
This has the waveform list (input arrays)
[WAVEFORMS]
RT_SGL_WF0=
waveform_sgl2=
; This has the details of the RT waveform
[RT_SGL_WF0]
Size=5
Address=0
Type=SGL
; This has the details of the FPGA waveform
[waveform_sgl2]
Size=3
Address=1811C
Type=SGL
; This is the address of each AI peripheral. Note that it
 contains fixedpoint (keyword: start with FXP)
[AI]
FXP_aifxp4=18154
aiMod4AI2=18038
; This has the details of the Fixedpoint (only FPGA)
[FXP_aifxp4]
Sign=1
Word Length=64
Integer Word Length=0
```





### 4.2 EPICS IOC - CFG.INI (6/12)

```
[SCALERS]
73
     SCALER_DIGITAL=0
     SCALER_ANALOG=1
75
76
     [SCALER_ANALOG]
78
     Enable=1800E
79
     Gate=18006
     OneShot=1800A
81
     Counters=18010
82
     Preset Values=18000
83
     Number of Counters=2
84
     Done=18016
     [SCALER_DIGITAL]
87
     Enable=1801E
     Gate=18022
90
     OneShot=1801A
     Counters=18028
91
     Preset Values=1802C
92
     Number of Counters=2
93
94
     Done=18026
```





## 4.2 EPICS IOC – Settings (7/12)

Introduce path of deviceSupportCrio to IOC \$(TOP)/configure/RELEASE

```
TEMPLATE_TOP=$(EPICS_BASE)/templates/makeBaseApp/top
ASYN = /usr/local/epics-nfs/modules/R3.15.6/synApps/R6.0/support/asyn-R4-33
STD = /usr/local/epics-nfs/modules/R3.15.6/synApps/R6.0/support/std-R3-5
devSupCRIO=/usr/local/epics-nfs/modules/R3.15.6/crio-dev-sup/2019_06_11_01
```

Introduce deviceSupportCrio library name to IOC \$(TOP)/CRIOApp/src/Makefile

```
# Add all the support libraries needed by this IOC
CRIO_LIBS += std
CRIO_LIBS += asyn
CRIO_LIBS += devSupCRIO
```





### 4.2 EPICS IOC - crio-ioc.cmd (8/12)

```
1 #!/home/ABTLUS/SOL/crio-ioc/bin/linux-x86_64/CRIO
 3 epicsEnvSet("TOP","/usr/local/epics/apps/crio-ioc")
 4 epicsEnvSet("EPICS_BASE","/usr/local/epics-nfs/base/R3.15.6")
 5 epicsEnvSet("IOC","iocCRIO")
 6 epicsEnvSet("CONFIG","/usr/local/epics/apps/config/crio-ioc")
 7 epicsEnvSet("AUTOSAVE","/opt/autosave")
 8 cd ${TOP}
10 dbLoadDatabase "dbd/CRIO.dbd"
11 CRIO_registerRecordDeviceDriver pdbbase
12 set_requestfile_path($(CONFIG))
13 set_savefile_path($(AUTOSAVE))
14 set_pass1_restoreFile("crioioc.sav", "")
16 crioSupSetup("${CONFIG}/cfg.ini" , 1)
17
18 cd ${TOP}/iocBoot/${IOC}
20 dbLoadTemplate "${CONFIG}/bi.db.sub"
21 dbLoadTemplate "${CONFIG}/bo.db.sub"
22 dbLoadTemplate "${CONFIG}/ai.db.sub"
23 dbLoadTemplate "${CONFIG}/ao.db.sub"
24 dbLoadTemplate "${CONFIG}/scaler.db.sub"
25 dbLoadTemplate "${CONFIG}/waveform.db.sub"
26 iocInit
28 < "$(CONFIG)/init-pv.cmd"
30 create_monitor_set("crioioc.req", 1, "")
32 dbl
```





### 4.2 EPICS IOC – templates (9/12)

```
file "$(TOP)/db/devAICRIO.db.template"

{
   pattern

   {BL, EQ, DTYP, PIN, DESC}

   {"SOL", "CRIO:9215A:AIO", "CrioAI", "Mod4/AIO", "This is a Description of AIO"}

   {"SOL", "CRIO:9215A:AIO", "CrioAI", "Mod4/AII", "This is a Description of AII"}

}
```

AI db substitutions

This name must correspond to the name in the cfg.ini file

```
1 record(ai, "$(BL):$(EQ)) {
2    field(INP, "@$(PIN)")
3    field(DTYP, "$(DTYP)")
4    field(SCAN, ".1 second")
5    field(DESC, "$(DESC)")
6 }
7
```

```
Al db template
```

```
[AI]
Mod4/AI0=180B0
Mod4/AI1=180AC

; This has the bit mapping of BI0.
[BI0]
0=Mod3/DI00
1=Mod3/DI01
```





RT\_B0L\_WF6 6 RT U64 WF7 9

### 4.2 Automatic generation (10/12)

| } | BI INI NAME   | BI DB NAME | BI DESCRIPTION               |
|---|---------------|------------|------------------------------|
| Ļ | RT_BOL_BITest | RT:BI0     | This is a Description of BI0 |
| , | DIO           | 9403A:BI0  | This is a Description of BI1 |
| j | DI1           | 9403A:BI1  | This is a Description of BI2 |
| , | DI2           | 9403A:BI2  | This is a Description of BI3 |
| 3 | DI3           | 9403A:BI3  | This is a Description of BI4 |
| ) | DI4           | 9403A:BI4  | This is a Description of BI5 |
| ) | DI5           | 9403A:BI5  | This is a Description of BI6 |
|   | DI6           | 9403A:BI6  | This is a Description of BI7 |
| 2 | DI7           | 9403A:BI7  | This is a Description of BI8 |
|   |               |            |                              |

List of all variables (Needs to be filled by user)



IOC necessary files



Ready to go
Cfg.ini,
Substitutions,
autosave &
initial values!

```
[Settings]
Use Shared Memory=1
Path=/usr/local/epics/apps/config/crio-ioc/
Shared Memory Path=/labview_linux_sm
Destination Crio IP=127.0.0.1
Bitfile Name=NiFpga_fpga_all_example.lvbitx
Signature=B8DB11918E46DEE17EF2D2F610AEB859
Shared Memory Size=4096

[BIAddresses]
BI0=18040
RT_B0L_BITest=34

[BI0]
0=DI0
1=DI1
2=DI2
```

Nheengatu necessary files





### 4.2 EPICS IOC – templates (11/12)

For help:./gen\_cfg.py -h





## 4.2 Generated folder (12/12)

Generate **reference** folder that has all files necessary to reproduce that specific setup



### Big picture







# NFS setup-bl folder



```
[dawood.alnajjar@nfs-epics SOL] tree CRIO4/
CRI04/
— epics
        apps
            config
                crio-ioc
                    ai.db.sub
                    ao.db.sub
                    bi.db.sub
                    bo.db.sub
                    cfg.ini
                    crioioc.req
                    init-pv.cmd
                    NiFpga fpga all example.lvbitx
                    reference
                        cfq.csv
                        command.sh
                        reference.h
                        RT.list
                    scaler.db.sub
                    waveform.db.sub
                crio-ioc.cmd
          — crio-ioc -> /usr/local/epics-nfs/apps/crio-ioc/2019 06 25 01/
        base -> /usr/local/epics-nfs/base/R3.15.6/
```





### IOC compilation needed?

 Docker compilation environment\* that compiles using a processor of another host and using the CRIO SSD for testing on the CRIO







### Softwares used

- EPICS 3.15
- Synapps 6.0 (Scaler, Autosave)
- Labview 2018
- 2018.5 linuxRT firmware
- Compact RIO 9035/9045





### Exception handling

- Errors in the cfg.ini file, templates, or any inconsistency found appear on the EPICS IOC command prompt
  - e.g. Error on read [LibCrioLinux] Property [RT\_DBL\_AI0]: Query returned null.
- Non-critical exceptions are redirected to epics terminal, and the IOC does not stop functioning, so check your messages!





### **Known** limitations

- Binary inputs are limited to 64 bits
- Moving U64, I64 (64-bits) also is lossy since these variables are converted to double in EPICS, and double is 52 bits precision
- Number of indicators and controls on FPGA is limited due to available resources





# Nheengatu - Sharepoint Nheengatu - git repository