



# FETMX8MP-C OKMX8MP-C

Embedded Development Platform

**Hardware Manual**

Rev. 1.1

# CONTENTS

|                                                           |          |
|-----------------------------------------------------------|----------|
| <b>Copyright Notice</b>                                   | <b>2</b> |
| <b>Overview</b>                                           | <b>3</b> |
| <b>Application</b>                                        | <b>4</b> |
| <b>Revision History</b>                                   | <b>5</b> |
| <b>1. NXP i.MX 8M Plus Series Description</b>             | <b>6</b> |
| 1.1 Overview . . . . .                                    | 6        |
| 1.2 Features . . . . .                                    | 6        |
| <b>2. OKMX8MPQ-C Development Board M Core Description</b> | <b>8</b> |
| 2.1 Overview . . . . .                                    | 8        |
| 2.2 M Core Interfaces . . . . .                           | 8        |
| <b>3. Software Design Bare Metal Routines</b>             | <b>9</b> |
| 3.1 GPIO . . . . .                                        | 9        |
| 3.1.1 Hardware Connection . . . . .                       | 9        |
| 3.1.2 Software Implementation . . . . .                   | 10       |
| 3.1.3 Experimental Phenomena . . . . .                    | 11       |
| 3.2 UART . . . . .                                        | 11       |
| 3.2.1 Hardware Connection . . . . .                       | 11       |
| 3.2.2 Software Implementation . . . . .                   | 12       |
| 3.2.3 Experimental Phenomena . . . . .                    | 13       |
| 3.3 I2C . . . . .                                         | 13       |
| 3.3.1 Hardware Connection . . . . .                       | 13       |
| 3.3.2 Software Implementation . . . . .                   | 14       |
| 3.3.3 Experimental Phenomena . . . . .                    | 14       |
| 3.4 CAN . . . . .                                         | 15       |
| 3.4.1 Hardware Connection . . . . .                       | 15       |
| 3.4.2 Software Implementation . . . . .                   | 16       |
| 3.4.3 Experimental Phenomena . . . . .                    | 17       |
| 3.5 CAN-FD . . . . .                                      | 18       |
| 3.5.1 Hardware Connection . . . . .                       | 18       |
| 3.5.2 Software Implementation . . . . .                   | 19       |
| 3.5.3 Experimental Phenomena . . . . .                    | 20       |
| 3.6 SPI . . . . .                                         | 21       |
| 3.6.1 Hardware Connection . . . . .                       | 22       |
| 3.6.2 Software Implementation . . . . .                   | 22       |
| 3.6.3 Experimental Phenomena . . . . .                    | 24       |
| 3.7 GPT-Timer . . . . .                                   | 25       |
| 3.7.1 Hardware Connection . . . . .                       | 25       |
| 3.7.2 Software Implementation . . . . .                   | 25       |
| 3.7.3 Experimental Phenomena . . . . .                    | 26       |
| 3.8 GPT-Capture . . . . .                                 | 27       |
| 3.8.1 Hardware Connection . . . . .                       | 27       |

|                                                  |           |
|--------------------------------------------------|-----------|
| 3.8.2 Software Implementation . . . . .          | 27        |
| 3.8.3 Experimental Phenomena . . . . .           | 28        |
| 3.9 PWM . . . . .                                | 29        |
| 3.9.1 Hardware Connection . . . . .              | 29        |
| 3.9.2 Software Implementation . . . . .          | 29        |
| 3.9.3 Experimental Phenomena . . . . .           | 31        |
| 3.10 Audio SAI . . . . .                         | 31        |
| 3.10.1 Hardware Connection . . . . .             | 31        |
| 3.10.2 Software Implementation . . . . .         | 32        |
| 3.10.3 Experimental Phenomena . . . . .          | 34        |
| 3.11 Audio Rate Conversion ASRC . . . . .        | 34        |
| 3.11.1 Hardware Connection . . . . .             | 34        |
| 3.11.2 Software Implementation . . . . .         | 35        |
| 3.11.3 Experimental Phenomena . . . . .          | 37        |
| 3.12 Watchdog . . . . .                          | 37        |
| 3.12.1 Hardware Connection . . . . .             | 37        |
| 3.12.2 Software Implementation . . . . .         | 37        |
| 3.12.3 Experimental Phenomena . . . . .          | 39        |
| 3.13 Temperature Monitor TMU . . . . .           | 40        |
| 3.13.1 Hardware Connection . . . . .             | 40        |
| 3.13.2 Software Implementation . . . . .         | 40        |
| 3.13.3 Experimental Phenomena . . . . .          | 41        |
| 3.14 SDMA . . . . .                              | 41        |
| 3.14.1 Hardware Connection . . . . .             | 41        |
| 3.14.2 Software Implementation . . . . .         | 42        |
| 3.14.3 Experimental Phenomena . . . . .          | 42        |
| 3.15 RDC . . . . .                               | 43        |
| 3.15.1 Hardware Connection . . . . .             | 43        |
| 3.15.2 Software Implementation . . . . .         | 43        |
| 3.15.3 Experimental Phenomena . . . . .          | 44        |
| 3.16 IPC . . . . .                               | 45        |
| 3.16.1 Hardware Connection . . . . .             | 45        |
| 3.16.2 Software Implementation . . . . .         | 45        |
| 3.16.3 Experimental Phenomena . . . . .          | 46        |
| <b>4. Freertos Routines Design with Software</b> | <b>48</b> |
| 4.1 Freertos-generic . . . . .                   | 48        |
| 4.1.1 Hardware Connection . . . . .              | 48        |
| 4.1.2 Software Implementation . . . . .          | 48        |
| 4.1.3 Experimental Phenomena . . . . .           | 51        |
| 4.2 Freertos-Peripheral . . . . .                | 52        |
| 4.2.1 Hardware Connection . . . . .              | 52        |
| 4.2.2 Software Implementation . . . . .          | 53        |
| 4.2.3 Experimental Phenomena . . . . .           | 55        |
| <b>5. Use of OK8MP platform M-core SDK</b>       | <b>56</b> |

Document classification:  Top secret  Secret  Internal information  Open

---

**CHAPTER  
ONE**

---

**COPYRIGHT NOTICE**

The copyright of this manual belongs to Baoding Folinx Embedded Technology Co., Ltd. Without the written permission of our company, no organizations or individuals have the right to copy, distribute, or reproduce any part of this manual in any form, and violators will be held legally responsible.

Forlinx adheres to copyrights of all graphics and texts used in all publications in original or license-free forms.

The drivers and utilities used for the components are subject to the copyrights of the respective manufacturers. The license conditions of the respective manufacturer are to be adhered to. Related license expenses for the operating system and applications should be calculated/declared separately by the related party or its representatives.

---

CHAPTER  
TWO

---

## OVERVIEW

This manual is intended to familiarize users with the product quickly, understand the functions and testing methods of the M core interface. It mainly describes the overview of the M core interface on the development board, source code analysis and testing methods, as well as how to troubleshoot some issues that may arise during usage. In the process of testing, some commands are annotated to facilitate the user's understanding, mainly for practical use. Please refer to the MCU User Compilation Manual provided by Forlinx to view application program compilation, development environment building, program download and simulation, etc.

There are total five chapters:

- Chapter 1. provides an overview of the NXP i.MX 8M Plus, giving a brief introduction to the overall architecture and features of the i.MX 8M Plus processor;
- Chapter 2. is the overall overview of the M-core on the development board, which briefly introduces the resources of the M-core;
- Chapter 3. is the M core interface bare machine (no system) use function introduction, program analysis and test;
- Chapter 4. is the introduction of the function of Freertos system of M-core interface, program analysis and test;
- Chapter 5. is an introduction to the M core SDK usage.

| Format                               | Meaning                                                                |
|--------------------------------------|------------------------------------------------------------------------|
| Blue font on gray background         | Refers to commands entered at the command line(Manual input required). |
| Black font on gray background        | Serial port output message after entering a command                    |
| <b>Bold black on gray background</b> | Key information in the serial port output message                      |
| //                                   | Interpretation of input instructions or output information             |

---

**CHAPTER  
THREE**

---

**APPLICATION**

This software manual is applicable to the OKMX8MPQ-C platform Linux 5.4.70 operating system of Forlinx.

---

**CHAPTER  
FOUR**

---

**REVISION HISTORY**

| Date       | Version | Revision History                            |
|------------|---------|---------------------------------------------|
| 20/04/2023 | v1.0    | OKMX8MPQ-C M7 User's Manual Initial Version |

## **1. NXP I.MX 8M PLUS SERIES DESCRIPTION**

### **1.1 Overview**

It features a 4 x Cortex-A53+Cortex-M7 heterogeneous architecture, with the M7 core operating at a high frequency of up to 800 MHz. It supports dual floating-point operations, enabling rapid program loading and enhanced real-time signal response. The Co-processor M7 core serves as the MCU to simulate real-time control, catering to the industrial control, communications networking, and other complex application scenarios that require real-time control. This helps to enhance the safety and dependability of various application scenarios.

### **1.2 Features**

- High-performance Arm® Cortex® -M7, 800MHz operating frequency
- Double floating point budget
- 32 KB instruction cache
- 32 KB data cache
- 256 KB TCM



## 2. OKMX8MPQ-C DEVELOPMENT BOARD M CORE DESCRIPTION

### 2.1 Overview

OKMX8MPQ-C is a cost-effective development board developed by Forlinx based on the industrial-grade NXP i.MX 8M Plus processor. The OKMX8MPQ-C is composed of a SoM and a carrier board. The carrier board features a wealth of peripheral resource module interfaces and comes with detailed module demonstration routines. For detailed information about the SoM and carrier board, please refer to the document OKMX8MPQ-C\_Linux User Manual. This manual primarily introduces the resources and usage of the M-core.

### 2.2 M Core Interfaces

| Function | Description                               | Table of Contents                          |
|----------|-------------------------------------------|--------------------------------------------|
| GPIO     | User's light                              | boards\evkmimx8mp\driver_examples\gpio     |
| UART     | Debugging serial port                     | \boards\evkmimx8mp\driver_examples\uart    |
| I2C      | RTC                                       | \boards\evkmimx8mp\driver_examples\i2c     |
| CAN      | Master-slave pair test                    | \boards\evkmimx8mp\driver_examples\flexcan |
| CAN-FD   | Master-slave pair test                    | \boards\evkmimx8mp\driver_examples\canfd   |
| SPI      | Master-slave pair test                    | \boards\evkmimx8mp\driver_examples\ecspi   |
| GPT      | Timer and input capture                   | \boards\evkmimx8mp\driver_examples\gpt     |
| PWM      | Waveform output                           | \boards\evkmimx8mp\driver_examples\pwm     |
| SAI      | Recording and playback                    | \boards\evkmimx8mp\driver_examples\sai     |
| ASRC     | Audio sample rate conversion              | \boards\evkmimx8mp\driver_examples\asrc    |
| WDOG     | Watchdog                                  | \boards\evkmimx8mp\driver_examples\wdog    |
| TMU      | Processor temperature detection           | \boards\evkmimx8mp\driver_examples\tmu     |
| SDMA     | Memory-to-memory copy                     | \boards\evkmimx8mp\driver_examples\sdma    |
| RDC      | Multi-core peripherals, memory allocation | \boards\evkmimx8mp\driver_examples\rdc     |
| RPMMSG   | Multi-core communication                  | \boards\evkmimx8mp\multicore_examples      |
| FreeRTOS | Examples of real-time systems             | \boards\evkmimx8mp\rtos_examples           |

### 3. SOFTWARE DESIGN BARE METAL ROUTINES

#### 3.1 GPIO

The OKMX8MPQ-C development board supports 2 GPIO output (running water light).

##### 3.1.1 Hardware Connection

The demo does not have any special hardware connections, only utilizing the carrier board's 2 x LED.





### 3.1.2 Software Implementation

#### (1) Initialization

GPIO initialization mainly includes pin and mode configuration.

The details are as follows:

Pin configuration: LED1 is connected to pin GPIO5\_08, and LED2 is connected to pin GPIO5\_09

```
IOMUXC_SetPinMux(IOMUXC_ECSPI1_MISO_GPIO5_IO08, 0U); // LED1
```

```
IOMUXC_SetPinMux(IOMUXC_ECSPI1_SS0_GPIO5_IO09, 0U); // LED2
```

Mode Configuration: Set two GPIOs as output pins

```
gpio_pin_config_t led_config = {kGPIO_DigitalOutput, 0, kGPIO_NoInvert}; //Output mode
```

```
GPIO_PinInit(EXAMPLE_LED1_GPIO, EXAMPLE_LED1_GPIO_PIN, &led_config)
```

```
GPIO_PinInit(EXAMPLE_LED2_GPIO, EXAMPLE_LED2_GPIO_PIN, &led_config)
```

#### (2) Execution process

After the GPIO initialization is completed, turn over the level of the GPIO every 1 second to control the on and off of the indicator light.

```
while (1)
```

```
{
```

```
SDK_DelayAtLeastUs(1000000, SDK_DEVICE_MAXIMUM_CPU_CLOCK_FREQUENCY);
```

```
if (g_pinSet)
```

```
{
```

```
GPIO_PinWrite(EXAMPLE_LED1_GPIO, EXAMPLE_LED1_GPIO_PIN, 0U);
```

```

GPIO_PinWrite(EXAMPLE_LED2_GPIO, EXAMPLE_LED2_GPIO_PIN, 0U);
g_pinSet = false;
}
else
{
    GPIO_PinWrite(EXAMPLE_LED1_GPIO, EXAMPLE_LED1_GPIO_PIN, 1U);
    GPIO_PinWrite(EXAMPLE_LED2_GPIO, EXAMPLE_LED2_GPIO_PIN, 1U);
    g_pinSet = true;
}
}

```

### 3.1.3 Experimental Phenomena

- (1) After the compilation is completed, as described in Chapter 3.1.1 of “OKMX8MPQ-C\_MCU \_ User’s Compilation Manual-V1.0”, manually load the M-core program in uboot;
- (2) M-core debugging serial port displays the information shown below, and you will observe 2 LEDs blinking at the same time.



