# Industrial IoT Baremetal Framework Developer Guide



# Contents

| Chapter 1 Introduction                       | . 3 |
|----------------------------------------------|-----|
| 1.1 QorlQ layerscape processor               |     |
| 1.2 Baremetal framework for QorlQ layerscape | 3   |
| 1.3 Supported processors and boards          | 4   |

| Chapter 2 Getting started                                 | 5  |
|-----------------------------------------------------------|----|
| 2.1 Hardware and software requirements                    |    |
| 2.2 Hardware setup                                        | 5  |
| 2.2.1 LS1021A-IoT board                                   | 5  |
| 2.2.2 LS1028ARDB, LS1043ARDB, or LS1046ARDB               |    |
| 2.2.3 i.mx6q-sabresd board                                |    |
| 2.3 Building the baremetal images from U-Boot source code | 7  |
| 2.3.1 Building U-Boot binary for the master core          | 7  |
| 2.3.2 Building baremetal binary for slave cores           | 8  |
| 2.4 Building the image through OpenIL                     | 9  |
| 2.4.1 Getting OpenIL                                      |    |
| 2.4.2 Building the baremetal images                       | 10 |
| 2.4.3 Booting up the Linux with Baremetal                 |    |

| Cha | pter 3 Running examples           | 14   |
|-----|-----------------------------------|------|
|     | 3.1 Preparing the console         |      |
|     | 3.2 Running the bare metal binary | . 14 |

| pter 4 Development based on baremetal framework |    |
|-------------------------------------------------|----|
| 4.1 Developing the baremetal application        |    |
| 4.2 Example software                            | 16 |
| 4.2.1 Main file app.c                           |    |
| 4.2.2 Common header files                       |    |
| 4.2.3 GPIO file                                 | 17 |
| 4.2.4 I2C file                                  |    |
| 4.2.5 IRQ file                                  |    |
| 4.2.6 QSPI file                                 |    |
| 4.2.7 Ethernet                                  |    |
| 4.2.8 USB file                                  | 21 |
| 4.2.9 PCIe file                                 |    |
| 4.2.10 CAN file                                 |    |
| 4.2.11 ENETC file                               | 24 |
| 4.3 ICC module                                  | 24 |
| 4.3.1 ICC examples                              | 27 |
| 4.4 Hardware resource allocation                |    |
| 4.4.1 LS1021A-IoT board                         |    |
| 4.4.2 LS1043ARDB or LS1046ARDB board            |    |
| 4.4.3 LS1028ARDB board                          |    |

| Chapter 5 Revision histo | y37 |
|--------------------------|-----|
|--------------------------|-----|

# Chapter 1 Introduction

This document provides an overview of the OpenIL Baremetal framework, the features it supports, and getting started with baremetal framework using the supported NXP Layerscape platforms. It also describes how to run a sample baremetal framework on the host environment and develop customer specific applications based on QorIQ Layerscape baremetal framework.

# 1.1 QorlQ layerscape processor

QorIQ layerscape processors are based on Arm® technology.

With the addition of NXP's next-generation QorIQ Layerscape series processors built on Arm core technology, the processor portfolio extends performance to the smallest form factor—from power-constrained networking and industrial applications to new virtualized networks and embedded systems requiring an advanced datapath and network peripheral interfaces.

# 1.2 Baremetal framework for QorlQ layerscape

The baremetal framework targets to support the scenarios that need low latency, real-time response, and high-performance. There is no OS running on the cores and customer specific application runs on that directly. The figure below depicts the baremetal framework architecture.



Figure 1. Baremetal framework architecture

The main features of the baremetal framework are as follows:

- Core0 runs as master which runs the operating system such as Linux, Vxworks.
- Slave cores run on the baremetal application.
- Easy assignment of different IP blocks to different cores.
- Interrupts between different cores and high-performance mechanism for data transfer.
- Different UART for core0 and slave cores for easy debug.

#### Industrial IoT Baremetal Framework Developer Guide, Rev. 1.6, 08/2019

Introduction

• Communication via shared memory.

The master core0 runs the U-Boot, it then loads the baremetal application to the slave cores and starts the baremetal application. The following figure depicts the boot flow diagram:



Figure 2. Baremetal framework boot flow diagram

## **1.3 Supported processors and boards**

The table below lists the industrial IoT features supported by various NXP processors and boards.

| Processor | Board          | Main features supported                                                              |
|-----------|----------------|--------------------------------------------------------------------------------------|
| LS1021A   | LS1021A-IoT    | GPIO, IRQ, IPI, data transfer, IFC, I2C, UART, QSPI, USB, PCIE, Flexcan              |
| LS1043A   | LS1043ARDB     | GPIO, IRQ, GPIO, IPI, data transfer, Ethernet, IFC, I2C, UART, FMan, USB, PCIe       |
| LS1046A   | LS1046ARDB     | GPIO, IRQ, GPIO, IPI, data transfer, Ethernet, IFC, I2C, UART, FMan, QSPI, USB, PCIe |
| LS1028A   | LS1028ARDB     | I2C, UART, ENETC, IPI, data transfer                                                 |
| i.mx6q    | i.mx6q-sabresd | UART                                                                                 |

| Table 1 | . Industrial lo | T features | supported by | NXP processors |
|---------|-----------------|------------|--------------|----------------|
|---------|-----------------|------------|--------------|----------------|

# Chapter 2 Getting started

This section describes how to set up the environment and run the baremetal examples on slave cores (assuming that the core0 is the master core and the other cores are the slave cores).

# 2.1 Hardware and software requirements

The following are required for running baremetal framework scenarios:

- Hardware: LS1021A-IoT, LS1043ARDB, LS1046ARDB, LS1028ARDB, i.mx6q-sabresd, or other QorIQ Layerscape boards and serial cables
- Software: OpenIL v1.4 release or later

# 2.2 Hardware setup

This section describes the hardware setup required for the NXP boards for running the baremetal framework examples.

# 2.2.1 LS1021A-IoT board

Two serial cables are needed. One is used for core0, which connects to USB0/K22 port for UART0, and another one is used for slave cores, which connect J8 and J17 together for UART1. The table below describes the pins for UART1.

| Pin name | Function   |
|----------|------------|
| J8 pin7  | Ground     |
| J17 pin1 | Uart1_SIN  |
| J17 pin2 | Uart1_SOUT |

#### Table 2. UART pins

The figure below depicts the UART1 hardware connections.



Figure 3. UART1 hardware connection

In order to test GPIO, connect the two pins, GPIO24 and GPIO25 together, which are through J502.3 and J502.5.



Figure 4. GPIO24 and GPIO25 connection on LS1021A-IoT

## 2.2.2 LS1028ARDB, LS1043ARDB, or LS1046ARDB

In case, either the LS1028ARDB, LS1043ARDB, or LS1046ARDB is used for developing the OpenIL Baremetal framework, two serial cables are needed. One is used for core0, which connects to UART1 port, and the other one is used for slave cores, which connects to the UART2 port.

# 2.2.3 i.mx6q-sabresd board

Two serial cables are needed. One is used for core0, which connects to USB port for UART1, and another one is used for slave cores, which connect J8 and U19 together for UART3. The table below describes the pins for UART3.

| Table 3. | Pins | for | UA | RT3 |
|----------|------|-----|----|-----|
|----------|------|-----|----|-----|

| Pin       | Description |
|-----------|-------------|
| U19 pin36 | Ground      |
| U19 pin30 | UART3_RXD   |
| U19 pin31 | UART3_TXD   |

The figure below depicts the UART3 hardware connections.



Figure 5. UART3 hardware connections

# 2.3 Building the baremetal images from U-Boot source code

There are two methods to build the baremetal images. One method is to compile the images in a standalone way, which is described in this section. Another method is to build the bare metal images using OpenIL framework, which is described in Building the image through OpenIL.

# 2.3.1 Building U-Boot binary for the master core

Perform the steps mentioned below:

1. Download U-Boot source from the path below:

https://github.com/openil/u-boot.git.

#### Getting started

- 2. Check it out to the tag OpenIL-u-boot-201908.
- 3. Configure cross-toolchain on your host environment.
- 4. Then, build the U-Boot and flash it into the SD card as SD boot for core0 using the commands below:

```
/* building u-boot image for ls1021aiot board */
  $make ls1021aiot_sdcard_baremetal_defconfig
  $make
```

#### NOTE

The default setting enables CAN3/4 feature for baremetal. If you want to enable the GPIO baremetal feature, you need to run the following commands:

```
$make ls1021aiot_sdcard_baremetal_defconfig
$make menuconfig
ARM architecture --->
Baremetal feature select (GPI03_15/27 supported) --->
$make
```

5. For the LS1028ARDB, LS1043ARDB, or LS1046ARDB, use the following commands:

```
/* building u-boot image for ls1028ardb board */
   $make ls1028ardb_sdcard_baremetal_defconfig
   $make
/* building u-boot image for ls1043ardb board */
   $make ls1043ardb sdcard baremetal defconfig
```