## 3.2 UART

In the M-core SDK package, there are 7 serial port examples. These include “auto\_baudrate\_detect” for automatic baud rate detection, “polling” for polling-based sending and receiving, and “sdma\_transfer” and “idle\_detect\_sdma\_transfer” for DMA-based sending and receiving. Interrupt, interrupt \_ transfer, and interrupt \_ rb \_ transfer are interrupt methods to complete the sending and receiving routine. We primarily introduce interrupt routines.

### 3.2.1 Hardware Connection

By using a USB to Type-C connection, you can connect the debug port of a computer and a development board. The computer’s Device Manager will generate two serial ports. One serial port is for debugging the A-core Linux, and the other one is for debugging the M-core. Serial port configuration: Baud rate is 115200, data bits are 8, no flow control, stop bits are



1.

### 3.2.2 Software Implementation

#### (1) Initialization

UART initialization typically includes the following steps:

The details are as follows:

Clock Configuration: Select UART4 clock as 80MHZ.

```
CLOCK_SetRootMux(kCLOCK_RootUart4, kCLOCK_UartRootmuxSysPll1Div10); // Use SysPLL1 10 frequency 80MHZ
```

```
CLOCK_SetRootDivider(kCLOCK_RootUart4, 1U, 1U); // uart4 divided by 1 80MHZ/ 1= 80MHZ
```

Pin configuration: select the transceiver pin of UART4.

```
IOMUXC_SetPinMux(IOMUXC_UART4_RXD_UART4_RX, OU);
```

```
IOMUXC_SetPinMux(IOMUXC_UART4_TXD_UART4_TX, OU);
```

Communication configuration: Baud rate is 115 200, no check, data bit is 8, no flow control, stop bit is 1.

```
config->baudRate_Bps = 115200U; // Baud rate
```

```
config->parityMode = kUART_ParityDisabled; //No parity
```

```
config->dataBitsCount = kUART_EightDataBits; // 8-bit data bits
```

```
config->stopBitCount = kUART_OneStopBit; // 1 stop bit
```

```
config->enableRxRTS = false;
```

```
config->enableTxCTS = false;
```

Interrupt enable: configures and enables the receive completion and exception overload interrupts.

```
UART_EnableInterrupts(DEMO_UART, kUART_RxDataReadyEnable | kUART_RxOverrunEnable);
```

```
EnableIRQ(DEMO_IRQn);
```

#### (2) Execution process

After initialization, if data is received, an interrupt is triggered and the received data is resent.

```
while ((UART_GetStatusFlag(DEMO_UART, kUART_TxReadyFlag)) && (rxIndex != txIndex))
{
    UART_WriteByte(DEMO_UART, demoRingBuffer[txIndex]);

    txIndex++;
    txIndex %= DEMO_RING_BUFFER_SIZE;
}
```

### 3.2.3 Experimental Phenomena

- (1) After the compilation is completed, as described in Chapter 3.1.1 of “OKMX8MPQ-C\_MCU \_ User’s Compilation Manual-V1.0”, manually load the M-core program in uboot;
- (2) Input any character in the M core debugging serial port, trigger the receiving interrupt, and then send it out. The typed character can be seen in the M core debugging serial port.



## 3.3 I2C

In the M Core SDK package, there are three serial port routines. Among them, “polling\_b2b\_transfer” is a polling-based example for bidirectional data transfer between a master and a slave device. “interrupt\_b2b\_transfer” is an example that uses interrupts for bidirectional data transfer between a master and a slave device. interrupt\_PCF8563 controls the RTC routine using interrupt mode. We focus on the interrupt\_PCF8563 routine.

### 3.3.1 Hardware Connection

This routine mainly uses the RTC chip PCF8563 on the development board, and there is no other special connection.



### 3.3.2 Software Implementation

#### (1) Initialization

I2C initialization mainly includes clock configuration, pin configuration, communication configuration and interrupt configuration.

The details are as follows:

Clock Configuration: Select I2C3 clock as 16MHZ.

```
CLOCK_SetRootMux(kCLOCK_RootI2c3, kCLOCK_I2cRootmuxSysPll1Div5); // Set I2C clock source160MHZ
```

```
CLOCK_SetRootDivider(kCLOCK_RootI2c3, 1U, 10U); // After dividing the frequency by 10, the I2C bus clock is 160 MHz / 10 = 16 MHz.
```

Pin configuration: select the transceiver pin of I2C3.

```
IOMUXC_SetPinMux(IOMUXC_I2C3_SCL_I2C3_SCL, 1U);
```

```
IOMUXC_SetPinMux(IOMUXC_I2C3_SDA_I2C3_SDA, 1U);
```

Communication configuration: Baud rate is 100K, master mode, PCF8563 address is 0x51.

```
masterConfig->baudRate_Bps = 100000U; // I2C bus baud rate of 100K
```

```
masterConfig->enableMaster = true; // Set master station mode
```

```
#define PCF8563_ADDR 0x51
```

Interrupt Enable: install and enable interrupts.

```
s_i2cMasterIsr = I2C_MasterTransferHandleIRQ; // Install interrupt
```

```
(void)EnableIRQ(s_i2cIrqqs[instance]); // Enable interrupt
```

#### (2) Execution process

After initialization is complete, wait for commands. If it's a command to set the time, retrieve the time value via the debug serial port and update it to the RTC chip through an I2C write command. If reading the time command, use the I2C read command to obtain RTC time, as displayed via debug serial port.

```
PRINTF("\r\n Please input 'S/R' ? If set time ,press 'S' and read time press 'R' \r\n );
```

```
ch = 0;
```

```
ch = GETCHAR();
```

```
PUTCHAR(ch);
```

```
if((ch == ' s') || (ch == ' S')) // set means setting time
```

```
{
```

```
get_input_time();
```

```
PCF8563_set_time();
```

```
}
```

```
else if((ch == ' r') || (ch == ' R')) // read means reading time
```

```
{
```

```
PCF8563_get_time();
```

```
}
```

### 3.3.3 Experimental Phenomena

(1) After the compilation is completed, as described in Chapter 3.1.1 of “OKMX8MPQ-C\_MCU \_ User’s Compilation Manual-V1.0”, manually load the M-core program in uboot;

(2) Debugging via the M-core serial port, after entering “ s ”, you can set the time. Enter the command “ 2023-01-18 15:34:06 ” to successfully set the time;

(3) Debugging via the M-core serial port, after entering “ r ” , you can read the time, observing it continuously run-

```
Please input 'S/R'? If set time ,press 'S' and read time press 'R'
r
Read datetime from PCF8563: 2023- 1-18 15:34: 6 week: 3

Please input 'S/R'? If set time ,press 'S' and read time press 'R'
s
Please set time as the format:'2017-05-05 15:30:30'
:2023-01-18 15:35:00
write time to PCF8563 sucess

Please input 'S/R'? If set time ,press 'S' and read time press 'R'
r
Read datetime from PCF8563: 2023- 1-18 15:35: 1 week: 3

Please input 'S/R'? If set time ,press 'S' and read time press 'R'
ning.
```

## 3.4 CAN

M-core SDK package contains four CAN examples. Among them, “loopbac” and “loopback\_transfer” are examples of self-loopback sending and receiving. “interrupt\_transfer” demonstrates master-slave communication using interrupt mode. The ping\_pong\_buffer\_transfer is to complete the master-slave receiving and sending in the interrupt buffer mode. We will focus on the interrupt\_transfer routine.

### 3.4.1 Hardware Connection

This routine demonstrates the process of CAN master-slave communication, which requires two 8mp development boards, using DuPont cables to connect the H and L of CAN1.



### 3.4.2 Software Implementation

#### (1) Initialization

CAN initialization typically includes the following steps:

The details are as follows:

Clock Configuration: Select CAN1 clock as 80MHz.

```
CLOCK_SetRootMux(kCLOCK_RootFlexCan1, kCLOCK_FlexCanRootmuxSysPll1); // CAN1 clock source is: SYSTEM PLL1 800MHZ
```

```
CLOCK_SetRootDivider(kCLOCK_RootFlexCan1, 2U, 5U); // Divide-by-10 to 80MHz
```

Pin configuration: select the transceiver pin of CAN1.

```
IOMUXC_SetPinMux(IOMUXC_SAI2_TXC_CAN1_RX, 0U);
```

```
IOMUXC_SetPinMux(IOMUXC_SAI2_RXC_CAN1_TX, 0U);
```

Baud rate: Standard CAN speed of 100 Kbit/s. Subsequent programs allocate segment settings such as seg1 and seg2 based on the bus clock and the configured baud rate.

```
pConfig->bitRate = 100000U; // The baud rate of CAN control area is 100KB
```

Turn off the self-loop: If the self-loop is turned on, the CAN1 data will loop back in the chip and will not reach the external pin. During program debugging, the interference of the external terminal can be eliminated. However, in real applications, it is necessary to turn off the self-loop and send and receive data from the external pin.

```
pConfig->enableLoopBack = false; // Do not loop, use external pin
```

Frame format: 11-bit standard data frame is used, and the user can also try the extended frame later. You need to set your own ID so that other devices on the bus can identify it.

```
mbConfig.format = kFLEXCAN_FrameFormatStandard; // 11-bit standard frame, non-extended frame
```

```
mbConfig.type = kFLEXCAN_FrameTypeData; // Data frame non-remote frame
```

---

```
mbConfig.id    = FLEXCAN_ID_STD(rxIdentifier); // The frame ID is used to distinguish between different devices on the
but
```

Receive filtering: The user can set the receive filtering rules, so that only the data of a specific frame ID can be received to reduce the amount of data processed by the application.

```
rxIdentifier = 0;
```

```
FLEXCAN_SetRxMbGlobalMask(EXAMPLE_CAN, FLEXCAN_RX_MB_STD_MASK(rxIdentifier, 0, 0));//Receive all ID data
```

Interrupt Enable: install and enable interrupts.

```
FLEXCAN_EnableInterrupts(base,          (uint32_t)kFLEXCAN_BusOffInterruptEnable      |      (uint32_t)kFLEX
CAN_ErrorInterruptEnable |(uint32_t)kFLEXCAN_RxWarningInterruptEnable | (uint32_t)kFLEXCAN_TxWarningInterruptEnable |(uint32_t)
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanRxWarningIRQ[instance]));
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanTxWarningIRQ[instance]));
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanWakeUpIRQ[instance]));
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanErrorIRQ[instance]));
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanBusOffIRQ[instance]));
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanMbIRQ[instance]));
```

## (2) Execution process

After initialization is complete, wait for a command. On one development board, entering “B” via the serial port sets it as a CAN slave station awaiting receiving. Upon completing data receiving, print a message and send back the received data.

```
FLEXCAN_TransferReceiveNonBlocking(EXAMPLE_CAN, &flexcanHandle, &rxXfer);// Receive data
```

```
LOG_INFO("Rx MB ID: 0x%3x, Rx MB data: 0x%x, Time stamp: %d\r\n", frame.id >> CAN_ID_STD_SHIFT,frame.dataByte0,frame.timestamp);
```

```
FLEXCAN_TransferSendNonBlocking(EXAMPLE_CAN, &flexcanHandle, &txXfer); // Return data
```

Enter ‘A’ via the serial port, followed by any key. It is actively transmitted as a can master station, and then receives the return data from the slave station. After completing a transfer, enter any key to trigger the next transfer.

```
FLEXCAN_TransferSendNonBlocking(EXAMPLE_CAN, &flexcanHandle, &txXfer) // Send data
```

```
FLEXCAN_TransferReceiveNonBlocking(EXAMPLE_CAN, &flexcanHandle, &rxXfer); // Receive data
```

```
LOG_INFO("Rx MB ID: 0x%3x, Rx MB data: 0x%x, Time stamp: %d\r\n", frame.id >> CAN_ID_STD_SHIFT,frame.dataByte0,frame.timestamp);
```

```
LOG_INFO(" Press any key to trigger the next transmission!\r\n\r\n");
```

### 3.4.3 Experimental Phenomena

(1) After the compilation is completed, as described in Section 3.1.1 of the “OKMX8MPQ-C\_MCU - User Compilation Manual - V1.0” , manually load the M-core program in U-Boot;

(2) Debug the serial port in the M core. Input data B into the serial port of the development board which will act as the CAN slave station and wait for incoming data. Once data receiving is complete, print a message and return the received data;

```
***** FLEXCAN Interrupt EXAMPLE *****
Message format: Standard (11 bit id)
Message buffer 9 used for Rx.
Message buffer 8 used for Tx.
Interrupt Mode: Enabled
Operation Mode: TX and RX --> Normal
*****
Please select local node as A or B:
Note: Node B should start first.
Node:b
bitRate_Bps:100000,quantum:100000,MAX_EPRESDIV:1023,sourceClock_Hz:80000000
CAN BRS:prediv:9 propseg:58 seg1:9 seg2:9 jumpwidth:9
Start to Wait data from Node A

Rx MB ID: 0x321, Rx MB data: 0x0, Time stamp: 39082
Wait Node A to trigger the next transmission!
```

- (3) Debug the serial port in the M core, input A in the serial port of the other development board, input any key as the active transmission of the CAN master station, and then receive the return data from the slave station. After completing a transfer, enter any key to trigger the next transfer.

```
***** FLEXCAN Interrupt EXAMPLE *****
Message format: Standard (11 bit id)
Message buffer 9 used for Rx.
Message buffer 8 used for Tx.
Interrupt Mode: Enabled
Operation Mode: TX and RX --> Normal
*****
Please select local node as A or B:
Note: Node B should start first.
Node:a
bitRate_Bps:100000,quantum:100000,MAX_EPRESDIV:1023,sourceClock_Hz:80000000
CAN BRS:prediv:9 propseg:58 seg1:9 seg2:9 jumpwidth:9
Press any key to trigger one-shot transmission

Rx MB ID: 0x123, Rx MB data: 0x0, Time stamp: 42501
Press any key to trigger the next transmission!
```

## 3.5 CAN-FD