```
/* building u-boot image for ls1046ardb board */
  $make ls1046ardb_sdcard_baremetal_defconfig
  $make
```

6. For the i.mx6q-sabresd, use the following commands:

```
/* building u-boot image for i.mx6q-sabresd board */
  $make imx6q-sabresd_baremetal_defconfig
  $make
```

## 2.3.2 Building baremetal binary for slave cores

Perform the steps mentioned below:

\$make

1. Download the project source from the following path:

https://github.com/openil/u-boot.git.

- 2. Check it out to the tag OpenIL-Baremetal-201908.
- 3. Configure cross-toolchain on your host environment.
- 4. Then run the following commands:

\$git checkout baremetal

```
/* build baremetal image for ls1021-iot board */
$make ls1021aiot_baremetal_defconfig
$make
/* build baremetal image for ls1028ardb board */
$make ls1028ardb_sdcard_baremetal_defconfig
$make
/* build baremetal image for ls1043ardb board */
$make ls1043ardb_defconfig
$make
/* build baremetal image for ls1046ardb board * /
$make ls1046ardb_baremetal_defconfig
$make
/* build baremetal image for i.mx6q-sabresd board * /
$make mx6sabresd_baremetal_defconfig
$make
```

5. Finally, the file, U-Boot.bin used for bare metal is generated. Copy it to the the the server directory.

# 2.4 Building the image through OpenIL

There are two methods to build the baremetal images. One method is to compile the images in a standalone way which is described in Building the baremetal images from U-Boot source code. The second method is to build the bare metal images using OpenIL framework, which is described in this section.

The OpenIL project (Open Industry Linux) is designed for embedded industrial usage. It is an integrated Linux distribution for industry. With the OpenIL v1.1 or later releases, the Baremetal can be built and implemented conveniently.

Currently, there are two LS1021A-IoT baremetal defconfig files, a LS1028ARDB baremetal defconfig file, a LS1043ardb baremetal defconfig file, as listed below:

- configs/nxp ls1021aiot baremetal defconfig
- configs/nxp ls1021aiot baremetal ubuntu defconfig
- configs/nxp\_ls1028ardb\_baremetal-64b\_defconfig
- configs/nxp\_ls1043ardb\_baremetal-64b\_defconfig
- configs/nxp\_ls1046ardb\_baremetal-64b\_defconfig
- configs/imx6q-sabresd baremetal defconfig

## 2.4.1 Getting OpenIL

OpenIL v1.6 release is available at the following URL:

https://github.com/openil/openil.git.

To follow development, make a clone of the Git repository using the command below:

```
$ git clone https://github.com/openil/openil.git
$ cd openil
# checkout to the tag OpenIL-201908
```

```
$ git checkout -b OpenIL-201908 OpenIL-201908
```

NOTE

- Build everything as a normal user. Root user permissions are not required for configuring and using OpenIL. By running all commands as a regular user, you protect your system against packages behaving badly during compile and installation.
- Do not use make -jN to build OpenIL as the top-level parallel make is currently not supported.

#### CAUTION

The parameter  $PERL_MM_OPT$  would be defined in case Perl local::lib is installed on your system. You should unset this option before starting Buildroot, otherwise the compilation of Perl related packages will fail. For example, you might encounter the error information of the PERL\_MM\_OPT parameter while running the make command in a host Linux environment such as this:

```
make[1]: *** [core-dependencies] Error 1
make: *** [_all] Error 2
```

To resolve it, just unset the PERL\_MM\_OPT option.

\$ unset PERL\_MM\_OPT

### 2.4.2 Building the baremetal images

This section describes the steps for building the baremetal images for LS1021A-IoT, LS1043ARDB, and LS1046ARDB boards.

#### 2.4.2.1 Building the baremetal images for LS1021-IoT board

The two LS1021A-IoT Baremetal default configuration files can be found in the configs directory.

```
• configs/nxp ls1021aiot baremetal defconfig
```

```
• configs/nxp ls1021aiot baremetal ubuntu defconfig
```

The files include all the necessary U-Boot, Baremetal, kernel configurations, and application packages for the filesystem. Based on these files, you can build a complete Linux + Baremetal environment for the LS1021A-IoT platform without any major changes.

To build the final LS1021A-IoT Baremetal images, run the following commands:

```
$ cd openil
$ make nxp_ls1021aiot_baremetal_defconfig
$ make
# or make with a log
$ make 2>&1 | tee build.log
```

NOTE The make clean command should be implemented before any other new compilation.

After the correct compilation, you can find all the images for the platform in the output/images directory.

Here is a view of the directory, output/images/:

```
bm-u-boot.bin --- baremetal image run on slave core
boot.vfat
ls1021a-iot.dtb --- dtb file for ls1021a-iot
rootfs.ext2
rootfs.ext2.gz
rootfs.ext2.gz.uboot --- ramdisk can be used for debug on master core
rootfs.ext4.gz -> rootfs.ext2.gz
```

```
rootfs.tar
sdcard.img --- entire image can be programmed into the SD
uboot-env.bin
u-boot-with-spl-pbl.bin --- uboot image for ls1021a-iot master core
uImage --- kernel image for ls1021a-iot master core
version.json
```

For more details about Open Industry Linux, refer to the document Open\_Industrial\_Linux\_User\_Guide.

#### 2.4.2.2 Building the baremetal images for LS1028ARDB, LS1043ARDB, or LS1046ARDB

The baremetal default configuration files for LS1028ARDB, LS1043ARDB, or LS1046ARDB are located in the *configs* directory. These are:

- configs/nxp ls1028ardb baremetal-64b defconfig
  - configs/nxp ls1043ardb baremetal-64b defconfig

• configs/nxp ls1046ardb baremetal-64b defconfig

These files include all the necessary U-Boot, Baremetal, kernel configurations, and application packages for the filesystem. Based on these files, you can build a complete Linux + Baremetal environment for the LS1043A-RDB or LS1046A-RDB platform without any major changes.

Run the following commands to build the final LS1043ARDB or LS1043ARDB baremetal images:

\$ cd openil

#### Then:

```
$ make nxp ls1028ardb baremetal-64b defconfig
```

```
or
```

```
$ make nxp ls1043ardb baremetal-64b defconfig
```

or

```
$ make nxp ls1043ardb baremetal-64b defconfig
```

Then, use:

```
$ make
# or make with a log
$ make 2>&1 | tee build.log
```

NOTE

The make clean command should be implemented before any other new compilation.

After the correct compilation, you can find all the images for the platform in the output/images directory.

Here is a view of the directory, output/images/ for ls1043a-rdb:

bm-u-boot.bin --- baremetal image run on slave core
boot.vfat
fsl-ls1043a-rdb-sdk.dtb --- dtb file for ls1043a-rdb
rootfs.ext2
rootfs.ext2.gz
rootfs.ext2.gz.uboot --- ramdisk can be used for debug on master core

#### Industrial IoT Baremetal Framework Developer Guide, Rev. 1.6, 08/2019

```
    rootfs.ext4.gz -> rootfs.ext2.gz
    rootfs.tar
    sdcard.img --- entire image can be programmed into the SD
    uboot-env.bin
    uboot-with-spl-pbl.bin --- uboot image for ls1043ardb master core
    kernel-ls1043a-rdb.itb --- kernel, dtb, and rootfs image for ls1043ardb master core
    version.json
```

For more details about Open Industry Linux, refer to the document Open\_Industrial\_Linux\_User\_Guide.

### 2.4.2.3 Building the baremetal images for i.mx6q-sabresd board

The two i.mx6q-sabresd Baremetal default configuration files can be found in the configs directory.

configs/imx6q-sabresd baremetal defconfig

The files include all the necessary U-Boot, Baremetal, kernel configurations, and application packages for the filesystem. Based on these files, you can build a complete Linux + Baremetal environment for the i.mx6q-sabresd platform without any major changes. To build the final i.mx6q-sabresd Baremetal images, run the following commands:

```
$ cd openil $ make imx6q-sabresd_baremetal_defconfig
$ make # or make with a log $ make 2>&1 | tee build.log
```

NOTE The make clean command should be implemented before any other new compilation.

After the correct compilation, you can find all the images for the platform in the output/images directory. Here is a view of the directory, output/images/:

```
bm-u-boot.bin --- baremetal image run on slave core
boot.vfat
imx6q-sabresd.dtb --- dtb file for imx6q-sabresd
rootfs.ext2
rootfs.ext2.gz
rootfs.ext4.gz -> rootfs.ext2.gz
rootfs.tar
sdcard.img --- entire image can be programmed into the SD
SPL
u-boot.bin --- uboot image for imx6q-sabresd master core
u-boot.img
zImage --- kernel image for imx6q-sabresd master core
```

For more details about Open Industry Linux, refer to the document, Open\_Industrial\_Linux\_User\_Guide.