There are four CAN-FD routines in the M-core SDK package, in which the loopback and loopback \_ transfer are self-loopback and self-receiving autonomous routines, and the interrupt \_ transfer is an interrupt mode to complete the master-slave transceiver routines. The ping \_ pong \_ buffer \_ transfer is to complete the master-slave receiving and sending in the interrupt buffer mode. We will focus on the interrupt \_ transfer routine.

### 3.5.1 Hardware Connection

This routine demonstrates the process of CAN-FD master-slave communication. Two 8mp development boards are required, and DuPont cable is used to connect the H and L of the two development boards CAN1.



### 3.5.2 Software Implementation

#### (1) Initialization

CAN initialization typically includes the following steps:

The details are as follows:

Clock Configuration: Select CAN1 clock as 80MHz.

```
CLOCK_SetRootMux(kCLOCK_RootFlexCan1, kCLOCK_FlexCanRootmuxSysPll1); // CAN1 clock source is: SYSTEM PLL1 800MHZ
```

```
CLOCK_SetRootDivider(kCLOCK_RootFlexCan1, 2U, 5U); // Divide-by-10 to 80MHz
```

Pin configuration: select the transceiver pin of CAN1.

```
IOMUXC_SetPinMux(IOMUXC_SAI2_TXC_CAN1_RX, 0U);
```

```
IOMUXC_SetPinMux(IOMUXC_SAI2_RXC_CAN1_TX, 0U);
```

Baud rate: \*\* CAN Baud rate: CAN-FD supports variable rate, that is, the baud rate of the control area and the data area can be different. The maximum baud rate of the control area is 1Mbit/s, and the maximum baud rate of the data area is 8Mbit/s. The subsequent program assigns the values of seg1, seg2, etc. set for the time period according to the bus clock and the set baud rate.

```
pConfig->bitRate = 1000000U; // Baud rate of CAN-FD control area is 1m
```

```
pConfig->bitRateFD = 8000000U; // Baud rate of CAN-FD data area is 8m
```

CAN-FD enable: In addition to enabling the CAN-FD, the variable baud rate also needs to be enabled, otherwise the maximum rate of the data area is the same as that of the control area, and the maximum rate is 1Mbit/s.

```
base->MCR |= CAN_MCR_FDEN_MASK; // CAN-FD enable
```

```
fdctrl |= CAN_FDCTRL_FDRATE_MASK; // Variable baud rate enable
```

Turn off the self-loop: If the self-loop is turned on, the CAN1 data will loop back in the chip and will not reach the external pin. During program debugging, the interference of the external terminal can be eliminated. However, in real applications, it is necessary to turn off the self-loop and send and receive data from the external pin.

```
pConfig->enableLoopBack = false; // Do not loop, use external pin
```

Frame format: This time we use the 11-bit standard data frame, and the customer can also try the extended frame later. You need to set your own ID so that other devices on the bus can identify it.

```
mbConfig.format = kFLEXCAN_FrameFormatStandard; // 11-bit standard frame, non-extended frame
```

```
mbConfig.type = kFLEXCAN_FrameTypeData; // Data frame non-remote frame
```

```
mbConfig.id = FLEXCAN_ID_STD(rxIdentifier); // The frame ID is used to distinguish between different devices on the bus
```

Receive filtering: The user can set the receive filtering rules, so that only the data of a specific frame ID can be received to reduce the amount of data processed by the application.

```
rxIdentifier = 0;
```

```
FLEXCAN_SetRxMbGlobalMask(EXAMPLE_CAN, FLEXCAN_RX_MB_STD_MASK(rxIdentifier, 0, 0)); // Receive all ID data
```

Interrupt Enable: install and enable interrupts.

```
FLEXCAN_EnableInterrupts(base, (uint32_t)kFLEXCAN_BusOffInterruptEnable | (uint32_t)kFLEXCAN_ErrorInterruptEnable | (uint32_t)kFLEXCAN_RxWarningInterruptEnable | (uint32_t)kFLEXCAN_TxWarningInterruptEnable | (uint32_t)kFLEXCAN_WakeUpInterruptEnable);
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanRxWarningIRQ[instance]));
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanTxWarningIRQ[instance]));
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanWakeUpIRQ[instance]));
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanErrorIRQ[instance]));
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanBusOffIRQ[instance]));
```

```
(void)EnableIRQ((IRQn_Type)(s_flexcanMbIRQ[instance]));
```

## (2) Execution process

After initialization is complete, wait for commands. On one of the development boards, input “B” through the serial port to configure it as a CAN-FD slave station that waits for data receiving. Upon completing data receiving, print a message and send the received data back.

```
FLEXCAN_TransferFDReceiveNonBlocking(EXAMPLE_CAN, &flexcanHandle, &rxXfer); // Receive data
```

```
LOG_INFO("Rx MB ID: 0x%3x, Rx MB data: 0x%x, Time stamp: %d\r\n", frame.id >> CAN_ID_STD_SHIFT, frame.dataByte0, frame.timestamp);
```

```
FLEXCAN_TransferFDSendNonBlocking(EXAMPLE_CAN, &flexcanHandle, &txXfer); // Return data
```

Input A in the serial port of the other development board as the active transmission of the CAN-FD master station, and then receive the return data from the slave station. After completing a transfer, enter any key to trigger the next transfer.

```
FLEXCAN_TransferSendNonBlocking(EXAMPLE_CAN, &flexcanHandle, &txXfer); // Send data
```

```
FLEXCAN_TransferFDReceiveNonBlocking(EXAMPLE_CAN, &flexcanHandle, &rxXfer); // Receive data
```

```
LOG_INFO("Rx MB ID: 0x%3x, Rx MB data: 0x%x, Time stamp: %d\r\n", frame.id >> CAN_ID_STD_SHIFT, frame.dataByte0, frame.timestamp);
```

```
LOG_INFO(" Press any key to trigger the next transmission!\r\n\r\n");
```

## 3.5.3 Experimental Phenomena

(1) After the compilation is completed, as described in Section 3.1.1 of the “OKMX8MPQ-C\_MCU - User Compilation Manual - V1.0”, manually load the M-core program in U-Boot;

(2) Debug the serial port in the M core. Input data B into the serial port of the development board which will act as the CAN-FD slave station and wait for incoming data. Once data receiving is complete, print a message and return the received data.

```
***** FLEXCAN Interrupt EXAMPLE *****
Message format: Standard (11 bit id)
Message buffer 9 used for Rx.
Message buffer 8 used for Tx.
Interrupt Mode: Enabled
Operation Mode: TX and RX --> Normal
*****
Please select local node as A or B:
Note: Node B should start first.
Node:b
bitRate_Bps:1000000,quantum:1000000,MAX_EPRESDIV:1023,sourceClock_Hz:80000000
CAN BRS:prediv:0 propseg:46 seg1:15 seg2:15 jumpwidth:15
Start to Wait data from Node A

Rx MB ID: 0x321, Rx MB data: 0x0, Time stamp: 27657
Wait Node A to trigger the next transmission!
```

- (3) Debug the serial port in the M core, input A in the serial port of the other development board, and then input any key as the active transmission of the CAN-FD master station, and then receive the return data from the slave station. After completing a transfer, enter any key to trigger the next transfer.

```
Node:***** FLEXCAN Interrupt EXAMPLE *****
Message format: Standard (11 bit id)
Message buffer 9 used for Rx.
Message buffer 8 used for Tx.
Interrupt Mode: Enabled
Operation Mode: TX and RX --> Normal
*****
Please select local node as A or B:
Note: Node B should start first.
Node:a
bitRate_Bps:1000000,quantum:1000000,MAX_EPRESDIV:1023,sourceClock_Hz:80000000
CAN BRS:prediv:0 propseg:46 seg1:15 seg2:15 jumpwidth:15
Press any key to trigger one-shot transmission

Rx MB ID: 0x123, Rx MB data: 0x0, Time stamp: 32337
Press any key to trigger the next transmission!
```

## 3.6 SPI

There are three SPI routines in the M-core SDK package, in which the `ecspi_loopback` is a self-loop self-receiving self-sending routine, and the `interrupt_B2B_transfer` is a master-slave receiving and sending routine completed in the interrupt mode. The `polling_b2b_transfer` completes the master-slave transceiver routine in the polling mode. We focus on the `interrupt_b2b_transfer` routine.

### 3.6.1 Hardware Connection

This routine demonstrates the process of SPI master-slave communication. Two 8 MP development boards are required. Connect the two development boards SPI with Dupont cable.



Connect the SPI of the two development boards one-to-one using DuPont wires with the following wire sequence:

| Development Board 1 — SPI Mast Mode | Development Board Location | Development Board 2 — SPI Slave Mode |
|-------------------------------------|----------------------------|--------------------------------------|
| Pin Name                            |                            | Pin Name                             |
| MISO                                | P40-10                     | MISO                                 |
| MOSI                                | P40-8                      | MOSI                                 |
| SCK                                 | P40-1                      | SCK                                  |
| SSO                                 | P40-3                      | SSO                                  |
| GND                                 | P40-4/P40-7                | GND                                  |

### 3.6.2 Software Implementation

#### (1) SPI main initialization

SPI initialization mainly includes initializing the bus clock, pins, and corresponding registers.

The details are as follows:

Bus Clock: The SPI bus clock is sourced at 800 MHz, divided by 10 to 80 MHz.

```
CLOCK_SetRootMux(kCLOCK_RootEcspi2, kCLOCK_EcspiRootmuxSysPll1); //SPI bus clock PLL1-800MHZ
```

```
CLOCK_SetRootDivider(kCLOCK_RootEcspi2, 2U, 5U); //The frequency division factor is 2 * 5 = 10, and the bus clock is set to 80MHz
```

Pin configuration: Select the four pins of SPI2.

```
IOMUXC_SetPinMux(IOMUXC_ECSPI2_MISO_ECSPI2_MISO, 0U); // SPI2-MISO
```

```

IOMUXC_SetPinMux(IOMUXC_ECSPI2_MOSI_ECSPI2_MOSI, 0U); // SPI2-MOSI
IOMUXC_SetPinMux(IOMUXC_ECSPI2_SCLK_ECSPI2_SCLK, 0U); // SPI2-SCLK
IOMUXC_SetPinMux(IOMUXC_ECSPI2_SS0_ECSPI2_SS0, 0U); // SPI2-SS0
SPI rate: Set the rate to 500K.
#define TRANSFER_BAUDRATE 500000U // rate 500K
Data length selection: 8 bit.
config->burstLength = 8; // Data length 8bit
Four Mode Selection: The four combinations of CPOL and CPHA are the four modes of SPI.
config->clockInactiveState = kECSPI_ClockInactiveStateLow; // Clock SCL: Low when active, high when idle
config->dataLineInactiveState = kECSPI_DataLineInactiveStateLow; // data MOSI&MISO: Low when active, high when idle
config->chipSelectActiveState = kECSPI_ChipSelectActiveStateLow; // chip selection SS: Low is selected, high is invalid
config->polarity = kECSPI_PolarityActiveHigh; // Clock signal polarity, i.e., SCLK active high (idle low) if CPOL is 0 and SCLK active low (idle high) if CPOL is 1.
config->phase = kECSPI_ClockPhaseFirstEdge; // Clock phase: If CPHA is 0, data is sampled on the first edge (rising or falling) of the serial clock. If CPHA is 1, data is sampled on the second edge (rising or falling) of the serial clock.

```

Main mode selection: Set SPI as the main mode.

```
config->channelConfig.channelMode = kECSPI_Master; // Master mode
```

Channel selection: A SPI has four hardware chip selection signals, and each chip selection signal is a hardware channel. This program selects channel 0.

```
config->channel = kECSPI_Channel0; // Channel 0
```

Turn off the self-loop: If the self-loop is turned on, the SPI data will loop back in the chip and will not reach the external pin. During program debugging, the interference of the external terminal can be eliminated. However, in real applications, it is necessary to turn off the self-loop and send and receive data from the external pin.

```
Config->enableLoopBack = false; // Do not loop, use external pin
```

## (2) SPI main execution flow

The SPI interface of development board 1 is in master mode, which enables sending and receiving interrupts. The SPI interface of development board 2 is in slave mode, which enables receiving and sending interrupts.

The SPI master sends a 64-byte receipt, and the SPI slave sends the data back after receiving it. After receiving the returned information, the SPI master compares whether the received and transmitted data are consistent, and outputs the comparison result. If they are consistent, this transmission is over, and wait for the input of any key to start the next transmission.

Sending data: EXAMPLE\_ECSPI\_MASTER\_BASEADDR represents SPI2, g\_m\_handle is the SPI instance which includes sending and receiving interrupts along with their callback functions, and masterXfer is the 64-byte data to be sent.

```
ECSPI_MasterTransferNonBlocking(EXAMPLE_ECSPI_MASTER_BASEADDR, &g_m_handle, &masterXfer); //Master mode interrupt mode send data
```

Receiving data: Both the sending and receiving on the SPI bus are controlled in master mode, so the process for the receiving function is the same as sending.

### **Comparison of receiving and sending data:**

```

for (i = 0U; i < TRANSFER_SIZE; i++)
{
    if (masterTxData[i] != masterRxData[i])
    {
        errorCount++;
    }
}

```

### (3) SPI slave initialization

SPI slave mode initialization should be consistent with the master mode, except for setting the operating mode to slave mode, all other settings remain the same.

The details are as follows:

Master-slave mode selection: Set SPI to slave mode

```
config->channelConfig.channelMode = kECSPI_Slave; // Slave mode
```

### SPI main execution flow

The SPI interface of development board 2 is in slave mode, which enables receiving and sending interrupts.

The SPI slave enters the waiting - for - receiving state. After the chip select signal becomes valid, it acquires data through the reception interrupt and sends back information, then enters the receiving state again.

Receiving data: EXAMPLE\_ECSPI\_SLAVE\_BASEADDR represents SPI2. g\_m\_handle is an SPI instance, which includes the send and receive interrupts and their callback functions. slaveXfer stores the received data.

```
ECSPI_SlaveTransferNonBlocking(EXAMPLE_ECSPI_SLAVE_BASEADDR, &g_s_handle, &slaveXfer);
```

//Receive data in slave interrupt mode

Data transmission: Both the sending and receiving of the SPI bus are controlled by the master mode. Therefore, the process of the receiving function is the same as sending.

### 3.6.3 Experimental Phenomena

- (1) After the compilation is completed, as described in Section 3.1.1 of the “OKMX8MPQ-C MCU - User Compilation Manual - V1.0”, manually load the M-core program in U-Boot;
- (2) Board 2 is powered on first. The M-core program starts, initializes the SPI, and enters a waiting state for receiving;
- (3) Board 1 is powered on later. The M-core program starts, initializes the SPI, and actively sends 64 bytes of data;
- (4) Board 2 receives the data through the SPI, prints the received data through the serial port, and then sends the received data again;

```
This is ECSPI slave transfer completed callback.  
It's a successful transfer.
```

```
Slave starts to transmit data!  
This is ECSPI slave transfer completed callback.  
It's a successful transfer.
```

**Slave receive:**

|    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| 4  | 5  | 6  | 7  | 8  | 9  | A  | B  | C  | D  | E  | F  | 10 | 11 | 12 | 13 |
| 14 | 15 | 16 | 17 | 18 | 19 | 1A | 1B | 1C | 1D | 1E | 1F | 20 | 21 | 22 | 23 |
| 24 | 25 | 26 | 27 | 28 | 29 | 2A | 2B | 2C | 2D | 2E | 2F | 30 | 31 | 32 | 33 |
| 34 | 35 | 36 | 37 | 38 | 39 | 3A | 3B | 3C | 3D | 3E | 3F | 40 | 41 | 42 | 43 |

**Slave example is running...**

```
Slave starts to receive data!
```

- (5) Board 1 receives the feedback information through the SPI, prints the received data through the serial port; Compare it with the sent data and output the result.

```

ECSPI board to board interrupt example.
This example use one board as master and another as slave.
Master and slave uses interrupt way. Slave should start first.
Please make sure you make the correct line connection. Basically, the connection is:
ECSPI_master -- ECSPI_slave
    CLK      --  CLK
    PCS      --  PCS
    MOSI     --  MOSI
    MISO     --  MISO
    GND      --  GND

Master transmit:
 1  2  3  4  5  6  7  8  9  A  B  C  D  E  F 10
11 12 13 14 15 16 17 18 19 1A 1B 1C 1D 1E 1F 20
21 22 23 24 25 26 27 28 29 2A 2B 2C 2D 2E 2F 30
31 32 33 34 35 36 37 38 39 3A 3B 3C 3D 3E 3F 40
Start receive data from slave.

ECSPI transfer all data matched!

Master received:
 1  2  3  4  5  6  7  8  9  A  B  C  D  E  F 10
11 12 13 14 15 16 17 18 19 1A 1B 1C 1D 1E 1F 20
21 22 23 24 25 26 27 28 29 2A 2B 2C 2D 2E 2F 30
31 32 33 34 35 36 37 38 39 3A 3B 3C 3D 3E 3F 40

Press any key to run again

```

- (6) At this time, enter any key in the development board 1 debugging serial port, and it will start a new round of the SPI send and receive process.

## 3.7 GPT-Timer

The timer in the M-core SDK package is a 1-second interrupt routine.

### 3.7.1 Hardware Connection

This routine does not have any special wiring.

### 3.7.2 Software Implementation

#### (1) Initialization

Timing interrupt initialization mainly includes bus clock, working status, counting mode, timing setting and interrupt initialization.

The details are as follows:

Bus Clock: The timer timing interrupt uses the ipg\_clk bus with a frequency of 66M, then 2 divided to 33M.

```
config->clockSource = kGPT_ClockSource_Periph; //Clock source: ipg_clk 66M
```

```
config->divider = 1U;
```

Operating state in the same mode: In sleep and debug modes, timer interrupts do not function.

```
config->enableRunInStop = true; // In stop mode, the timer's periodic interrupt continues to work.
```

```
config->enableRunInWait = true; // In wait mode, the timer's periodic interrupt continues to work.
```

```
config->enableRunInDoze = false; // In sleep mode, the timer's periodic interrupt doesn't work.
```

```
config->enableRunInDbg = false; // In debugging mode, the timer's periodic interrupt doesn't work.
```

Counting mode selection: The timer has two working modes: set-and-forget and free-running. The difference between these two working modes is as follows:

set-and-forget mode: The EPIT counter fetches its initial value from the load register EPITx\_LR and cannot directly write data to the counter register. Whenever the counter reaches 0, data from EPITx\_LR is reloaded into the counter, repeating indefinitely.

free-running mode: When the counter reaches 0, it restarts counting from 0xFFFFFFFF, not from the load register EPITx\_LR. Watch out for technological overflow when calculating the difference.

This routine is in set-and-forget mode.

```
config->enableFreeRun = false;
```

Timing setting: By setting different initial comparison values, interrupts can be obtained at different times. If gptFreq is set to 33M, it results in a 1-second interrupt; if set to 3.3M, it results in a 100-ms interrupt. If it is 0.33M, it is a 10ms interrupt.

```
GPT_SetOutputCompareValue(EXAMPLE_GPT, kGPT_OutputCompare_Channel1, gptFreq);
```

Interrupt Enable: Enable the timer interrupt.

```
GPT_EnableInterruptions(EXAMPLE_GPT, kGPT_OutputCompare1InterruptEnable);
```

## (2) Execution process

After the initialization of the timed interrupt is completed, the timed interrupt function will start, and the interrupt will be triggered after the timer count reaches the comparison value, and the user can execute the corresponding business application in the interrupt.

```
void EXAMPLE_GPT_IRQHandler(void)
{
    // Clear interrupt flag
    GPT_ClearStatusFlags(EXAMPLE_GPT, kGPT_OutputCompare1Flag);
    PRINTF("\r\n GPT 1 second interrupt is occurred !");
}
```

### 3.7.3 Experimental Phenomena

(1) After the compilation is completed, as described in Chapter 3.1.1 of “OKMX8MPQ-C\_MCU \_ User’s Compilation Manual-V1.0”, manually load the M-core program in uboot;

(2) After powering on the development board, the M-core program starts and completes the initialization of the timer interrupt. Press any key, and you will see the timer interrupt output printing information on the M-core’s debug serial port.

```
Press any key to start the example
Starting GPT timer ...
GPT 1 second interrupt is occurred !
```

## 3.8 GPT-Capture

The capture in the M-core SDK package is an input capture routine.

### 3.8.1 Hardware Connection

Channel 1 pin capture near resistor R206, defaulting to high level on the WM8960 audio chip attachment.



### 3.8.2 Software Implementation

## (1) Initialization

Timer capture initialization includes bus clock, pin configuration, operating mode, trigger method, and interrupt initialization.

The details are as follows:

**Bus clock:** The timer capture uses the ipg\_clk bus at a frequency of 66 MHz, divided by 2 to 33 MHz.

```
config->clockSource = kGPT_ClockSource_Pериф; //Clock source: ipg clk 66M
```

```
config->divider = 1U;
```

Pin configuration: The 8MP supports dual input capture, each with independent pins. This example uses the first input capture route.

|OMUXC\_SetPinMux(|OMUXC\_SAI3\_TXC\_GPT1\_CAPTURE1\_OU|);

**Operating mode in different modes:** In sleep and debug modes, timer capture does not output.

```
config->enableRunInStop = true; // In stop mode, the timer captures and continues to work
```

```
config->enableRunInWait = true; // In wait mode, the timer captures and contin
```

```
config->enableRunInDoze = false; // Timer capture does not work in sleep mode
```

```
config->enableRunInDbg = false; // Timer capture does not work in debugging mode
```

**Counting mode selection:** The timer has two working modes: set-and-forget and free-running.

these two working modes is as follows:

set-and-forget mode: The EPIT counter fetches its initial value from the load register EPITx\_LR and cannot directly write data to the counter register. Whenever the counter reaches 0, data from EPITx\_LR is reloaded into the counter, repeating indefinitely.

free-running mode: When the counter reaches 0, it restarts counting from 0xFFFFFFFF, not from the load register EPITx\_LR. Watch out for technological overflow when calculating the difference.

This routine is in set-and-forget mode.

```
config->enableFreeRun = false;
```

Triggering methods: Support for rising edge trigger, falling edge trigger, any edge trigger. This example uses rising edge trigger.

```
GPT_SetInputOperationMode(DEMO_GPT_BASE,BOARD_GPT_INPUT_CAPTURE_CHANNEL kGPT_InputOperation_RiseEdge)
```

Interrupt Enable: Upon detecting a rising edge at the capture pin, the system enters an interrupt.

```
GPT_EnableInterrupts(DEMO_GPT_BASE, BOARD_GPT_CHANNEL_INTERRUPT_ENABLE);
```

## (2) Execution process

After timer capture initialization is completed, the timer capture function starts. The timer count begins with the initial value loaded from the register EPITx\_LR and decrements continuously. After detecting a rising edge on the input pin, you can read the current timer value at that moment.

To obtain the timer count: Upon entering the capture interrupt, you can read the timer count from DEMO\_GPT\_BASE, representing capture channel 1.

```
captureVal = GPT_GetInputCaptureValue(DEMO_GPT_BASE, BOARD_GPT_INPUT_CAPTURE_CHANNEL);
```

### 3.8.3 Experimental Phenomena

(1) After the compilation is completed, as described in Chapter 3.1.1 of “OKMX8MPQ-C\_MCU \_ User’s Compilation Manual-V1.0”, manually load the M-core program in uboot;

(2) To wait for a rising edge interrupt after timer capture initialization. When shorting the R206 pin to the GND pin repeatedly, you will observe the timer count at the moment of capturing the rising edge in the debugging serial port of the M core.

#### GPT input capture example

Once the input signal is received the input capture value is printed

Capture value =1

Capture value =9f068beb

Capture value =b2cbf608

Capture value =bd7cfe47

Capture value =bf4afbc2

Capture value =c0d62cef

Capture value =c2ddaa82

Capture value =c359f6fd

Capture value =c3c17f3b

## 3.9 PWM

In the M-core SDK package, there is an example routine for PWM output with a continuously changing duty cycle waveform.

### 3.9.1 Hardware Connection

Use a DuPont wire to extend pin U36-4 (PWM4) on the development board, and connect an oscilloscope to this pin and GND.



### 3.9.2 Software Implementation

#### (1) Initialization

PWM initialization mainly includes bus clock, pin, working state, default level and interrupt initialization.

The details are as follows:

Bus Clock: PWM uses a low rate 32 K clock bus, divided by 1 again is still 32 K.

```
config->clockSource = kPWM_LowFrequencyClock; // Select clock source, low rate 32K
```

```
config->prescale = 0U; // Clock division, 0 represents 1 division
```

Pin configuration: Select PWM4 as the output pin.

```
IOMUXC_SetPinMux(IOMUXC_SAI5_RXFS_PWM4_OUT, 0U);
```

Chip mode operating status: In stop, sleep, idle, and debug modes, PWM does not output.

```
config->enableStopMode = false; // PWM does not work in stop mode
```

```
config->enableDozeMode = false; // PWM does not work in sleep mode
```

```
config->enableWaitMode = false; // PWM does not work in waiting mode
```

```
config->enableDebugMode = false; // PWM does not work in debugging mode
```

Output level: Set to output high level first, and then output low level when flipping.

```
config->outputConfig = kPWM_SetAtRolloverAndClearAtcomparison; // By default, output high level first, and output low level when flipping
```

FIFO setting: No special settings.

```
config->fifoWater = kPWM_FIFOWaterMark_2; // When set to 2, it means that the FIFO is empty when the free position is greater than or equal to 3
```

```
config->sampleRepeat = kPWM_EachSampleOnce; // Use FIFO once per sample
```

```
config->byteSwap = kPWM_BitNoSwap; // The byte order remains unchanged
```

```
config->halfWordSwap = kPWM_HalfWordNoSwap; // Do not perform half character exchange
```

Frequency setting: Set the period value to 30, the frequency is:  $32K / (30+2) = 1K$ , DEMO\_PWM\_BASEADDR represents PWM4.

```
PWM_SetPeriodValue(DEMO_PWM_BASEADDR, 30);
```

Interrupt Enable: When the PWM count reaches the period value of 30, indicating the completion of PWM waveform output and an empty FIFO, an interrupt is generated.

```
PWM_EnableInterrupts(DEMO_PWM_BASEADDR, kPWM_FIFOEmptyInterruptEnable);
```

### 1. Execution process

After PWM initialization is complete and PWM functionality begins, upon entering the interrupt, the duty cycle continues to increase until it reaches 100%. Afterward, it decreases continuously until it reaches 0%. Then, it increases again continuously, followed by another decrease. A PWM wave with changing duty cycle is continuously output according to this rule.

Set Duty Cycle: Duty Cycle =  $(\text{pwmDutycycle} + 2) / (30 + 2)$ . You can change the duty cycle by increasing or decreasing the value of pwmDutyCycle.

```
PWM_SetSampleValue(DEMO_PWM_BASEADDR, pwmDutycycle);
```

Increase Duty Cycle: After one PWM output cycle completes and enters the interrupt, continuously increase pwmDutyCycle to increase the duty cycle. Until it reaches 100%.

```
if (pwmDutyUp)
{
    // Increase the duty cycle until it reaches the limited value of 30.

    if (++pwmDutycycle > PWM_PERIOD_VALUE)
    {
        pwmDutycycle = PWM_PERIOD_VALUE;
        pwmDutyUp = false;
    }
}
```

Reduce Duty Cycle: Once the duty cycle reaches 100%, begin to decrease the pwmDutycycle value, thereby reducing the duty cycle until it ultimately reaches 0.

```
else
{
    // Reduce the duty cycle until it reaches 0.

    if (--pwmDutycycle == 0U)
    {
        pwmDutyUp = true;
    }
}
```

### 3.9.3 Experimental Phenomena