### 2.4.3 Booting up the Linux with Baremetal

Use the following steps to bootup the Linux + Baremetal system with the images built from OpenIL.

For platforms that can be booted up from the SD card, there is just one step required to program the sdcard.img into SD card.

- 1. Insert an SD card (of at least 2 GB size) into any Linux host machine.
- 2. Then, run the following commands:

```
 \ sudo dd if=./output/images/sdcard.img of=/dev/sdx
```

```
# or in some other host machine:
```

\$ sudo dd if=./output/images/sdcard.img of=/dev/mmcblkx # find the right SD Card device name in your host machine and replace the "sdx" or "mmcblkx".

3. Then, insert the SD card into the target board (LS1021A-IoT) and power on.

After the above mentioned steps are complete, the Linux system is booted up on the master core (core 0), and the Baremetal system is booted up on slave core (core 1) automatically.

# Chapter 3 Running examples

The following sections describe how to run the baremetal examples on the host environment for LS1021A-IoT board. Similar steps can be followed for LS1028A, LS1043A, and LS1046A reference design boards (RDB).

# 3.1 Preparing the console

Prepare a USB to TTL serial line, connect the pins to LPUART of Arduino pins to use it as UART1, please refer to Hardware setup.

In order to change LPUART to UART pins, modify 44<sup>th</sup> byte of RCW to 0x00.

In current baremetal framework design, two UART ports are used as console. UART1 is used for master core and UART2 is used for slave cores.

- For LS1021A, UART2 is pin-muxed with LPUART, so need to change the RCW to enable UART2 on master core. This modification has been implemented in the code for LS1021A-IoT board, so no need to do any code modification for this board.
- For customer specific boards, you need to apply the change to the RCW file:

```
bit[354~356] = 000
bit[366~368] = 111
```

# 3.2 Running the bare metal binary

As described earlier, there are two methods to compile the baremetal framework, a standalone method and using OpenIL. These are described in Building the baremetal images from U-Boot source code and Building the image through OpenIL respectively. If using OpenIL to compile the baremetal image, the barematal image is included in the sdcard imag and the master core runs the baremetal image on slave cores automatically. If using the standalone compilation method, you need to perform the steps below to run the baremetal binary from U-Boot prompt of master core:

1. After starting U-Boot on the master, download the bare metal image: u-boot.bin on 0x84000000 using the command below:

```
=> tftp 0x84000000 xxxx/u-boot.bin
```

Where

- xxxx is your tftp server directory.
- 0x84000000 is the address of CONFIG\_SYS\_TEXT\_BASE on bare metal.
- 2. Then, start the baremetal cores using the command below:

=> cpu start 0x84000000

3. Last, the UART1 port displays the logs, and the bare metal application runs on slave cores successfully.

The figure below displays a sample output log.

```
U-Boot 2017.07-21736-g7fb4afc-dirty (Mar 15 2018 - 15:50:12 +0800)
CPU:
      Freescale LayerScape LS1021E, Version: 2.0, (0x87081120)
Clock Configuration:
       CPU0(ARMV7):1000 MHz,
      Bus:300 MHz, DDR:800 MHz (1600 MT/s data rate),
Reset Configuration Word (RCW):
      00000000: 0608000a 0000000 00000000 00000000
      00000010: 20000000 08407900 60025a00 21046000
      00000020: 0000000 0000000 0000000 00038000
      00000030: 20024800 841b1340 00000000 00000000
I2C:
      ready
I2C: ready
DRAM: 256 MiB
EEPROM: NXID v16777216
In:
      serial
Out: serial
Err: serial
Core[1] in the loop...
i2c read: 0xa0
[ok]i2c test ok
IRQ 0 has been registered as SGI
IRQ 195 has been registered as HW IRQ
SGI signal: Core[l] ack irq : 0
[ok]GPIO test ok
```

Figure 6. Baremetal output logs

# Chapter 4 Development based on baremetal framework

This chapter describes how to develop customer specific application based on QorIQ layerscape baremetal framework.

# 4.1 Developing the baremetal application

The directory "app" in the U-boot repository includes the test cases for testing the I2C, GPIO and IRQ init features. You can write actual applications and store them in this directory.

# 4.2 Example software

This section describes how to analyze a GPIO sample code and use it to write the actual application.

# 4.2.1 Main file app.c

The file <industry-Uboot path>/app.c, is the main entrance for all applications. Users can modify the app.c file to add their applications.

- If using standalone method to build the baremetal image as described in Building the baremetal images from U-Boot source code, just change the directory to industry-Uboot path to check the app.c file.
- If using OpenIL to compile the baremetal binary, you need to change the directory to output/build/bm-uboot-Baremetal-Framework/ to check the app.c file.

The following is a sample code of the file app.c that shows how to add the example test cases of I2C, IRQ, and GPIO.

```
void corel_main(void)
{
    test_i2c();
    test_irq_init();
    test_gpio();
    return;
}
```

## 4.2.2 Common header files

There are some common APIs provided by baremetal. The table below describes the header files that include the APIs.

| Header file    | Description                                                                     |
|----------------|---------------------------------------------------------------------------------|
| asm/io.h       | Read/Write IO APIs.<br>For example,raw_readb,raw_writeb, out_be32, and in_be32. |
| linux/string.h | APIs for manipulating strings.<br>For example, strlen, strcpy, and strcmp.      |
| linux/delay.h  | APIs used for small pauses.<br>For example, udelay and mdelay.                  |

#### Table 4. Common header file description

Table continues on the next page...

| Header file   | Description                                              |
|---------------|----------------------------------------------------------|
| linux/types.h | APIs specifying common types.<br>For example,u32 andu64. |
| common.h      | Common APIs.<br>For example, printf and puts.            |

Table 4. Common header file description (continued)

# 4.2.3 GPIO file

The file uboot/app/test\_gpio.c is one example to test the GPIO feature, and shows how to write a GPIO application.

Here is an example for the Is1021aiot board:

On Is1021aiot board, first you need the GPIO header file, asm-generic/gpio.h, which includes all interfaces for the GPIO. Then, configure GPIO25 to OUT direction, and configure GPIO24 to IN direction. Last, by writing the value 1 or 0 to GPIO25, you can receive the same value from GPIO24.

The table below shows the APIs used in the file test\_gpio.c example.

| Function declaration                    | Description                                                                                                                                                                                                                                 |
|-----------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| gpio_request (ngpio, label)             | <ul> <li>Requests GPIO.</li> <li>ngpio - The GPIO number, such as 25, that is for GPIO25.</li> <li>label - the name of GPIOI request.</li> <li>Returns 0 if OK, -1 on error.</li> </ul>                                                     |
| gpio_direction_output (ngpio,<br>value) | Configures the direction of GPIO to OUT and writes the value to it. <ul> <li>ngpio - The GPIO number, such as 25,that is, for GPIO25.</li> <li>value - the value written to this GPIO.</li> </ul> Returns 0 if low, 1 if high, -1 on error. |
| gpio_direction_input (ngpio);           | Configures the direction of GPIO to IN.<br>ngpio - The GPIO number, such as 24, that is for GPIO24;<br>Returns 0 if ok, -1 on error.                                                                                                        |
| gpio_get_value (ngpio)                  | <ul> <li>Reads the value.</li> <li>ngpio - The GPIO number, such as 24, that is for GPIO24;</li> <li>value - the value written to this GPIO.</li> <li>Returns 0 if low, 1 if high, -1 on error.</li> </ul>                                  |
| gpio_free (ngpio)                       | Frees the GPIO just requested.<br>• ngpio - The GPIO number, such as 24, that is for GPIO24;<br>Returns 0 if ok, -1 on error.                                                                                                               |

 Table 5. GPIO APIs and their description

Development based on baremetal framework

# 4.2.4 I2C file

The file uboot/app/test\_i2c.c can be used as an example to test the I2C feature and shows how to write an I2C application.

On Is1021aiot board, first, you need to first include the I2C header file, i2c.h, which includes all interfaces for I2C. Then, you need to read a fixed data from offset 0 of Audio codec device (0x2A), if the data is 0xa0, the message [ok]I2C test ok is displayed on the console.

On Is1043ardb board, read a fixed data from offset 0 of INA220 device(0x40). If the data is 0x39, a message, [ok]I2C test ok is displayed on the console.

The table below shows the APIs used in the sample file, test\_i2c.c.