- (1) After the compilation is completed, as described in Chapter 3.1.1 of “OKMX8MPQ-C\_MCU \_ User’s Compilation Manual-V1.0”, manually load the M-core program in uboot;
- (2) After the M-core program starts and completes PWM initialization, PWM begins outputting a waveform with continuously changing duty cycle;
- (3) On the oscilloscope, you can observe the waveform, and if you look closely, you will see that the frequency is 1 kHz and the duty cycle is continuously changing.



### 3.10 Audio SAI

In the M-core SDK package, there are four audio routines/examples. interrupt\_transfer uses interrupts to play sound; sdma\_transfer uses DMA to play sound; interrupt\_record\_playback uses interrupts for simultaneous recording and playback; sdma\_record\_playback uses DMA for simultaneous recording and playback. We will mainly focus on the interrupt\_transfer routine.

#### 3.10.1 Hardware Connection

Insert the headphones into the headphone jack and plug the speaker into the left or right speaker.



### 3.10.2 Software Implementation

## Initialization

The 8MP audio functionality includes SAI and I2C components. I2C is used to configure Codec chip registers, while SAI handles clock output and audio data sending and receiving. Initialization includes I2C, SAI and Codec chips.

\* \* 120 \* \*

I<sup>2</sup>C initialization mainly includes initializing the bus clock, pins, baud rate, and address.

The details are as follows:

Bus clock: I2C uses SysPLL1 Div5 bus, operating at a frequency of 160M, further divided by 10 to 16M.

```
CLOCK_SetRootMux(kCLOCK_RootI2c3, kCLOCK_I2cRootmuxSysPll1Div5); // I2C audio source: SysPLL1 Div5 160MHz
```

```
CLOCK_SetRootDivider(kCLOCK_RootI2c3_1U_10U); // Divide-by-10 160MHZ / 10 = 16MHz
```

Pin configuration: This routine uses I2C3.

IOMIIXC\_SetPinMux(IOMIIXC\_I2C3\_SCI\_I2C3\_SCI\_1U);

[OMI]XC\_SetPinMux([OMI]XC\_I2C3\_SDA\_I2C3\_SDA\_1U);

Baud rate and address: Set the baud rate to 100 K and the Codec chip address to 0x1A.

```
#define WM8960_I2C_ADDR    0x1A
#define WM8960_I2C_BAUDRATE (100000U)
```

Transmission Function: Write configuration parameters to the Codec registers via the transmission function.

```
I2C_MasterTransferBlocking(s_i2cBases[i2cMasterHandle->instance], &transfer);
```

### SAI

SAI initialization mainly includes the initialization of the bus clock, pins, audio settings, synchronization mode, etc. The details are as follows:

Bus clock: SAI uses the AUDIO PLL1 bus with a frequency of 393M and a frequency division of 32 to 12.288M.

```
CLOCK_SetRootMux(kCLOCK_RootSai3, kCLOCK_SaiRootmuxAudioPll1); // audio source: AUDIO PLL1 393216000HZ
```

```
CLOCK_SetRootDivider(kCLOCK_RootSai3, 1U, 32U); // Divide-by-32 393216000HZ / 32 = 12.288MHz
```

Pin configuration: This routine uses SAI3.

```
IOMUXC_SetPinMux(IOMUXC_SAI3_MCLK_AUDIO_MIX_SAI3_MCLK, 0U);
IOMUXC_SetPinMux(IOMUXC_SAI3_TXC_AUDIO_MIX_SAI3_TX_BCLK, 0U);
IOMUXC_SetPinMux(IOMUXC_SAI3_TXD_AUDIO_MIX_SAI3_TX_DATA0, 0U);
IOMUXC_SetPinMux(IOMUXC_SAI3_TXFS_AUDIO_MIX_SAI3_TX_SYNC, 0U);
```

\*\* Audio configuration: \*\* 8MP acts as the master, MCLK configured at 12.288MHz, sampling rate at 16KHz, audio data is 16-bit, dual-channel (left and right). According to the above configuration, BCLK = 16KHz \* 2 channels \* 16 bits = 512KHz. The codec chip needs synchronized modification of register settings to match the settings of the 8MP.

```
#define DEMO_SAI_MASTER_SLAVE kSAI_Master      // 8MP is in charge
#define DEMO_AUDIO_DATA_CHANNEL (2U)           // Left and right channels
#define DEMO_AUDIO_BIT_WIDTH  kSAI_WordWidth16bits // 16 bits wide
#define DEMO_AUDIO_SAMPLE_RATE (kSAI_SampleRate16KHz) // Sampling rate 16 K
```

Synchronous Mode: Asynchronous Sending, Synchronous Receiving

Both sending and receiving use the BCLK and SYNC of Tx. The sending operation is first enabled and then disabled after the sending.

```
#define DEMO_SAI_TX_SYNC_MODE kSAI_ModeAsync
#define DEMO_SAI_RX_SYNC_MODE kSAI_ModeSync
```

### Codec WM8960

Codec chip initialization mainly includes power supply and register initialization. The details are as follows:

Chip power supply: 8 MP is enabled by GPIO control power supply, and WM8960 works normally.

```
gpio_pin_config_t gpio_config = {kGPIO_DigitalOutput, 0, kGPIO_NoIntmode};
GPIO_PinInit(CODEC_POWER_GPIO, CODEC_POWER_GPIO_PIN, &gpio_config);
GPIO_PinWrite(CODEC_POWER_GPIO, CODEC_POWER_GPIO_PIN, 1U);
```

Parameter configuration: WM8960 is consistent with 8MP SAI.

```
wm8960_config_t wm8960Config = {
    .i2cConfig = {.codecl2CInstance = BOARD_CODEC_I2C_INSTANCE, .codecl2CSourceClock = BOARD_CODEC_I2C_CLOCK_FREQ},
    .route = kWM8960_RoutePlaybackandRecord, // Recording and playback channel
    .rightInputSource = kWM8960_InputSingleEndedMic, // Input via input1
    .playSource = kWM8960_PlaySourceDAC, // The playback is derived from the DAC
    .slaveAddress = WM8960_I2C_ADDR, // I2C address 0x1A
    .bus = kWM8960_BusI2S, // I2C baud rate 100K
```

```
.format      = {.mclk_HZ  = 12288000U,          // MCKL 12.288M
.sampleRate = kWM8960_AudioSampleRate16KHz, // Sampling rate 16k
.bitWidth   = kWM8960_AudioBitWidth16bit}, // 16 bits wide
.master_slave = false,                  // wm8960 works as slave
.enableSpeaker = true,                  // Speaker playback enable
};
```

Register configuration: Update the configuration to the appropriate register by writing data to the WM8960 via I2C.

```
CODEC_Init(&codecHandle, &boardCodecConfig);
WM8960_CHECK_RET(WM8960_WriteReg(handle, WM8960_POWER1, 0xFE), ret);
```

### (2) Execution process

After initializing the SAI bus and WM8960 with the 8MP, sound data is sent using interrupt mode. The sound data is stored in music.h and is approximately 48K.

```
temp      = (uint32_t)music;
xfer.data = (uint8_t *)temp;
xfer.dataSize = MUSIC_LEN;
SAI_TransferSendNonBlocking(DEMO_SAI, &txHandle, &xfer);
```

### 3.10.3 Experimental Phenomena

- (1) After the compilation is completed, as described in Chapter 3.1.1 of “OKMX8MPQ-C\_MCU \_ User’s Compilation Manual-V1.0”, manually load the M-core program in uboot;
- (2) The development board is powered on, the M-core program is started, and a “beep” sound can be heard from the headphones and speakers, which lasts only 1-2 seconds due to limited memory.

## 3.11 Audio Rate Conversion ASRC

There are three ASRC routines in the M-core SDK package. asrc\_m2m\_polling uses polling for copying sound data at different sampling rates from memory to memory. asrc\_m2m\_sdmar uses DMA for the same purpose. asrc\_p2p\_out\_sdma uses DMA for copying sound data between peripherals. We focus on asrc\_m2m\_polling, which requires significant memory and loads the M-core program into DDR due to the lack of NorFlash on the development board.

### 3.11.1 Hardware Connection

Insert the headphones.



### **3.11.2 Software Implementation**

## (1) Initialization

Initializations involving audio-related aspects have been mentioned in the section on audio and will not be detailed here. ASRC initialization mainly involves initializing input sources, output sources, filters, and resampling.

The details are as follows:

Input source: Set the sampling rate to 16KBits with data in 16-bit integer format.

```
config->contextInput.sampleRate = inSampleRate; // Sample rate before conversion
```

```
config->contextInput.watermark = FSL_ASRC_INPUT_FIFO_DEPTH / 2U; // FIFO depth is 128
```

```
config->contextInput.accessCtrl.accessIterations = 1; //Sequential access count per source
```

```
config->contextInput.accessCtrl.accessGroupLen = (uint8_t)channels; // Number of channels in the
```

```
config->contextInput.accessCtrl.accessLen = (uint8_t)channels; //Number of channels
```

```
config->contextInput.dataFormat.dataPosition = 0U; // Input data sample location
```

```

config->contextInput.dataFormat.dataWidth = kASRC_DataWidth16Bit; // 16 bits wide
config->contextInput.dataFormat.dataType = kASRC_DataTypeInteger; // Integer
config->contextInput.dataFormat.dataSign = kASRC_DataSigned; // Data signature
Output audio source: Set the sampling rate to 48 KBits and the data to 16-bit integer.
config->contextOutput.sampleRate = outSampleRate; // Sample rate after conversion
config->contextOutput.watermark = FSL_ASRC_OUTPUT_FIFO_DEPTH / 8U; // FIFO depth is 64/8 =8
config->contextOutput.accessCtrl.accessIterations = 1; // Number of sequential fetches per source
config->contextOutput.accessCtrl.accessGroupLen = (uint8_t)channels; // Number of channels in the context
config->contextOutput.accessCtrl.accessLen = (uint8_t)channels; // Number of channels per source 1
config->contextOutput.dataFormat.dataPosition = 0; // Input data sample location
config->contextOutput.dataFormat.dataEndianness=kASRC_DataEndianLittle; // Output data format little endian
config->contextOutput.dataFormat.dataWidth = kASRC_DataWidth16Bit; // 16 bits wide
config->contextOutput.dataFormat.dataType = kASRC_DataTypeInteger; // Integer
config->contextOutput.dataFormat.dataSign = kASRC_DataSigned; // Data signature
config->contextOutput.enableDither = false; // The output path contains a tpdf dither function. 32-bit floating point output modes (16, 20, 24, 32-bit) output modes are not supported
config->contextOutput.enableIEC60958 = false; // Does not support iec60958 digital audio

```

Filter: Initialization Mode: Zero-fill the right half of the sample buffer.

Stop Mode: Replicate the last valid sample to fill the left half of the sample buffer.

Pre-filtering Phase 1 Output: Stored as 32-bit floating-point values.

```
config->contextPrefilter.initMode=kASRC_SampleBufferFillZeroOnInit; // Pre-filtering Initialization Mode
```

```
config->contextPrefilter.stopMode=kASRC_SampleBufferFillLastSampleOnStop; // Pre-filtering Stop
```

```
config->contextPrefilter.stage1Result=kASRC_PrefilterStage1ResultFloat; // The output of Pre-filtering Phase 1 is stored as 32-bit floating-point numbers.
```

Resampling configuration: Zero-fill the right half of the sample buffer; Stop mode: Copy the last sample to fill the left half of the sample buffer; Resampling junction at 128.

```
config->contextResampler.initMode=kASRC_SampleBufferFillZeroOnInit; // Resampling initialization mode
```

```
config->contextResampler.stopMode=kASRC_SampleBufferFillLastSampleOnStop; // Resampling stop mode
```

```
config->contextResampler.tap =kASRC_ResamplerTaps_128; // Resampling interface
```

## (2) Execution process

After the CPU completes the audio and ASRC initialization, play the sound at the original sampling rate, then start ASRC to perform audio sampling rate conversion, generate a new audio file, and finally play the sound at the converted sampling rate.

Play the sound at the original sample rate:

```
saiPlayAudio((uint8_t *)music, MUSIC_LEN); // Play the original baud rate sound
```

Start ASRC to complete the audio sample rate conversion:

```
ASRC_TransferBlocking(DEMO_ASRC, DEMO_ASRC_CONTEXT, &asrcTransfer); // Sound frequency conversion
```

Configure the converted audio into the audio-related registers.

```
SAI_TxSetBitClockRate(DEMO_SAI, DEMO_AUDIO_MASTER_CLOCK, DEMO_ASRC_OUT_SAMPLE_RATE, DEMO_AUDIO_BIT_WIDTH, DEMO_AUDIO_MASTER_CLOCK_SOURCE);
```

Play the sound at the converted sample rate:

```
saiPlayAudio((uint8_t *)s_asrcOutBuffer, ASRC_GetContextOutSampleSize(DEMO_ASRC_IN_SAMPLE_RATE, MUSIC_LEN, 2U, DEMO_ASRC_OUT_SAMPLE_RATE));
```

### 3.11.3 Experimental Phenomena

(1) After compilation, as described in Section 3.1.2 of the “OKMX8MPQ-C\_MCU-User Compilation Manual-V1.0,” manually load the M-core program in Uboot;

(2) After the CPU completes the initialization of audio and ASRC, log the source audio information via the serial port, then play the sound at the original sampling rate. Next, start the ASRC to complete the audio sampling rate conversion, generating a new audio file. Finally, output the new source audio information via the serial port and play the sound at the converted sampling rate. **Two beeps can be heard through the headset.**

**ASRC memory to memory polling example.**

**Playback raw audio data**

sample rate : 16000

channel number: 2

frequency: 215HZ.

**Playback converted audio data**

sample rate : 48000

channel number: 2

frequency: 215HZ.

**ASRC memory to memory polling example finished.**



## 3.12 Watchdog

Forlinx OKMX8MP-C development board supports watchdog functionality. You can set the timeout and feeding intervals. If the watchdog isn’t fed on time due to program anomalies, it triggers a software reset to maintain normal operation. M-core SDK package includes examples for normal watchdog feeding and handling watchdog timeout scenarios. Users can modify the watchdog feeding time and waiting time to simulate normal and abnormal conditions.