| Function declaration                                                                   | Description                                                                                                                                                                                                                                                                                                                                                                                                                             |
|----------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| int i2c_set_bus_num (unsigned int bus)                                                 | Sets the I2C bus.<br>bus- bus index, zero based<br>Returns 0 if OK, -1 on error.                                                                                                                                                                                                                                                                                                                                                        |
| int i2c_read (uint8_t chip, unsigned int addr,<br>int alen, uint8_t *buffer, int len)  | <ul> <li>Read data from I2C device chip.</li> <li>chip - I2C chip address, range 0127</li> <li>addr - Memory (register) address within the chip</li> <li>alen - Number of bytes to use for addr (typically 1, 2 for larger memories, 0 for register type devices with only one register)</li> <li>buffer - Where to read/write the data</li> <li>len - How many bytes to read/write</li> <li>Returns 0 if ok, not 0 on error</li> </ul> |
| int i2c_write (uint8_t chip, unsigned int addr,<br>int alen, uint8_t *buffer, int len) | <ul> <li>Writes data to i2c device chip.</li> <li>chip - I2C chip address, range 0127</li> <li>addr - Memory (register) address within the chip</li> <li>alen - Number of bytes to use for addr (typically 1, 2 for larger memories, 0 for register type devices with only one register)</li> <li>buffer - Where to read/write the data</li> <li>len - How many bytes to read/write</li> <li>Returns 0 if ok, not 0 on error</li> </ul> |

 Table 6. I2C APIs and their description

## 4.2.5 IRQ file

The file, *uboot/app/test\_irq\_init.c* is an example to test the IRQ and IPI (Inter-Processor Interrupts) feature, and shows how to write an IRQ application.

First, you need the IRQ's header asm/interrupt-gic.h, which includes all interfaces for IRQ. Then, register an IRQ function for SGI 0. After setting a SGI signal, the CPU gets this IRQ and runs the IRQ function. You also need to register a hardware interrupt function to show how to use the external hardware interrupt.

SGI IRQ is used for inter-processor interrupts, and it can only be used between bare metal cores. In case you want to communicate between baremetal core and Linux core, refer to the ICC chapter. SGI IRQ id is 0-15, but 8 is reserved to be used for ICC.

The table below shows the APIs used in the sample file, *test\_irq\_init.c*.

| Return type API name<br>(parameter list)                        | Description                                                                                                                                              |
|-----------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------|
| void gic_irq_register (int<br>irq_num, void (*irq_handle)(int)) | <ul> <li>Registers an IRQ function.</li> <li>irq_num- IRQ id, 0-15 for SGI, 16-31 for PPI, 32-1019 for SPI</li> <li>irq_handle - IRQ function</li> </ul> |
| void gic_set_sgi (int core_mask,<br>u32 hw_irq)                 | Sets a SGI IRQ signal.<br>• core_mask - target core mas<br>• hw_irq - IRQ id                                                                             |
| void gic_set_target (u32<br>core_mask, unsigned long<br>hw_irq) | Sets the target core for hw IRQ.<br>• core_mask - target core mas<br>• hw_irq - IRQ id                                                                   |
| void gic_set_type (unsigned<br>long hw_irq)                     | Sets the type for hardware IRQ to identify whether the corresponding interrupt is edge-<br>triggered or level-sensitive.<br>• hw_irq - IRQ id            |

#### Table 7. IRQ APIs and their description

### 4.2.6 QSPI file

The file uboot/app/test\_qspi.c provides an example that can be used to test the QSPI feature. The below steps show how to write a QSPI application:

- 1. First, locate the QSPI header files spi flash.h and spi.h, which include all interfaces for QSPI.
- 2. Then, initialize the QSPI flash. You need to erase the corresponding flash area and confirm erase operation is successful.
- 3. Now, read or write to the flash with an offset of 0x3f00000 and size of 0x40000.

The table below shows the APIs used in the file  $test_qsip.c$  example.

| Table | 8. | QSPI | APIs |
|-------|----|------|------|
|-------|----|------|------|

| API name (type)                       | Description                                               |
|---------------------------------------|-----------------------------------------------------------|
| spi_find_bus_and_cs(bus,cs, &bus_dev, | Finds if the SPI device already exists.                   |
| &new)                                 | • "bus" - bus index, zero based.                          |
|                                       | <ul> <li>"cs" – the value to chip select mode.</li> </ul> |
|                                       | <ul> <li>"bus_dev" - If the bus is found.</li> </ul>      |
|                                       | <ul> <li>"new" – If the device is found.</li> </ul>       |
|                                       | Returns 0 if ok, -ENODEV on error.                        |

Table continues on the next page ...

| API name (type)                                       | Description                                                                                                                                                                                                                                                                                                                                                           |
|-------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| spi_flash_probe_bus_cs(bus, cs, speed,<br>mode, &new) | Initializes the SPI flash device. <ul> <li>"bus" - bus index, zero based.</li> <li>"cs" – the value to Chip Select mode.</li> <li>"speed" – SPI flash speed, can use 0 or CONFIG_SF_DEFAULT_SPEED.</li> <li>"mode" –SPI flash mode, can use 0 or CONFIG_SF_DEFAULT_MODE.</li> <li>"new" – If the device is initialized.</li> </ul> Returns 0 if ok, -ENODEV on error. |
| dev_get_uclass_priv(new)                              | Gets the SPI flash.<br>• "new" - The device being initialized.<br>Returns flash if ok , NULL on error.                                                                                                                                                                                                                                                                |
| spi_flash_erase(flash, offset, size)                  | <ul> <li>Erases the specified location and length of the flash content, erases the content of all.</li> <li>"flash" - Flash is being initialized.</li> <li>"offset" - Flash offset address.</li> <li>"size" - Erase the length of the data.</li> <li>Returns 0 if ok, 10 on error.</li> </ul>                                                                         |
| spi_flash_read(flash, offset, len, vbuf)              | <ul> <li>Reads flash data to memory.</li> <li>"flash" - The flash being initialized.</li> <li>"offset" – Flash offset address.</li> <li>"size" - Read the length of the data.</li> <li>Returns 0 if ok, !0 on error.</li> </ul>                                                                                                                                       |
| spi_flash_write(flash, offset, len, buf)              | <ul> <li>Writes memory data to flash.</li> <li>"flash" - The flash being initialized.</li> <li>"offset" - Flash offset address.</li> <li>"size" - Write the length of the data.</li> <li>Returns 0 if ok, 10 on error.</li> </ul>                                                                                                                                     |

| Table 8. | QSPI | APIs | (continued) |
|----------|------|------|-------------|
|----------|------|------|-------------|

# 4.2.7 Ethernet

The file  $uboot/app/test_net.c$  provides an example to test the Ethernet feature and shows how to write a net application for using this feature.

Here is an example for the LS1043ARDB (or LS1046ARDB) board.

- 1. Connect one of Ethernet port of LS1043ARDB board to one host machine using Ethernet cable.
- 2. Configure the IP address of the host machine as 192.168.1.2.
- 3. Power up the LS1043ARDB board. If the network is connected, the mesage host 192.168.1.2 is alive is displayed on the console.

4. The IP address of the board and host machine are defined in the file test\_net.c. In this file, modify the IP address of LS1043ARDB board using variable <code>ipaddr</code> and change the IP address of host machine using variable <code>ping\_ip</code>.

The table below lists the Net APIs and their description.

| API name (type)                            | Description                                                                                                                                                  |
|--------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------|
| void net_init (void)                       | Initializes the network                                                                                                                                      |
| int net_loop (enum proto_t protocol)       | Main network processing loop. <ul> <li>enum proto_t protocol - protocol type</li> </ul>                                                                      |
| int eth_receive (void *packet, int length) | Read data from NIC device chip. <ul> <li>void *packet</li> <li>length - Network packet length</li> </ul> Returns length                                      |
| int eth_send (void *packet, int length)    | <ul> <li>Writes data to NIC device chip.</li> <li>packet - Network packet length</li> <li>length - Network packet length</li> <li>Returns length.</li> </ul> |

#### Table 9. Net APIs and their description

### 4.2.8 USB file

The file  $uboot/app/test_usb.c$  provides an example that can be used to test the USB features. The steps below show how to write a USB application:

- 1. Connect a USB disk to the USB port.
- 2. Include the header file, usb.h, which includes all APIs for USB.
- 3. Initialize the USB device using the usb init API.
- 4. Scan the USB storage device on the USB bus using the usb stor scanAPI.
- 5. Get the device number using the blk get devnum by type API.
- 6. Read data from the USB disk using the blk dread API.
- 7. Write data to the USB disk using the blk dwrite API.

The table below shows the APIs used in the file test\_usb.c example:

#### Table 10. USB APIs and their description

| Description                                                                  |
|------------------------------------------------------------------------------|
| Initializes the USB controller.                                              |
| Stops the USB controller.                                                    |
| Scans the USB and reports device information to the user if mode = 1         |
| <ul> <li>Mode – if mode = 1, the information is returned to user.</li> </ul> |
| Returns the current device or -1.                                            |
|                                                                              |

Table continues on the next page ...

| struct blk_desc<br>*blk_get_devnum_by_type(enum if_type<br>if_type, int devnum)                                  | Get a block device by type and number.                          |
|------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------|
|                                                                                                                  | <ul> <li>If_type – Block device type</li> </ul>                 |
|                                                                                                                  | devnum - device number                                          |
|                                                                                                                  | Returns point to block device descriptor, or NULL if not found. |
| unsigned long blk_dread(struct blk_desc                                                                          | Reads data from USB device                                      |
| *block_dev, lbaint_t start, lbaint_t blkcnt,<br>void *buffer);                                                   | <ul> <li>block_dev – block device descripter</li> </ul>         |
|                                                                                                                  | start – start block                                             |
|                                                                                                                  | <ul> <li>blkcnt – block number</li> </ul>                       |
|                                                                                                                  | <ul> <li>buffer – buffer to store the data</li> </ul>           |
|                                                                                                                  | Returns the block number from which, data is read.              |
| unsigned long blk_dwrite(struct blk_desc<br>*block_dev, lbaint_t start, lbaint_t blkcnt,<br>const void *buffer); | Writes data to USB device                                       |
|                                                                                                                  | <ul> <li>block_dev – block device descripter</li> </ul>         |
|                                                                                                                  | start – start block                                             |
|                                                                                                                  | <ul> <li>blkcnt – block number</li> </ul>                       |
|                                                                                                                  | <ul> <li>buffer – buffer to store the data</li> </ul>           |
|                                                                                                                  | Returns the block number to which data is written.              |

#### Table 10. USB APIs and their description (continued)

### 4.2.9 PCIe file

The file app/test\_pcie.c provides a sample code to test PCIe and network card (such as e1000) features. The steps below show how to write a PCIe and net application:

- 1. Insert a PCIe network card (such as e1000) into PCIe2, or PCIe3 slot (if it exists).
- 2. Configure the IP address of the host machine to 192.168.1.2.
- 3. Include the files include/pci.h and include/ netdev.h.
- 4. Initialize the PCIe controller using the pci init API.
- 5. Get uclass device by its name using the uclass\_get\_device\_by\_seq API.
- 6. Initialize the PCIe network device using the  $pci_eth_init$  API.
- 7. Begin pinging the host machine using the net loop API.

The table below shows the APIs used in the file test pcie.c example.

#### Table 11. PCIe APIs and their description

| API name (type)     | Description                                               |
|---------------------|-----------------------------------------------------------|
| void pci_init(void) | Initializes the PCIe controller. Does not return a value. |
|                     |                                                           |

Table continues on the next page...

| int uclass_get_device_by_seq(enum             | Gets the uclass device based on an ID and sequence:                           |
|-----------------------------------------------|-------------------------------------------------------------------------------|
| uclass_id id, int seq, struct udevice **devp) | <ul> <li>id – uclass ID</li> </ul>                                            |
|                                               | seq – sequence                                                                |
|                                               | devp – Pointer to device                                                      |
|                                               | Returns 0 if Ok, negative on error.                                           |
| static inline int pci_eth_init(bd_t *bis)     | Initializes network card on the PCIe bus                                      |
|                                               | <ul> <li>Bis – struct containing variables accessed by shared code</li> </ul> |
|                                               | Returns the number of network cards.                                          |
| int net_loop (enum proto_t protocol)          | Main network processing loop.                                                 |
|                                               | <ul> <li>enum proto_t protocol - protocol type</li> </ul>                     |
|                                               | Returns 0 if Ok, negative value on error.                                     |

#### Table 11. PCIe APIs and their description (continued)

### 4.2.10 CAN file

The file  $app/test_flexcan.c$  provides a sample test case to test flexcan and CANopen features. The following steps show the design process:

- 1. Register the receive interruption function for flexcan.
- 2. Register an overflow interruption function for flextimer.
- 3. Initialize a list of callback functions.
- 4. Set the node ID of this node.

The table below shows the APIs used in the file test flexcan.c example.

#### Table 12. CAN APIs and their description

| API name (type)                                                        | Description                                                                                                                                                     |  |  |
|------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------|--|--|
| void test_flexcan(void)                                                | It is the test code entry for flexcan.                                                                                                                          |  |  |
| void flexcan_rx_irq(struct can_module<br>*canx)                        | Flexcan receive interruption function. <ul> <li>canx – flexcan interface</li> </ul>                                                                             |  |  |
| void flexcan_receive(struct can_module<br>*canx, struct can_frame *cf) | <ul> <li>Flexcan receives CAN data frame.</li> <li>canx – flexcan interface</li> <li>cf – CAN message</li> </ul>                                                |  |  |
| UNS8 setState(CO_Data* d,<br>e_nodeState newState)                     | <ul> <li>Sets node state</li> <li>d – object dictionary</li> <li>newState – The state that needs to be set</li> <li>Returns 0 if Ok, &gt; 0 on error</li> </ul> |  |  |
| void canDispatch(CO_Data* d, Message<br>*m)                            | <ul> <li>CANopen handles data frames that CAN receive:</li> <li>d – object dictionary</li> <li>m – Received CAN message</li> </ul>                              |  |  |

Table continues on the next page ...

| int flexcan_send(struct can_module<br>*canx, struct can_frame *cf) | flexcan interface sends CAN message<br>• canx – flexcan interface<br>• cf – CAN message |
|--------------------------------------------------------------------|-----------------------------------------------------------------------------------------|
| void flextimer_overflow_irq(void)                                  | Flextimer overflow interruption handler function                                        |
| void timerForCan(void)                                             | CANopen virtual clock. Call this function per 100us.                                    |

Table 12. CAN APIs and their description (continued)

#### • The following log shows the CANopen slave node state:

```
=> flexcan error: 0x42242!
Note: slave node entry into the stop mode!
Note: slave node initialization is complete!
Note: slave node entry into the preOperation mode!
Note: slave node entry into the operation mode!
Note: slave node initialization is complete!
Note: slave node entry into the preOperation mode!
Note: slave node entry into the operation mode!
```

## 4.2.11 ENETC file

The file app/test\_net.c provides an example to test ENETC Ethernet feature and shows how to write a net application for using this feature. This is a special case of using Net APIs.

test\_net for ENETC is only an example for the LS1028ARDB board (CONFIG\_ENETC\_COREID\_SET enabled).

- 1. Connect ENETC port of LS1028ARDB board to one host machine using Ethernet cable.
- 2. Configure the IP address of the host machine as 192.168.1.2.
- 3. Power up the LS1028ARDB board. If the network is connected, the mesage host 192.168.1.2 is alive is displayed on the console.
- 4. The IP address of the board and host machine are defined in the file test\_net.c. In this file, modify the IP address of LS1028ARDB board using variable ipaddr and change the IP address of host machine using variable ping ip.

The table below lists the Net APIs for ENETC and their description, refer to section 4.2.7 for other Net APIs.

| API name (type) Description |                                                           |
|-----------------------------|-----------------------------------------------------------|
| void pci_init(void)         | Initializes the PCIe controller. Does not return a value. |
| void eth_initialize(void)   | Initializes the ethernet.                                 |

Table 13. ENETC APIs and their description

# 4.3 ICC module

Inter-core communication (ICC) module works on Linux core (master) and Baremetal core (slave), provides the data transfer between cores via SGI inter-core interrupt and share memory blocks. It can support multi-core silicon platform and transfer the data concurrently and efficiently.

ICC module is structured based on two basics:

 SGI: Software-generated Interrupts in ARM GIC, used to generate inter-core interrupts. The ICC module uses the number 8 SGI interrupt for all Linux and Baremetal cores. • Shared memory: A memory space shared by all platform cores. The base address and size of the share memory should be defined in header files before compilation.

ICC modules can work concurrently, lock-free among multi-core platform, and support broadcast case with Buffer Descriptor Ring mechanism.

The figure below shows the basic operating principle for data transfer from Core 0 to Core 1. After the data writing and head point moving to next, Core 0 triggers a SGI (8) to Core 1, then Core 1 gets the BD ring updated status and reads the new data, then moves the tail point to next.



Figure 7. BD rings for inter-core communication

For a multi-core platform (that is, four cores), the total BD rings are arranged as shown in the following figure. (See the BD rings on Core 0 and Core 1.)

| Core0 to Core1 | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 |
|----------------|---|---|---|---|---|---|---|---|---|---|
| Core0 to Core2 | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 |
| Core0 to Core3 | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 |
|                |   |   |   |   |   |   |   |   |   |   |
| Core1 to Core0 | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 |
| Core1 to Core2 | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 |
| Core1 to Core3 | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 |

#### Figure 8. BD rings for multi-core platform

All the ICC ring structures, BD structures and blocks for data are in the shared memory. A four-core platform ICC module would map the shared memory as shown in the figure below.



Figure 9. ICC shared memory map for the four-core platform

Generally, Core 0 runs Linux as master core, other cores run Baremetal as slaves. They obtain the same size of share memory to structure the rings and BDs, and split the blocks space with 4k unit for each block. The reserved space at the top of the share memory is out of the ICC module and for the custom usage.

For LS1021A platform with two cores, the share memory map is defined as:

- The total share memory size is 256 MB.
- The reserved space for custom usage is 16 MB at the top of the share memory space.
- Core 0 runs Linux as master core, the share memory size for ICC is 120 MB, in which the ring and BD structure space is 2 M, and the block space for data is 118 MB with 4K for each block.
- Core 1 runs Baremetal as slave core, the share memory size for ICC is 120 MB, in which the ring and BD structure space is 2M, and the block space for data is 118 MB with 4K for each block.

The ICC module includes two parts of the code:

- ICC code for Linux user space, works for data transfer between master core and slave cores. The code is integrated into the OpenIL and named icc package. After the compilation, the icc binary is put into the Linux filesystem.
- ICC code for Baremetal, runs on every slave core, works for data transfer between baremetal cores and master core.

The ICC code for Linux user space in OpenIL directory:

package/icc/src/

- ----- inter-core-comm.c
- inter-core-comm.h --- include the header file to use ICC module
- Makefile

The ICC code for Baremetal in Baremetal directory:

baremetal/

- ---- arch/arm/include/asm/inter-core-comm.h --- include the header file to use ICC module

L--- cmd/icc.c --- the example case commands

The APIs ICC exported out for usage in both Linux user space and Baremetal code.

| APIs                                                                                             | Description                                                                                                                                   |
|--------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------|
| unsigned long icc_ring_state(int coreid)                                                         | Checks the ring and block state.<br>Returns:<br>• 0 - if empty<br>• !0 - the working block address currently.                                 |
| Unsigned long icc_block_request(void)                                                            | Requests a block, which is ICC_BLOCK_UNIT_SIZE size.<br>Returns:<br>• 0 - failed<br>• !0 - block address can be used.                         |
| void icc_block_free(unsigned long block)                                                         | Frees a block requested.<br>Be careful if the destination cores are working on this block.                                                    |
| int icc_irq_register(int src_coreid, void<br>(*irq_handle)(int, unsigned long, unsigned<br>int)) | Registers ICC callback handler for received data.<br>Returns:<br>• 0 - on success<br>• -1 - if failed.                                        |
| int icc_set_block(int core_mask, unsigned int<br>byte_count, unsigned long block)                | Sends the data in the block to a core or multi-core.<br>This triggers the SGI interrupt.<br>Returns:<br>• 0 - on success<br>• -1 - if failed. |
| void icc_show(void)                                                                              | Shows the ICC basic information.                                                                                                              |
| int icc_init(void)                                                                               | Initializes the ICC module.                                                                                                                   |

#### Table 14. ICC APIs

## 4.3.1 ICC examples

This section provides example commands for use cases in both Linux user space and Baremetal code. They can be used to check and verify the ICC module conveniently.

1. In Linux user space, use the command  ${\tt icc}$  to display the supported cases.

```
root@OpenIL-Ubuntu:~# icc
icc show - Shows all icc rings status at this core
icc perf <core_mask> <counts> - ICC performance to cores <core_mask> with <counts> bytes
icc send <core_mask> <data> <counts> - Sends <counts> <data> to cores <core_mask>
icc irq <core_mask> <irq> - Sends SGI <irq> ID[0 - 15] to <core_mask>
icc read <addr> <counts> - Reads <counts> 32bit register from <addr>
icc write <addr> <data> - Writes <data> to a register <addr>
```

Development based on baremetal framework

2. Likewise, in Baremetal system, use the command icc to view the supported cases.

```
root@OpenIL-Ubuntu:~# icc send 0x2 0x1f 128
gic_base: 0xb6fa0000, share_base: 0xa7e87000, share_phy: 0xb0000000, block_phy: 0xb0200000
ICC send testing ...
Target cores: 0x2, bytes: 128
ICC send: 128 bytes to 0x2 cores success
all cores: reserved_share_memory_base: 0xbf000000; size: 16777216
mycoreid: 0; ICC_SGI: 8; share_memory_size: 125829120
block_unit_size: 4096; block number: 30208; block_idx: 0
#ring 0 base: 0xa7e87000; dest_core: 0; SGI: 8
desc_num: 128; desc_base: 0xb0000048; head: 0; tail: 0
busy_counts: 0; interrupt_counts: 0
#ring 1 base: 0xa7e87024; dest_core: 1; SGI: 8
desc_num: 128; desc_base: 0xb0000448; head: 1; tail: 1
busy_counts: 0; interrupt_counts: 1
```

3. The ICC module command examples on LS1021A-IoT Linux (Core 0) + Baremetal (Core 1) system:

Run icc send 0x2 0x1f 128 to send 128 bytes data 0x1f to core 1.

```
root@OpenIL-Ubuntu:~# icc send 0x2 0x1f 128
gic_base: 0xb6fa0000, share_base: 0xa7e87000, share_phy: 0xb0000000, block_phy: 0xb0200000
ICC send testing ...
Target cores: 0x2, bytes: 128
ICC send: 128 bytes to 0x2 cores success
all cores: reserved_share_memory_base: 0xbf000000; size: 16777216
mycoreid: 0; ICC_SGI: 8; share_memory_size: 125829120
block_unit_size: 4096; block number: 30208; block_idx: 0
#ring 0 base: 0xa7e87000; dest_core: 0; SGI: 8
desc_num: 128; desc_base: 0xb0000048; head: 0; tail: 0
busy_counts: 0; interrupt_counts: 0
#ring 1 base: 0xa7e87024; dest_core: 1; SGI: 8
desc_num: 128; desc_base: 0xb0000448; head: 1; tail: 1
busy_counts: 0; interrupt_counts: 1
```

4. At the same time, Core 1 prints the receival information.

=> Get the ICC from core 0; block: 0xb0200000, bytes: 128, value: 0x1f

### 4.4 Hardware resource allocation

This section describes how to modify the hardware resource allocation depending on the application and used reference design board.

### 4.4.1 LS1021A-loT board

#### 4.4.1.1 Linux DTS

Remove cpu1 node on DTS, and remove all the devices that bare metal has used.

### 4.4.1.2 Memory configuration

LS1021A-IoT board has a 1 GB size DDR. The DDR memory can be configured into three partitions: 512M for core0 (Linux), 256M for core1 (bare metal), and 256M for shared memory.

The configuration is in the path: include/configs/ls1021aiot\_config.h.

#define CONFIG\_SYS\_DDR\_SDRAM\_SLAVE\_SIZE (256 \* 1024 \* 1024)
#define CONFIG SYS DDR SDRAM MASTER SIZE (512 \* 1024 \* 1024)

NOTE

Memory configuration must be consistent with the U-Boot configuration of core0.

Modify "config\_sys\_malloc\_len" in *include/configs/ls1021aiot\_config.h* to change the maximum size of malloc.

You can use functions included in malloc.h to allocate or free memory in your program. These functions are listed in the table below.

| API name (type)           | Description                                                                                            |  |
|---------------------------|--------------------------------------------------------------------------------------------------------|--|
| void_t* malloc (size_t n) | Allocates memory                                                                                       |  |
|                           | <ul> <li>n – length of allocated chunk</li> </ul>                                                      |  |
|                           | Returns a pointer to the newly allocated chunk                                                         |  |
| void free (void *ptr)     | Releases the chunk of memory pointed to by ptr (where ${\tt ptr}$ is a pointer to the chunk of memory) |  |

Table 15. Description of memory APIs

The memory configuration for bare metal is shown in the figure below.



Figure 10. Memory configuration

Development based on baremetal framework

### 4.4.1.3 GPIO

LS1021A has four GPIO controllers. The configuration is defined in the file:*arch/arm/dts/ls1021a-iot.dtsi*. You can add a GPIO node in the file ls1021a-iot.dtsi to assign a GPIO resource to different cores. Following is a sample code for adding a GPIO node.

```
&gpio2
{
  status = "okay";
};
```

# 4.4.1.4 I2C

LS1021A has three I2C controllers. Configure the I2C bus on 1s1021aiot config.h using the commands below:

```
// include/configs/ls1021aiot_config.h:
#define CONFIG_SYS_I2C_MXC_I2C1 /* enable I2C bus 1 */
#define CONFIG_SYS_I2C_MXC_I2C2 /* enable I2C bus 2 */
#define CONFIG SYS I2C MXC I2C3 /* enable I2C bus 3 */
```

## 4.4.1.5 Hardware interrupts

LS1021A has six IRQs as external IO signals connected to interrupt the controller. We can use these six IRQs on bare metal cores. The ids for these signals, IRQ0-IRQ5 are: 195, 196, 197, 199, 200, and 201.

GIC interrupt APIs are defined in the file, asm/interrupt-gic.h. The following example shows how to register a hardware interrupt:

```
//register HW interrupt
void gic_irq_register(int irq_num, void (*irq_handle)(int));
void gic_set_target(u32 core_mask, unsigned long hw_irq);
void gic_set_type(unsigned long hw_irq);
```

### 4.4.1.6 IFC