### 3.12.1 Hardware Connection

This routine does not have any special wiring.

### 3.12.2 Software Implementation

#### (1) Initialization

Watchdog initialization primarily includes enabling functions, setting operating modes, configuring timeouts, watchdog feeding intervals, and initializing interrupts.

The details are as follows:

Enable: Enable the watchdog function.

```
config->enableWdog = true; // Enable the watchdog
```

Operating modes: Enable or disable the watchdog in stop, low-power, or debug modes.

```
config->workMode.enableWait = false; // In stop mode, the watchdog stops working
```

```
config->workMode.enableStop = false; // In wait mode, the watchdog stops working
```

```
config->workMode.enableDebug = false; // In debug mode, the watchdog stops working
```

Timeout time: Setting range is 0.5s-16s, step is 0.5s.

```
config.timeoutValue = 0xFU; // Timeout period: (0xF+1)/2 = 8 sec
```

Watchdog feeding time: Set in the range of 0.5s to 16s, with an increment of 0.5s.

```
config.interruptTimeValue = 0x4U; // Time to enter interrupt before dog feed timeout reset (0x4)/2 = 2 sec
```

Watchdog interrupt enable: If the watchdog is not fed within the watchdog timeout period, a watchdog interrupt can be triggered before the CPU is reset.

```
config->enableInterrupt = true; // Enable the interrupt
```

## (2) Execution process

After the CPU completes watchdog initialization, setting the timeout and feeding interval, if the feeding interval is less than the timeout period, the watchdog will be regularly fed, and the program will run normally. If the feeding interval is greater than the timeout period, the chip will reset due to timeout. The reset reason can be obtained from the WRSR register.

After the CPU starts running, the chip's reset reason can be read from the WRSR register, which includes power-on reset, software reset, and watchdog timeout reset.

```
switch (resetFlag & (kWDOG_PowerOnResetFlag | kWDOG_TimeoutResetFlag | kWDOG_SoftwareResetFlag))
{
    case kWDOG_PowerOnResetFlag:
        PRINTF(" Power On Reset!\r\n");
        break;
    case kWDOG_TimeoutResetFlag:
        PRINTF(" Time Out Reset!\r\n");
        break;
    case kWDOG_SoftwareResetFlag:
        PRINTF(" Software Reset!\r\n");
        break;
    default:
        PRINTF(" Error status!\r\n");
        break;
}
```

When the development board is powered on, the chip resets due to power-on. At this point, the watchdog reset register is set, initiating a software reset of the chip.

```
// Trigger a software reset if the system is reset after power up
```

```
if (resetFlag & kWDOG_PowerOnResetFlag)
{
    PRINTF("\r\n- 1.Testing System reset by software trigger... ");
    WDOG_TriggerSystemSoftwareReset(DEMO_WDOG_BASE);
}
```

After the chip resets with a software reset detected, the program initializes the watchdog timer with a timeout and enters an infinite `while(1)` loop without feeding the watchdog, leading to a watchdog timeout reset.

```
if (resetFlag & kWDOG_SoftwareResetFlag)
{
    PRINTF("\r\n- 2.Testing system reset by WDOG timeout.\r\n");
    WDOG_GetDefaultConfig(&config);
    config.timeoutValue = 0xFU; /* Timeout value is (0xF + 1)/2 = 8 sec. */
    WDOG_Init(DEMO_WDOG_BASE, &config);
```

```

PRINTF(" — wdog Init done—\r\n" );
// Program dead loop, nobody feed the dog, watchdog timeout reset
while (1)
{
}
}

```

After the program restarts and detects a reset due to a timeout, it resets the watchdog timeout and continuously feeds the watchdog to ensure normal program operation.

When setting the watchdog timeout and feeding interval for normal operation, the program feeds the watchdog and prints feeding information every 2 seconds if running correctly. When the watchdog times out, it triggers an interrupt and then resets. After resetting, the CPU reads the chip reset reason from the WRSR register, which indicates a timeout.

```

if (resetFlag & kWDOG_TimeoutResetFlag)
{
    PRINTF("\r\n- 3.Test the WDOG refresh function by using interrupt.\r\n" );
    WDOG_GetDefaultConfig(&config);
    config.timeoutValue = 0xFU; // Timeout period: (0xF+1)/2 = 8 sec.
    config.enableInterrupt = true;
    config.interruptTimeValue = 0x4U; // How long before reset to enter interrupt:(0x4)/2 = 2 sec
    WDOG_Init(DEMO_WDOG_BASE, &config);
    PRINTF(" — wdog Init done—\r\n" );
    while (1)
    {
        WDOG_Refresh(DEMO_WDOG_BASE);
        PRINTF("\r\nWDOG has been refreshed!" );
        delay(SystemCoreClock);
    }
}

```

### 3.12.3 Experimental Phenomena

- (1) Compilation program, as introduced in Section 3.1.1 of the “OKMX8MPQ-C\_MCU User Compilation Manual V1.0” , manually loads M core program in U-Boot;
- (2) Upon device power-up, it resets due to power-up, forcing a software reset.

```

***** System Start *****
System reset by: Power On Reset!

- 1. Testing System reset by software trigger...

```

After the program restarts due to a software reset, the watchdog is configured but not fed, causing a timeout and restart.

```

- 2. Testing system reset by WDOG timeout.
--- wdog Init done---

```

After the program resets, the reset reason is a watchdog timeout. At this point, the watchdog is being fed properly, and the program runs normally.

```
***** System Start *****
System reset by: Time Out Reset!

- 3.Test the WDOG refresh function by using interrupt.
--- wdog Init done---

WDOG has be refreshed!
```

## 3.13 Temperature Monitor TMU

OKMX8MP-C development board supports Temperature Monitoring Unit (TMU), capable of reading instantaneous and average chip temperatures, and setting temperature thresholds and interrupts. There are two TMU examples in the M-core SDK package. Temperature\_polling is an example for polling to read the chip temperature, while monitor\_threshold demonstrates how to set temperature limits and trigger interrupts. The following is an introduction to the monitor\_threshold routine.

### 3.13.1 Hardware Connection

This routine does not have any special wiring.

### 3.13.2 Software Implementation

#### (1) Initialization

TMU initialization mainly involves initializing monitoring points, temperature thresholds, and interrupts.

The details are as follows:

Monitoring points: 8MP supports two detection points. The primary monitoring point measures the chip's memory, while the secondary monitoring point is located near the ARM core. This example selects the primary monitoring point.

```
config->probeSelect = kTMU_ProbeSelectMainProbe; // Primary detection point
```

Filter: The average low-pass filter threshold is 0.5.

```
config->averageLPF = kTMU_AverageLowPassFilter0_5; //Average low-pass filter setting 0.5
```

Temperature monitoring threshold settings include: instantaneous temperature, average temperature, and average critical threshold.

```
config.thresholdConfig.immediateThresholdEnable = false; //Close high temperature immediate threshold
```

```
config.thresholdConfig.immediateThresholdValueOfMainProbe = DEMO_TMU_IMMEDIATE_THRESOLD; // High temperature instant threshold of main probe -10 °
```

```
config.thresholdConfig.AverageThresholdEnable = true; // Turn on high temperature average threshold
```

```
config.thresholdConfig.averageThresholdValueOfMainProbe = DEMO_TMU_AVERAGE_THRESOLD;
```

```
// High temperature average threshold of the main probe 50 °
```

```
config.thresholdConfig.AverageCriticalThresholdEnable = false; // Close high temperature average critical threshold
```

```
config.thresholdConfig.averageCriticalThresholdValueOfMainProbe = DEMO_TMU_AVERAGE_CRITICAL_THRESOLD; // High temperature average critical threshold of the main probe 88 °
```

Interrupt enable: When the temperature of the main probe exceeds the average temperature threshold, an interrupt is triggered.

```
TMU_EnableInterrupts(DEMO_TMU_BASE,kTMU_AverageTemperature0InterruptEnable);
(void)EnableIRQ(DEMO_TMU_IRQ);
```

#### (2) Execution process

8MP temperature measurement range is from -40°C to 105°C. The temperature value can be obtained from the register, with the highest bit (Bit 8) indicating the temperature polarity: 1 for negative and 0 for positive.

```
status = TMU_GetAverageTemperature(DEMO_TMU_BASE, config.probeSelect, &temp);
```

An interrupt will be triggered when the internal chip temperature exceeds the threshold. To avoid continuously entering the interrupt, the interrupt can be disabled inside the interrupt routine.

```
intStatus = TMU_GetInterruptStatusFlags(DEMO_TMU_BASE);
// One condition for clearing the interrupt status flag is that the actual temperature is below the threshold
TMU_ClearInterruptStatusFlags(DEMO_TMU_BASE, intStatus);
//To avoid going all the way to the interrupt, the interrupt can be turned off in the interrupt
(void)DisableIRQ(DEMO_TMU_IRQ);
```

### 3.13.3 Experimental Phenomena

(1) Compilation program, as introduced in Section 3.1.1 of the “OKMX8MPQ-C\_MCU User Compilation Manual V1.0” , manually loads M core program in U-Boot;

(2) After the M core program completes the TMU initialization, read the average temperature and display it;

(3) After waiting for the temperature to exceed 50°C, display the current temperature, trigger the interrupt, and disable the interrupt.

```
TMU monitor threshold example.
Initialization average temperature is positive 40 celsius degree
Average temperature is positive 50 celsius degree
High temperature average threshold to be reached.
```

## 3.14 SDMA

DMA (Direct Memory Access) provides an efficient method for transferring data between peripheral registers and memory, or between memory and memory. Memory-to-memory transfer refers to copying the contents of one specified memory area to another memory area. Similar to the C language memory copy function memcpy, DMA transfers can achieve higher transfer efficiency. In particular, DMA transfers use almost no CPU resources, which can save a lot of CPU resources.

There are two SDMA examples in the M-core SDK package. Memory\_to\_memory is an example of transferring four bytes using DMA; scatter-gather is an example of accumulating multiple four-byte chunks before performing a centralized transfer. The following is an introduction to the memory\_to\_memory routine.

### 3.14.1 Hardware Connection

This routine does not have any special wiring.

### 3.14.2 Software Implementation

#### (1) Initialization

DMA initialization mainly includes initialing rate, channel, transfer settings, and priority.

The details are as follows:

Rate: Set the SDMA frequency to be consistent with the CPU frequency.

```
config->enableRealTimeDebugPin = false; // If the real-time debug pin is enabled, the DMA is turned off to reduce power consumption
```

```
config->isSoftwareResetClearLock = true; // A software reset clears the lock bit, preventing SDMA scripts from being written to the SDMA
```

```
config->ratio = kSDMA_HalfARMClockFreq; // SDMA rate is half of the main frequency
```

Channel: Select DMA3 for 32 channels. This routine selects channel 1 and the number of buffers is 1.

```
handle->base = SDMAARM3; // Use DMA3
```

```
handle->channel = 1U; // Channel1
```

```
handle->bdCount = 1U; // Number of buffers
```

Transfer settings: Set the source address, destination address, number of bytes, and the number of transfers. The type is memory-to-memory transfer, triggered by software.

```
config->srcAddr=MEMORY_ConvertMemoryMapAddress(srcAddr, kMEMORY_Local2DMA); //Source address
```

```
config->destAddr=MEMORY_ConvertMemoryMapAddress(destAddr, kMEMORY_Local2DMA); //Destination Address
```

```
config->bytesPerRequest = bytesEachRequest; // Single byte transmission
```

```
config->transferSzie = transferSize; // Total bytes transmission
```

```
config->type = kSDMA_MemoryToMemory; // Transfer from memory to memory
```

```
config->scriptAddr = FSL_FEATURE_SDMA_M2M_ADDR; // Memory to memory script startup address
```

```
config->isEventIgnore = true; // 1 represents software trigger, 0 represents hardware trigger
```

```
config->isSoftTriggerIgnore = false; // If this bit is ignored, 1 indicates hardware event triggering, 0 indicates software triggering
```

```
config->eventSource = 0; // The event source number for this channel. 0 means no event, use software trigger.
```

Set priority: Different channels cannot transmit at the same time. Set corresponding priority for different channels to avoid conflict.

```
SDMA_SetChannelPriority(EXAMPLE_SDMAARM, 1, 2U);
```

#### (2) Execution process

After the CPU completes the DMA initialization, it starts the transfer process and completes the copying of data from the source address to the destination address in the DDR memory.

```
SDMA_StartTransfer(&g_SDMA_Handle);
```

### 3.14.3 Experimental Phenomena

(1) Compilation program, as introduced in Section 3.1.1 of the “OKMX8MPQ-C\_MCU User Compilation Manual V1.0” , manually loads M core program in U-Boot;

(2) After the CPU completes the DMA initialization, the serial port prints the value of the destination memory, initiates the transfer process, completes the copying of data from the source address to the destination address in the DDR memory, and then prints the value of the destination memory from the serial port.

```

SDMA memory to memory transfer example begin.

Destination Buffer:
0      0      0      0
SDMA memory to memory transfer example finish.
Destination Buffer:
1      2      3      4      █

```

## 3.15 RDC

Most of the 8MP peripherals can be used by the A and M cores. When a peripheral is used by more than one core simultaneously, the RDC function can lock access to this peripheral by a certain core to avoid peripheral output anomalies. RDC is called Resource Domain Controller, that is, Resource Domain Controller, which can realize the exclusivity and release of peripherals and memory of a certain domain through registers and mutual exclusion locks.

RDC routines are available in the M-core SDK package. Lists the functions of locking peripherals with RDC registers, locking peripherals with Sema42 mutexes, and locking memory with RDC registers.

### 3.15.1 Hardware Connection

This routine does not have any special wiring.

### 3.15.2 Software Implementation

#### (1) RDC control GPIO5

8MP supports up to four domains, so writing different values to the last eight bits of the register will allow the corresponding domains to read and write to a pin and memory enable, such as prohibiting the A-core domain 0 to read and write to a pin, set the register to 0xFC can be set, the last two bits are 0, which means that domain 0 is prohibited from reading and writing to the peripheral; the other six bits are 1, which means that domains 1 - 3 are allowed to read and write to the peripheral.

Each shared peripheral has a serial number in the RDC resource table to distinguish it from other peripherals, e.g. GPIO5 is defined in the RDC as follows:

```
kRDC_Periph_GPIO5      = 4U,      /**< GPIO5 RDC Peripheral */
```

In the M-core program, you can modify the program as follows to achieve the purpose of M-core exclusive GPIO5.

```
RDC_GetDefaultPeriphAccessConfig(&periphConfig); //Get RDC default configuration
```

```
periphConfig.periph = APP_RDC_PERIPH; // The peripheral devices to be controlled are GPIO5
```

```
periphConfig.policy &= ~(RDC_ACCESS_POLICY(APP_CUR_MASTER_DID, kRDC_ReadWrite)); // Prohibit M7 core reading and writing
```

```
RDC_SetPeriphAccessConfig(APP_RDC, &periphConfig); // Save RDC configuration
```

Accessing GPIO5 at this time will trigger a hardware error, and M7 will be restored to check the access right of GPIO5 in the error handling function.

```
// Restore M7 to verify access to GPIO5
```

```
PRINTF(" RDC Peripheral access GPIO5 error\r\n" );
```

```
periphConfig.policy |= RDC_ACCESS_POLICY(APP_CUR_MASTER_DID, kRDC_ReadWrite);
```

```
RDC_SetPeriphAccessConfig(APP_RDC, &periphConfig);
```

#### (2) RDC Sema42 mutex control GPIO5

RDC Sema42 is a semaphore specially used with the RDC function. When a domain acquires the sema42 lock of a peripheral, other domains cannot access the peripheral. They can only access the peripheral after the domain releases the lock.

The RDC Sema42 data lock is configured as follows:

```
RDC_GetDefaultPeriphAccessConfig(&periphConfig); //Get default RDC configuration  
periphConfig.policy = 0xFF; //Allow all domains to access  
periphConfig.periph = kRDC_Periph_GPIO5; //Pin isGPIO5  
periphConfig.enableSema = true; //Enable sema42 semaphore  
RDC_SetPeriphAccessConfig(RDC, &periphConfig); //Reset RDC access configuration
```

The M-core locks on GPIO5 to gain access to the pin:

```
RDC_SEMA42_Lock(APP_RDC_SEMA42, APP_RDC_SEMA42_GATE, APP_MASTER_INDEX, BOARD_DOMAIN_ID);
```

M-core releases the RDC Sema42 lock on GPIO5, releasing the use of the pin:

```
RDC_SEMA42_Unlock(APP_RDC_SEMA42, APP_RDC_SEMA42_GATE);
```

The M core does not acquire the lock initially. During this time, if it accesses GPIO, a hardware error is triggered. In the error handling function, the lock is acquired to restore the M7 core's access rights to the GPIO.

// Lock the sema42 door, and then peripheral devices can access it

```
RDC_SEMA42_Lock(APP_RDC_SEMA42, APP_RDC_SEMA42_GATE, APP_MASTER_INDEX, APP_CUR_MASTER_DID  
PRINTF(" Sema42 acess periph error \r\n" );
```

### (3) RDC control memory

In addition to supporting access to peripherals, RDC can also set access to memory. The details are as follows:

```
RDC_GetDefaultMemAccessConfig(&memConfig); // Get the default configuration of RDC memory  
memConfig.mem = APP_RDC_MEM; //OCRAM  
memConfig.baseAddress = APP_RDC_MEM_BASE_ADDR; //Start address 0x900000  
memConfig.endAddress = APP_RDC_MEM_END_ADDR; //End Address 0x920000  
memConfig.policy &= ~(RDC_ACCESS_POLICY(APP_CUR_MASTER_DID,kRDC_ReadWrite)); //M core cannot be accessed  
RDC_SetMemAccessConfig(APP_RDC, &memConfig); // Save RDC memory configuration  
Accessing this memory segment at this time will trigger a hardware error. The access rights to the memory for the M7 core will be restored in the error handling function.  
PRINTF(" RDC memory access error\r\n" );  
// Make the memory area accessible.  
memConfig.policy |= RDC_ACCESS_POLICY(APP_CUR_MASTER_DID, kRDC_ReadWrite);  
RDC_SetMemAccessConfig(APP_RDC, &memConfig);
```

### 3.15.3 Experimental Phenomena

- (1) Compilation program, as introduced in Section 3.1.1 of the “OKMX8MPQ-C MCU User Compilation Manual V1.0”, manually loads M core program in U-Boot;
- (2) The M7 core is configured to not access GPIO5. Attempting to control GPIO5 output triggers a hardware error. The access rights to GPIO5 are restored afterwards;
- (3) The M7 core is configured with Sema42 mutex. When GPIO5 is not locked, attempting to control GPIO5 output triggers a hardware error. Locking GPIO5 restores access rights to GPIO5;
- (4) The M7 core configuration cannot access the memory. At this time, accessing this segment of memory triggers a hardware error and restores the access right to this segment of memory.

```
RDC Example:  
RDC Peripheral access control  
RDC Peripheral access GPIO5 error  
RDC Peripheral access control with SEMA42  
Sema42 access periph error  
RDC memory region access control  
RDC memory access error
```

RDC Example Success



## 3.16 IPC

8MP features 4 x A53 cores and 1 x M7 core. A53 cores and M7 core exchange information through the RPmsg framework. MailBox interrupts notify the other party of incoming messages. The messages details is placed in shared memory on the DDR. Virtio is used to implement zero-copy data transfer. Detailed information is not described here; please refer to the document “Implementation of 8MP Dual-Core Communication Based on RPmsg” in the application notes folder.

There are two RPmsg routines in the M-core SDK package. rpmsg\_lite\_str\_echo\_rtos is the routine for passing arbitrary characters, andrpmsg\_lite\_pingpong\_rtos is the routine for dual-core interactive communication. We will focus on the rpmsg\_lite\_pingpong\_rtos routine.

### 3.16.1 Hardware Connection

This routine does not have any special wiring.

### 3.16.2 Software Implementation

The routine mainly performs four functions:

1. Initialize rpmsg\_lite;
2. Receive messages sent by Core A and save channel-related information;
3. Use therpmmsg send and receive functions to complete data ping-pong processing;
4. Complete the destruction and reclamation of rpmsg\_lite.

#### (1) rpmsg\_lite initialization

After the M-core program starts, it waits for the A-core to create the RPmsg channel information according to the resource-sharing table. When the creation of the DDR memory, Virtio, and mailbox notification unit at the bottom of the A-core is completed, the M-core informs the A-core that it can formally communicate.

```
my_rpmsg = rpmsg_lite_remote_init((void *)RPMSG_LITE_SHMEM_BASE, RPMSG_LITE_LINK_ID, RL_NO_FLAGS); //Bind shared memory address and channel ID

while (0 == rpmsg_lite_is_link_up(my_rpmsg))
{
}

(void)PRINTF(" Link is up!\r\n");

my_queue = rpmsg_queue_create(my_rpmsg); //Create RPMS queues and related control variables, callback functions, etc.

my_ept = rpmsg_lite_create_ept(my_rpmsg, LOCAL_EPT_ADDR, rpmsg_queue_rx_cb, my_queue); // Receive callback functions and bind them to the bottom virtio buffer
```

```

ns_handle = rpmsg_ns_bind(my_rpmsg, app_nameservice_isr_cb, ((void *)0)); // Dual-core notification message callback
function

//Some delay is introduced to avoid that the A-core message publication is not captured by the M-core. This can happen
when the M-core executes too fast and triggers the ns announcement message before the A-core registers the nameser-
vice_isr_cb

SDK_DelayAtLeastUs(1000000U, SDK_DEVICE_MAXIMUM_CPU_CLOCK_FREQUENCY); //1 sec

(void)rpmsg_ns_announce(my_rpmsg, my_ept, RPMSG_LITE_NS_ANNOUNCE_STRING, (uint32_t)RL_NS_CREATE);

(void)PRINTF(" Nameservice announce sent.\r\n" );

```

#### (2) Receive messages sent by the A-core and save channel-related information

The M-core receives the A-core Hello world! message, thus obtaining the A-core channel number, which will be used for subsequent ping-pong data transmissions.

```
//2 Waiting for A core to send hello world! information
```

```
(void)rpmsg_queue_recv(my_rpmsg, my_queue, (uint32_t *)&remote_addr, helloMsg, sizeof(helloMsg), ((void *)0), RL_BLOCK);

PRINTF(" msg = %s\r\n" ,helloMsg);
```

#### (3) Receive and transmit function of RPmsg to complete data ping-pong processing

The RPmsg transceiver function is used to complete the dual-core data transceiver, and the dual-core respectively adds 1 to the received data until 100 stops.

```
// 3 Ping-pong data receiving and sending. Because the amount of data is very small, there is no need for delay. When
sending large data, the delay between each frame is increased by several to tens of milliseconds.
```

```
while (msg.DATA <= 100U)
```

```
{
    (void)rpmsg_queue_recv(my_rpmsg, my_queue, (uint32_t *)&remote_addr, (char *)&msg, sizeof(THE_MESSAGE), ((void *)0), RL_BLOCK);

    (void)PRINTF(" M7 rx data:%d\r\n" ,msg.DATA);

    msg.DATA++;

    (void)rpmsg_lite_send(my_rpmsg, my_ept, remote_addr, (char *)&msg, sizeof(THE_MESSAGE) , RL_BLOCK);
}
```

#### (4) Destruction and recovery of RPmsg

```
// 4 After sending and receiving the data, perform the necessary cleanup work for rpmsg_lite.
```

```
(void)rpmsg_lite_destroy_ept(my_rpmsg, my_ept);

my_ept = ((void *)0);

(void)rpmsg_queue_destroy(my_rpmsg, my_queue);

my_queue = ((void *)0);

(void)rpmsg_ns_unbind(my_rpmsg, ns_handle);

(void)rpmsg_lite_deinit(my_rpmsg);

my_rpmsg = ((void *)0);

msg.DATA = 0U;
```

### 3.16.3 Experimental Phenomena

(1) Compilation program, as introduced in Section 3.1.1 of the “OKMX8MPQ-C\_MCU User Compilation Manual V1.0” , manually loads M core program in U-Boot;

(2) Linux Operation: The official SDK does not provide ready-made application layer invocation examples. There is only a ping-pong example encapsulated in the driver, which cannot be modified by the user at the application layer. It is necessary to manually load the module, produce the equipment file, write the A-core application program, and complete the data receiving and sending.

① Enter thebootcommand in the A-core debugging serial port to start the A-core.

② In the shell interface, enter:

insmod /lib/modules/5.4.70-2.3.0-00009-gd79f62857237/kernel/drivers/rpmsg/imx\_rpmsg\_tty.ko, This operation is required only once and is not required after subsequent power-up. (Where 5.4.70-2.3.0-00009-gd79f62857237 is the kernel version number, it may vary with Linux version upgrades, requiring users to synchronize and adjust commands accordingly.)

```
root@OK8MP:~# insmod /lib/modules/5.4.70-2.3.0-00009-gd79f62857237/kernel/drivers/rpmsg/imx_rpmsg_tty.ko
[ 121.247048] imx_rpmsg_tty virtio0.rpmsg-openamp-demo-channel.-1.30: new channel: 0x400 -> 0x1e!
[ 121.255935] Install rpmsg tty driver!
```

③ After power-on, you can directly enter or automatically run the script modprobe imx\_rpmsg\_tty. Enter the command ls /dev, and you can see the device file ttyRPMSG30.

④ Copy the rpmsg\_pingpong to /home/root, and enter the chmod 777 rpmsg\_pingpong command to modify the permissions.

⑤ Enter ./rpmsg\_pingpongCommand, the A-core will send hello world! To the M-core! There are 12 characters in total, and the M core carries out ping-pong sending and receiving processing synchronously.

At this time, when debugging the serial port in the M core, you will see the receiving and sending information.

```
RPMSG Ping-Pong FreeRTOS RTOS API Demo...
RPMSG Share Base Addr is 0x55000000
Link is up!
Nameservice announce sent.
M7 rx data:0
M7 rx data:2
M7 rx data:4
M7 rx data:6
M7 rx data:8
M7 rx data:10
M7 rx data:12
```

At this time, when debugging the serial port in the A core, you will see the receiving and sending information.

```
root@OK8MP:~# ./rpmsg_pingpong
open rpmsg tty successed
set rpmsg tty successed
A7 rx data: 1
A7 rx data: 3
A7 rx data: 5
A7 rx data: 7
A7 rx data: 9
A7 rx data: 11
A7 rx data: 13
```

## 4. FREERTOS ROUTINES DESIGN WITH SOFTWARE

In the embedded field, embedded real-time operating systems are being used more and more widely. The use of an embedded real-time operating system (RTOS) can be a more reasonable and effective use of CPU resources, simplify the design of application software, shorten the system development time, and better ensure the real-time and reliability of the system.

FreeRTOS is a mini real-time operating system kernel. As a lightweight operating system, features include: task management, time management, signal volume, message queues, memory management, logging functions, software timers, programming, etc., which can basically meet the needs of smaller systems

M core SDK package Freertos routines are divided into two categories, one is to introduce Freertos system component features, such as signals, mutual exclusion, queues, etc., and the other is to introduce the peripheral interfaces and how to use in Freertos, we pick the routines under these two categories for demonstration respectively.

### 4.1 Freertos-generic

Forlinx OKMX8MP-C development board supports FreeRTOS features. The sample code is as follows:

- \*\*● \*\*freertos\_event: Task event demonstration routine
- \*\*● \*\*freertos\_queue: Demo routine for intertask communication implemented by queue message
- \*\*● \*\*freertos\_mutex: Mutex lock usage routine
- \*\*● \*\*freertos\_sem: Semaphore usage routines
- \*\*● \*\*freertos\_swtimer: Usage of software counters and their callbacks.
- \*\*● \*\*freertos\_tickless: Use LPTMR delayed wake-up or hardware interrupt wake-up routines
- \*\*● \*\*freertos\_generic: task , queue , swtimer , tick hook , semaphore A demonstration routine is utilize in combination.