LS1021A-IoT board has no IFC device, but the LS1021A SoC has an IFC interface. Since IFC is multiplexed with QSPI, you need to modify the RCW to use the IFC interface, and add a configuration such as shown in the sample code provided below. Users can modify the code to support IFC as per the actual scenario.

```
#define CONFIG_FSL_IFC
#define CONFIG_SYS_CPLD_BASE 0x7fb00000
#define CPLD_BASE_PHYS CONFIG_SYS_CPLD_BASE
#define CONFIG_SYS_FPGA_CSPR_EXT (0x0)
#define CONFIG_SYS_FPGA_CSPR (CSPR_PHYS_ADDR(CPLD_BASE_PHYS) | \
CSPR_PORT_SIZE_8 | \
CSPR_MSEL_GPCM | \
CSPR_V)
#define CONFIG_SYS_FPGA_AMASK IFC_AMASK(64 * 1024)
#define CONFIG_SYS_FPGA_CSOR (CSOR_NOR_ADM_SHIFT(4) | \
CSOR_NOR_NOR_MODE_AVD_NOR | \
CSOR_NOR_NOR_MODE_AVD_NOR | \
CSOR_NOR_TRHZ_80)
/* CPLD Timing parameters for IFC GPCM */
#define CONFIG_SYS_FPGA_FTIM0 (FTIM0_GPCM_TACSE(0xf) | \
FTIM0 GPCM TEADC(0xf) | \
```

```
FTIM0_GPCM_TEAHC(0xf))
```

```
#define CONFIG_SYS_FPGA_FTIM1 (FTIM1_GPCM_TACO(0xff) | \
FTIM1 GPCM TRAD(0x3f))
```

```
#define CONFIG_SYS_FPGA_FTIM2 (FTIM2_GPCM_TCS(0xf) | \
FTIM2_GPCM_TCH(0xf) | \
FTIM2_GPCM_TWP(0xff))
```

```
#define CONFIG_SYS_FPGA_FTIM3 0x0
#define CONFIG_SYS_CSPR1_EXT CONFIG_SYS_FPGA_CSPR_EXT
#define CONFIG_SYS_CSPR1 CONFIG_SYS_FPGA_CSPR
#define CONFIG_SYS_CSOR1 CONFIG_SYS_FPGA_AMASK
#define CONFIG_SYS_CS1_FTIM0 CONFIG_SYS_FPGA_FTIM0
#define CONFIG_SYS_CS1_FTIM1 CONFIG_SYS_FPGA_FTIM1
#define CONFIG_SYS_CS1_FTIM2 CONFIG_SYS_FPGA_FTIM2
#define CONFIG_SYS_CS1_FTIM3 CONFIG_SYS_FPGA_FTIM3
```

### 4.4.1.7 USB

LS1021AIOT has a single DW3 USB controller, which is assigned to the second core, by default. Use the command make menuconfig to re-configure the U-Boot to assign it to other cores.

```
ARM architecture --->
[*] Enable baremetal
[*] Enable USB for baremetal
(1) USB0 is assigned to core1
(1) USB Controller numbers
```

### 4.4.1.8 PCIe

LS1021AIOT has two PCIe controllers. By deafult, one is assigned to core0 and the other is assigned to core1. Use the make menuconfig command to re-configure the U-Boot, in order to re-assign the cores.

```
ARM architecture --->
[*] Enable baremetal
[*] Enable PCIE for baremetal
(0) PCIe1 is assigned to core0
(1) PCIe2 is assigned to core1
(2) PCIe Controller numbers
```

### 4.4.1.9 FlexCAN

#### **Assigning CAN3 to Baremetal**

In baremetal, the port is allocated through the flexcan.c file. The flexcan.c path is industry-uboot/drivers/flexcan/flexcan.c. In this file, you need to define the following variables:

```
struct can_bittiming_t flexcan3_bittiming = CAN_BITTIM_INIT(CAN_500K);
```

#### Development based on baremetal framework

Note: You also need to set the bit timing and baud rate (500K) of the CAN port.

```
struct can_ctrlmode_t flexcan3_ctrlmode = {
.loopmode = 0, /* Indicates whether the loop mode is enabled */
.listenonly = 0, /* Indicates whether the only-listen mode is enabled */
.samples = 0,
.err_report = 1,
};
struct can_init_t flexcan3 = {
.canx = CAN3, /* Specify CAN port */
.bt = &flexcan3_bittiming,
.ctrlmode = &flexcan3_ctrlmode,
.reg_ctrl_default = 0,
.reg_esr = 0
};
```

#### **Optional Parameters**

#### • CAN port

| #define | CAN3 | (struct | can | module | *)CAN3 | BASE) |
|---------|------|---------|-----|--------|--------|-------|
| #define | CAN4 | (struct | can | module | *)CAN4 | BASE) |

#### • Baud rate

#define CAN\_1000K 10
#define CAN\_500K 20
#define CAN\_250K 40
#define CAN\_200K 50
#define CAN\_125K 80
#define CAN\_100K 100
#define CAN\_20K 500
#define CAN\_10K 1000
#define CAN\_10K 1000
#define CAN 5K 2000

Use the command make menuconfig for configuring the FlexCAN setting for LS1021A reference design boards, as shown below: .

```
Device Drivers --->
CAN support --->
[*] Support for Freescale FLEXCAN based chips
[*] Support for canfestival
```

### 4.4.2 LS1043ARDB or LS1046ARDB board

The following sections describe the hardware resource allocation for the LS1043ARDB or LS1046ARDB boards for implementing the supported features.

#### 4.4.2.1 Linux DTS

Remove cpu1, cpu2, cpu3 nodes on DTS, and remove all the devices that bare metal has used.

## 4.4.2.2 Memory configuration

This section describes the memory configuration for LS1043ARDB or LS1046ARDB boards.

The LS1043ARDB or LS1046ARDB boards have a 2GB size DDR. To use the baremetal framework, configure DDR into three partitions:

- 512M for core0 (Linux)
- 256M for core1(bare metal)
- 256M for core2(bare metal)
- 256M for core3(bare metal), and 256M for shared memory.

The configuration can be defined in the file include/configs/ls1043ardb.h.

```
#define CONFIG_SYS_DDR_SDRAM_SLAVE_SIZE (256 * 1024 * 1024)
#define CONFIG_SYS_DDR_SDRAM_MASTER_SIZE (512 * 1024 * 1024)
#define CONFIG_SYS_DDR_SDRAM_SHARE_RESERVE_SIZE (16 * 1024 * 1024)
#define CONFIG_SYS_DDR_SDRAM_SHARE_SIZE \ ((256 * 1024 * 1024) -
CONFIG_SYS_DDR_SDRAM_SHARE_RESERVE_SIZE)
```

#### NOTE

The memory configuration must be consistent with the U-Boot configuration of core0.

The memory configuration for bare metal is shown in the figure below.





The functions included in malloc.h in the table below can be used to allocate or free memory in program. Modify CONFIG SYS MALLOC LEN in include/configs/ls1043a common.h to change the maximum size of malloc.

| Table 16. | Memory | APIs | description |
|-----------|--------|------|-------------|
|-----------|--------|------|-------------|

| API name (type)           | Description                                                                                      |
|---------------------------|--------------------------------------------------------------------------------------------------|
| void_t* malloc (size_t n) | Allocates memory                                                                                 |
|                           | <ul> <li>"n" – length of allocated chunk</li> </ul>                                              |
|                           | Returns a pointer to the newly allocated chunk                                                   |
| void free (void *ptr)     | Releases the chunk of memory pointed to by ptr (where "ptr" is a pointer to the chunk of memory) |

Development based on baremetal framework

The GPIO for LS1043ARDB (or LS1046ARDB) has four GPIO controllers. You need to add a GPIO node in the file ls1043/6a-rdb.dts to assign a GPIO resource to different cores. The configuration can be done in the file arch/arm/dts/fsl-ls1043/6a-rdb.dts.

### 4.4.2.3 GPIO

LS1043/6A has four GPIO controllers. You can add a GPIO node in the file 1s1043/6a-rdb.dts to assign a GPIO resource to different cores. The configuration is in arch/arm/dts/fs1-1s1043/6a-rdb.dts. Use the command below to add a GPIO node:

```
&gpio2 {
    status = "okay";
};
```

## 4.4.2.4 I2C

This section describes how to configure the I2C bus on LS1028A, LS1043A, or LS1046A reference design boards.

The LS1043ARDB (or LS1028ARDB / LS1046ARDB) has four I2C controllers. You can configure the I2C bus using the ls1043ardb\_config.h (or ls1043ardb\_config.h) file using the commands below:

```
// include/configs/ls1043ardb_config.h:
#define CONFIG_SYS_I2C_MXC_I2C1 /* enable I2C bus 0 */
#define CONFIG_SYS_I2C_MXC_I2C2 /* enable I2C bus 1 */
#define CONFIG_SYS_I2C_MXC_I2C3 /* enable I2C bus 2 */
#define CONFIG_SYS_I2C_MXC_I2C4 /* enable I2C bus 3 */
#define CONFIG_SYS_I2C_MXC_I2C0_COREID 1
#define CONFIG_SYS_I2C_MXC_I2C1_COREID 2
#define CONFIG_SYS_I2C_MXC_I2C2_COREID 3
#define CONFIG_SYS_I2C_MXC_I2C3_COREID 1
```

The CONFIG\_SYS\_I2C\_MXC\_I2C0\_COREID defines the slave core that runs the I2C bus.

## 4.4.2.5 Hardware interrupts

LS1043A has twelve IRQs as external IO signals connected to interrupt the controller. These twelve IRQs can be used on bare metal cores. The ids for these signals, IRQ0-IRQ11 are: 163, 164, 165, 167, 168, 169, 177, 178, 179, 181, 182, and 183. GIC interrupt APIs are defined in asm/interrupt-gic.h. The following example shows how to register a hardware interrupt:

```
//register HW interrupt
void gic_irq_register(int irq_num, void (*irq_handle)(int));
void gic_set_target(u32 core_mask, unsigned long hw_irq);
void gic_set_type(unsigned long hw_irq);
```

## 4.4.2.6 IFC

The LS1043ARDB has an IFC flash device. Configure the IFC bus in ls1043ardb config.h using the command below:

#define CONFIG\_FSL\_IFC\_COREID 1

Here,  ${\tt config\_fsl\_ifc\_coreid}$  defines the slave core which runs this IFC bus.

### 4.4.2.7 QSPI

LS1046ARDB has a QSPI flash device. To configure the QSPI on ls1046ardb\_config.h, use the command below:

#define CONFIG FSL QSPI COREID 1

Here, the CONFIG FSL QSPI COREID defines the slave core that runs this QSPI.

#### 4.4.2.8 Ethernet

This section describes the Ethernet configuration settings for LS1043A or LS1046A reference design boards.

LS1043A or LS1046A has only one FMAN, so you need to remove the DPAA driver in Linux.

1. Disable the Linux DPAA driver using the settings below:

```
$make linux-menuconfig
Drivers --->
Staging drivers--->
< > Freescale Datapath Queue and Buffer management
```

2. Enter the Baremetal-Framework directory and then execute the commands below:

```
make menuconfig
ARM architecture --->
[*] Enable baremetal
[*] Enable fman for baremetal(1) FMAN1 is assigned to that
core
```

Configure FMAN to the specified core by modifying the FMAN1 is assigned to that core value, which is the default configuration, to core1.

#### 4.4.2.9 USB

This section describes the USB configuration setting for LS1043A and LS1046A reference design boards.

Both LS1043A and LS1046A have three DW3 USB controllers, we assign them to core1, core2 and core3 as the default setting. Re-configure them with command 'make menuconfig'.

```
ARM architecture --->
[*] Enable baremetal
[*] Enable USB for baremetal
(1) USB0 is assigned to core1
(2) USB1 is assigned to core2
(3) USB2 is assigned to core3
(3) USB Controller numbers
```

#### 4.4.2.10 PCIE

This section describes the PCIe configuration setting for LS1043A and LS1046A reference design boards.

Both LS1043A and LS1046A have three PCIe controllers, we assign them to core0, core1 and core2 as the default setting. Reconfigure them with command 'make menuconfig'.

```
ARM architecture --->
[*] Enable baremetal
```

Industrial IoT Baremetal Framework Developer Guide, Rev. 1.6, 08/2019

Development based on baremetal framework

```
(0) PCIe1 is assigned to core0
(1) PCIe2 is assigned to core1
(2) PCIe3 is assigned to core2
(3) PCIe Controller numbers
```

# 4.4.3 LS1028ARDB board

This section describes the ENETC configuration setting for LS1028A reference design boards.

### 4.4.3.1 ENETC

LS1028ARDB has only one ENETC controller in use, which is assigned to core1 as the default setting. The controller can be reconfigured by using the command, make menuconfig.

See the following:

```
ARM architecture --->
[*] Enable baremetal
[*] Enable ENETC for baremetal
  (1) Enetc1 is assigned to corel
   (1) ENETC Controller numbers
```

# Chapter 5 Revision history

The table below summarizes the revisions to this document.

| Date             | Version         | Topic cross-reference                                       | Change description                                                                                                   |
|------------------|-----------------|-------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------|
| 31/08/2019 1.6   |                 | Getting OpenIL                                              | Updated the Open IL Git tag to OpenIL-201908.                                                                        |
|                  | Getting started | Added support for i.mx6q-sabresd board.                     |                                                                                                                      |
|                  |                 |                                                             | Added the section describing hardware setup for i.mx6q-<br>sabresd board.                                            |
|                  |                 |                                                             | <ul> <li>Added the section Building the baremetal images for<br/>i.mx6q-sabresd board.</li> </ul>                    |
| 30/04/2019       | 1.5             | ENETC file                                                  | Added the section (a sample file for testing the ENETC feature).                                                     |
|                  |                 | LS1028ARDB board                                            | Added the section describing the ENETC configuration setting for LS1028ARDB                                          |
|                  |                 | Getting OpenIL                                              | Updated the Open IL Git tag.                                                                                         |
| 25/01/2019 1.4   |                 | Supported processors and boards                             | Added the support for LS1028ARDB. Modified and updated relevant sections throughout the document, where applicable.  |
|                  |                 | Getting OpenIL                                              | Updated the Open IL Git tag to OpenIL-u-boot-201901.                                                                 |
| 15/10/2018 1.3.1 | 1.3.1           | Ethernet                                                    | Updated the section, which describes Ethernet configuration settings for LS1043A or LS1046A reference design boards. |
|                  |                 | Getting OpenIL                                              | Updated the Open IL Git tag.                                                                                         |
|                  |                 | Ethernet                                                    | Updated the section.                                                                                                 |
| 31/08/2018       | 1.3             | Building the baremetal<br>images from U-Boot source<br>code | OpenIL Git tag updated to OpenIL-u-boot-201808.                                                                      |
|                  |                 | Getting OpenIL                                              | Updated the section.                                                                                                 |
|                  |                 | USB file, PCIe file, and CAN file.                          | Support for the USB, PCIE, and FlexCAN features enabled.<br>Added the sections in Section 4.2, Example software.     |
|                  |                 | USB, PCIe, and FlexCAN                                      | Added these sections to describe the hardware resource allocation for LS1021A-IoT board.                             |
|                  |                 | LS1043ARDB or<br>LS1046ARDB board                           | Added the section for hardware resource allocation for these boards.                                                 |
| 31/05/2018       | 1.2             | Getting started                                             | Added information relevant for LS1043ARDB and LS1046ARDB boards and for Open IL v1.2 support.                        |
|                  |                 | Ethernet                                                    | Updated the section.                                                                                                 |
| 02/04/2018       | 1.1             | -                                                           | Initial release aligned to Open IL v1.1.                                                                             |

#### Table 17. Document revision history

#### How To Reach Us

Home Page:

nxp.com

Web Support:

nxp.com/support

Information in this document is provided solely to enable system and software implementers to use NXP products. There are no express or implied copyright licenses granted hereunder to design or fabricate any integrated circuits based on the information in this document. NXP reserves the right to make changes without further notice to any products herein.

NXP makes no warranty, representation, or guarantee regarding the suitability of its products for any particular purpose, nor does NXP assume any liability arising out of the application or use of any product or circuit, and specifically disclaims any and all liability, including without limitation consequential or incidental damages. "Typical" parameters that may be provided in NXP data sheets and/or specifications can and do vary in different applications, and actual performance may vary over time. All operating parameters, including "typicals," must be validated for each customer application by customer's technical experts. NXP does not convey any license under its patent rights nor the rights of others. NXP sells products pursuant to standard terms and conditions of sale, which can be found at the following address: nxp.com/ SalesTermsandConditions.

While NXP has implemented advanced security features, all products may be subject to unidentified vulnerabilities. Customers are responsible for the design and operation of their applications and products to reduce the effect of these vulnerabilities on customer's applications and products, and NXP accepts no liability for any vulnerability that is discovered. Customers should implement appropriate design and operating safeguards to minimize the risks associated with their applications and products.

NXP, the NXP logo, Freescale, the Freescale logo, Layerscape, and QorlQ are trademarks of NXP B.V. Arm and Cortex are the registered trademarks of Arm Limited (or its subsidiaries) in the EU and/or elsewhere. All other product or service names are the property of their respective owners. All rights reserved.

© 2019 NXP B.V.

© NXP B.V. 2019.

#### All rights reserved.

For more information, please visit: http://www.nxp.com For sales office addresses, please send an email to: salesaddresses@nxp.com

> Date of release: 08/2019 Document identifier: IIoTBaremetalUG