Because the freertos \_ generic routine uses many FreeRTOS features, we focus on this routine.

#### 4.1.1 Hardware Connection

This routine does not have any special wiring.

#### 4.1.2 Software Implementation

Sample program content includes: task creation, queue, soft timer, system tick clock, semaphore, exception handling.

The details are as follows:

Task creation: The main function creates three tasks: queue sending, receiving, and semaphore.

```
// Create queue receive task
if(xTaskCreate(prvQueueReceiveTask,"Rx",configMINIMAL_STACK_SIZE+166,NULL,mainQUEUE_RECEIVE_TASK_PRIORITY,NULL)!=pdPASS)
// Create Queue send task
if(xTaskCreate(prvQueueSendTask," TX" ,configMINIMAL_STACK_SIZE+166, NULL, mainQUEUE_SEND_TASK_PRIORITY,
NULL )!=pdPASS)
// Create semaphore task
```

```
if(xTaskCreate(prvEventSemaphoreTask,"Sem",configMINIMAL_STACK_SIZE+166,NULL,mainEVENT_SEMAPHORE_TASK_PRIORITY,
NULL)!= pdPASS)
```

**Queue** Queue send task, block for 200ms and then send data to the queue; Queue receive task, block and read the queue, if the data is read correctly, print the number of items received in the queue at that moment.

```
// Queue sending task: send data to the queue after blocking for 200ms
```

```
static void prvQueueSendTask(void *pvParameters)
```

```
{
```

```
TickType_t xNextWakeTime;
```

```
const uint32_t ulValueToSend = 100UL;
```

```
xNextWakeTime = xTaskGetTickCount();
```

```
for (;;) {
```

```
//Task is blocked until 200ms delay is over
```

```
vTaskDelayUntil(&xNextWakeTime, mainQUEUE_SEND_PERIOD_MS);
```

```
//Send data to the queue. A block time of 0 means that the queue will return immediately when it is full
```

```
xQueueSend(xQueue, &ulValueToSend, 0);
```

```
}
```

```
}
```

//The queue receives the task, and the task is blocked to read the queue. If the data is read correctly, the number received by the queue at this time is printed.

```
static void prvQueueReceiveTask(void *pvParameters)
```

```
{
```

```
uint32_t ulReceivedValue;
```

```
for (;;) {
```

```
{
```

```
//The task keeps blocking until data is read from the queue
```

```
xQueueReceive(xQueue, &ulReceivedValue, portMAX_DELAY);
```

// The queue data is consistent with the sending, and the queue receiving quantity + 1 outputs the queue receiving quantity at this time

```
if (ulReceivedValue == 100UL)
```

```
{
```

```
ulCountOfItemsReceivedOnQueue++;
```

```
PRINTF(" Receive message counter: %d.\r\n" , ulCountOfItemsReceivedOnQueue);
```

```
}
```

```
}
```

```
}
```

Soft timer: Set the soft timer period 1s, when the time is up, call the callback function, record the number of times, and print via serial port.

```
//The time for creating the software timer task is 1 s, and the cycle is periodic.
```

```
xExampleSoftwareTimer = xTimerCreate(
```

```
“LEDTimer” ,
```

```
mainSOFTWARE_TIMER_PERIOD_MS,
```

```

pdTRUE,
(void *)0,
vExampleTimerCallback);
// Start the software timer
xTimerStart(xExampleSoftwareTimer, 0);
// Callback function
static void vExampleTimerCallback(TimerHandle_t xTimer)
{
//It enters the callback function once every 1s, and the count increases
ulCountOfTimerCallbackExecutions++;
PRINTF(" Soft timer: %d s.\r\n" , ulCountOfTimerCallbackExecutions);
}

```

**System Tick Clock:** By setting the configTICK\_RATE\_HZ in the FreeRTOSConfig.h file, the task tick interrupt frequency can be configured. When starting the task scheduler, the system will calculate the value to be written to the tick counter based on another variable, configCPU\_CLOCK\_HZ, which represents the CPU frequency. This will initiate the timer interrupt.

// Set the system clock tick to 1000/200=5ms

```
#define configTICK_RATE_HZ           ((TickType_t)200)
```

**Semaphore:** In each system tick clock interrupt, the function vApplicationTickHook is called. After accumulating 500 times, which is equivalent to  $500 * 5\text{ms} = 2.5\text{s}$ , a semaphore is sent. The semaphore task acquires the signal, counts it, and prints the cumulative count.

// The system beat is 5ms, and each  $500 * 5\text{ms} = 2.5\text{s}$  release event semaphore

```
void vApplicationTickHook(void)
```

```
{
```

```
BaseType_t xHigherPriorityTaskWoken = pdFALSE;
```

```
static uint32_t ulCount      = 0;
```

```
ulCount++;
```

```
if (ulCount >= 500UL)
```

```
{
```

//Release the event semaphore in the interrupt

```
xSemaphoreGiveFromISR(xEventSemaphore, &xHigherPriorityTaskWoken);
```

```
ulCount = 0UL;
```

```
}
```

```
}
```

//The task blocks and waits for the semaphore. After it is received, the number of times it is received increases and it is printed through the serial port.

```
static void prvEventSemaphoreTask(void *pvParameters)
```

```
{
```

```
for (;;)
```

```
{
```

The task blocks until the semaphore can be acquired

```
if (xSemaphoreTake(xEventSemaphore, portMAX_DELAY) != pdTRUE)
```

```
{
```

```

PRINTF(" Failed to take semaphore.\r\n" );
}

//The number of times the semaphore is received accumulate.
ulCountOfReceivedSemaphores++;

PRINTF(" Event task is running. Get semaphore :%d \r\n" ,ulCountOfReceivedSemaphores);

}
}

```

Exception Handling: When memory allocation fails, a stack error occurs, or the task is idle, the corresponding function is triggered, allowing users to add custom handling routines.

// Memory allocation failure function, this function is called when memory allocation fails.

```

void vApplicationMallocFailedHook(void)
{
for (;;)
;
}

```

// Stack error checking function, this function is called when a stack overflow occurs.

```

void vApplicationStackOverflowHook(TaskHandle_t xTask, char *pcTaskName)
{
(void)pcTaskName;
(void)xTask;
for (;;)
;
}

```

// Idle task, with the lowest priority and no practical meaning, it simply keeps the CPU busy. Users can add their own functions to it.

```

void vApplicationIdleHook(void)
{
volatile size_t xFreeStackSpace;
xFreeStackSpace = xPortGetFreeHeapSize();
if (xFreeStackSpace > 100)
{
}
}

```

### 4.1.3 Experimental Phenomena

- (1) Compile the program, as described in Chapter 3.1.1 of OKMX8MPQ-C \_ MCU-User Compilation Manual-V1.0, and manually load the M core program in uboot;
- (2) Queue: Every 200ms, the queue send task sends data, and the queue receive task retrieves data. It transitions from a blocked state to a running state, and prints the count;
- (3) Software timer: Every 1s, when the time expires, the callback function is called, and the count is printed;
- (4) Semaphore: Every 5ms, the system clock tick interrupt calls a function. After exceeding 500 times, the semaphore is released. Semaphore task acquires the semaphore, transitions from the blocked state to the running state, and prints the count.

```

Event task is running. Get semaphore :1
Receive message counter: 1.
Receive message counter: 2.
Receive message counter: 3.
Receive message counter: 4.
Soft timer: 1 s.
Receive message counter: 5.
Receive message counter: 6.
Receive message counter: 7.
Receive message counter: 8.
Receive message counter: 9.
Soft timer: 2 s.
Receive message counter: 10.
Receive message counter: 11.
Receive message counter: 12.
Event task is running. Get semaphore :2
Receive message counter: 13.
Receive message counter: 14.
Soft timer: 3 s.
Receive message counter: 15.
Receive message counter: 16.
Receive message counter: 17.
Receive message counter: 18.
Receive message counter: 19.
Soft timer: 4 s.

```

## 4.2 Freertos-Peripheral

The Forlinx OKMX8MP-C development board supports peripherals to fulfill the corresponding functions using FreeRTOS, the sample code is as follows:

- \*\*freertos\_uart: freertos serial port demo routines
- \*\*freertos\_i2c: freertos I2C demo routines
- \*\*freertos\_ecspi: freertos SPI demo routines

Since the freertos\_uart routine uses typical FreeRTOS features, we focus on analyzing this routine.

### 4.2.1 Hardware Connection

This routine does not have any special wiring.

## 4.2.2 Software Implementation

The sample program includes a serial port initialization task, serial port send task, serial port receive task.

The details are as follows:

**Serial port initialization task:** It mainly includes serial port peripheral initialization, sending and receiving mutex, and sending and receiving event groups. The serial port peripheral initialization has been shown in the naked running serial port routine and will not be described in detail here.

```
//Create serial port send mutex
handle->txSemaphore = xSemaphoreCreateMutex();
```

```
//Create serial port receive mutex
```

```
handle->rxSemaphore = xSemaphoreCreateMutex();
```

Creat a send event flag group

```
handle->txEvent = xEventGroupCreate();
```

Creat a receive event flag group

```
handle->rxEvent = xEventGroupCreate();
```

**Serial port sending** Get the signal volume before sending, start the sending process, and set the send completion event flag in the interrupt. After acquiring the event, the send task releases the send semaphore.

1 Get the transmit semaphore

```
if (pdFALSE == xSemaphoreTake(handle->txSemaphore, 0))
```

```
{
```

```
return kStatus_Fail;
```

```
}
```

```
handle->txTransfer.data = (uint8_t *)buffer;
```

```
handle->txTransfer.dataSize = (uint32_t)length;
```

// 2 Blocking transmission

```
status = UART_TransferSendNonBlocking(handle->base, handle->t_state, &handle->txTransfer);
```

```
if (status != kStatus_Success)
```

```
{
```

```
(void)xSemaphoreGive(handle->txSemaphore);
```

```
return kStatus_Fail;
```

```
}
```

// 3 Wait for the send completion event

```
ev = xEventGroupWaitBits(handle->txEvent, RTOS_UART_COMPLETE, pdTRUE, pdFALSE, portMAX_DELAY); //Wait for and determine multiple event bits
```

```
if ((ev & RTOS_UART_COMPLETE) == OU)
```

```
{
```

```
retval = kStatus_Fail;
```

```
}
```

// 4 After sending, release the sending semaphore.

```
if (pdFALSE == xSemaphoreGive(handle->txSemaphore)) // Release the semaphore
```

```
{
```

```
retval = kStatus_Fail;
```

```
}
```

Acquire the semaphore before receiving, call the serial receive function, and set the acquisition event flag in the interrupt. After the receive task obtains the event, release the receive semaphore.

```
// 1 Get the received semaphore
if (pdFALSE == xSemaphoreTake(handle->rxSemaphore, portMAX_DELAY))
{
    return kStatus_Fail;
}

handle->rxTransfer.data    = buffer;
handle->rxTransfer.dataSize = (uint32_t)length;
// 2 Serial port receive function
status = UART_TransferReceiveNonBlocking(handle->base, handle->t_state, &handle->rxTransfer, &n);
if (status != kStatus_Success)
{
    (void)xSemaphoreGive(handle->rxSemaphore);
    return kStatus_Fail;
}

// 3 Get the receive event
ev = xEventGroupWaitBits(handle->rxEvent, RTOS_UART_COMPLETE | RTOS_UART_RING_BUFFER_OVERRUN | RTOS_UART_HARDWARE_MAX_DELAY); // Wait for and determine the receive completion event bit

// 3.1 Hardware receive error
if ((ev & RTOS_UART_HARDWARE_BUFFER_OVERRUN) != 0U)
{
    UART_TransferAbortReceive(handle->base, handle->t_state);
    (void)xEventGroupClearBits(handle->rxEvent, RTOS_UART_COMPLETE); // Clear receive complete event bit
    retval     = kStatus_UART_RxHardwareOverrun;
    local_received = 0;
}

// 3.2 Receive buffer overload error
else if ((ev & RTOS_UART_RING_BUFFER_OVERRUN) != 0U)
{
    UART_TransferAbortReceive(handle->base, handle->t_state);
    (void)xEventGroupClearBits(handle->rxEvent, RTOS_UART_COMPLETE); // Clear receive complete event bit
    retval     = kStatus_UART_RxRingBufferOverrun;
    local_received = 0;
}

// 3.3 Receive complete
else if ((ev & RTOS_UART_COMPLETE) != 0U)
{
    retval     = kStatus_Success;
    local_received = length;
}
```

```

else
{
    retval = kStatus_UART_Error;
    local_received = 0;
}

// 4 Release the receive semaphore
if (pdFALSE == xSemaphoreGive(handle->rxSemaphore))
{
    retval = kStatus_Fail;
}

```

### 4.2.3 Experimental Phenomena

(1) Compile the program, as described in Chapter 3.1.1 of OKMX8MPQ-C \_ MCU-User Compilation Manual-V1.0, and manually load the M core program in uboot;

(2) After the device is powered on, it prints program information through the serial port. Then, by entering four characters through the keyboard, the M-core debugging serial port will echo those characters. The process is repeated to demonstrate that the program is running successfully.



FreeRTOS UART driver example!  
123456789012

## 5. USE OF OK8MP PLATFORM M-CORE SDK

- **Note: Please do not skip this paragraph.**

The development environment is the hardware and software platform that developers need during the development process. The development environment is not a fixed style. Here, we will explain in detail how to build an M-core development environment. If you already have a good understanding of embedded M-core development, you can build the environment according to your own needs. If the environment is different from this manual and an error is reported, you can search for relevant information from the TI forum and website to solve the problem. The environment setup method and the development environment provided in this manual have been tested by Forlinx. If you are not very familiar with the development of embedded M-core, please follow the method provided by Forlinx to set up the environment, or use the environment provided by Forlinx.

Please refer to the MCU User Compilation Manual for the details.