# **MCU Developer Guide**

Release Version: 1.1

E-mail: frank.wang@rock-chips.com

Release Date: 2017.12

Classified Level: Publicity

#### **Preface**

This document introduces the Cortex-M0 Devices Generic User Guide.

### **Chipset Version**

| NAME   | LINUX KERNEL VERSION |
|--------|----------------------|
| RK3399 | 4.4                  |

**Intended audience** This document (this guide) is intended primarily for the following readers:

Field Application Engineer

Software Engineer

## **Revision history**

| REVISION DATE | VERSION | AUTHOR     | REVISION DESCRIPTION |
|---------------|---------|------------|----------------------|
| 2017-09-20    | V1.0    | Frank.Wang | Initial version      |
| 2017-12-27    | V1.1    | Frank.Wang | modify format        |

### **MCU Developer Guide**

- 1 Rockchip MCU Overview
- 2 Primary Development
  - 2.1 Configuration Preparing
    - 2.1.1 Boot Address
    - 2.1.2 Address mapping
    - 2.1.3 Clock Configuration
    - 2.1.4 Reset Cancellation
  - 2.2 Other Configuration
    - 2.2.1 Enable JTAG Configuration
  - 2.3 MCU Communicated with SoC
    - 2.3.1 Mailbox
    - 2.3.2 Share Memory
- 3 Demo Code
  - 3.1 Fetch the Code
  - 3.2 Code Intruduction
    - 3.2.1 Directory

3.2.2 compiling3.2.3 Interrupt Development

4 MCU Debug

- 4.1 JTAG Debug
- 4.2 Console Printing
- 4.3 Read and Write Register

Reference Document

## 1 Rockchip MCU Overview

ARM® Cortex®-M processor family provide some benefits to developers, including: simple, easy-to-use, highly efficient, ultra-low power operation.

At the same time, the Cortex-M processor provides developers more functions at a lower cost, which has significant advantages in code reuse and development efficiency. For this advantage, it is widely used in the field of embedded devices. The simply introduction of Cortex-M0 and Cortex-M3 is as follows:

- The Cortex-M0 processor uses the ARMv6-M architecture based on a highly integrated, low-power 32-bit processor core; it uses the von Neumann architecture and base on the 16-bit Thumb instruction set included Thumb-2 technology.
- The Cortex-M3 is a 32-bit processor core used the ARMv7-M architecture. It uses Harvard architecture, which has an independent instruction bus and data bus, which allows the fetch and data access to be parallel.

Based on the above advantages of ARM® Cortex®-M, the MCUs currently integrated on the Rockchip SoC are described below:

 The RK3399 integrates two Cortex-M0s, "PMU M0" is used by ATF and the other "Perilp M0" are open to customers developed by themselves.

## **2 Primary Development**

## 2.1 Configuration Preparing

This chapter mainly introduces the basic method of Rockchip MCU development based on RK3399 Perilp M0.

#### 2.1.1 Boot Address

Take genetic boot way "miniloader + ATF + u-boot" as an example.

With this method, usually need to pack the MCU bin and ATF bin as "trust.img". Therefore, the following package configuration is required to add to the U-Boot.

```
tools/rk_tools/RKTRUST/RK3399TRUST.ini
...

[BL30_OPTION]

SEC=1

PATH=tools/rk_tools/bin/rk33/rk3399bl30_v1.00.bin

ADDR=0x00080000

...
```

As above shows, "PATH" is the storage path of MCU bin file and "ADDR" is the DDR address where MCU load start (also means the MCU's start address). Of course, this address needs a secure address reserved in DDR. This address will be passed to the miniloader, which will load the MCU bin from ROM to this address of DDR.

Of course, if you use U-Boot or other loader as the first-level boot loader, you can also pack and load the M0 firmware referred to above method.

For the MCU bin compiling, refer to chapter 3.2.2 Compilation.

### 2.1.2 Address mapping

The Cortex-M0/Cortex-M3 has a fixed Memory Map, which facilitates easy porting of software between different systems. The address space is divided into many different segments, you can refer to Chapter 2.2 chapter Memory model in Cortex-M0 Devices Generic User Guide and Chapter 7.4.2 in Rockchip RK3399 TRM. Commonly, we only need configuring the MCU's 0x000000000-0x1FFFFFFF address mapping.

It is note that the address mapping of the RK3399 M0 needs to be configured via SGRF, so please be sure to configure it in the module that can access the SGRF (usually in the miniloader or ATF). Take 2.1.1 Boot Address Load address as an example. The specific memory mapping configuration is as follows:

• Boot address configuration

```
1 sgrf_perilp_m0_con7 = 0xf << (4 + 16) | (0x080000 >>
      28) & 0x0f
2 sgrf_perilp_m0_con15 = 0xffff << 16 | (0x80000 >> 12) &
      0xffff
```

• Peripheral address configuration

The Rockchip MCU has configured the map address of the peripheral (0x40000000-0x5FFFFFFF) by default, which like:

```
ADDR\_MCU = ADDR\_CA72 - 0xB8000000
```

The peripherals here are those listed in the Rockchip RK3399 TRM Chapter 2 System Overview.

## 2.1.3 Clock Configuration

The Rockchip MCU clock source can be selected from CPLL or GPLL. Refer to the RK3399 TRM Chapter 3 CRU. In that chapter it is point out that the CLK configuration register is "CRU CLKSEL CON24 (0x0160)", where:

bit[15]: Select clock source, 1'b0: CPLL; 1'b1: GPLL bit[12:8]: Set the frequency-division, which is used to configure the operating frequency of the MCU.

• U-Boot reference code

```
1 arch/arm/cpu/armv8/rk33xx/clock-rk3399.c
 2 #ifdef CONFIG_PERILP_MCU
       /* peril m0 clk = 300MHz, select gpll as the
    source clock */
        clk_parent_hz = RKCLK_GPLL_FREQ_HZ;
5
        clk_child_hz = 300000000; /* HZ */
        div = rkclk_calc_clkdiv(clk_parent_hz,
    clk_child_hz, 1);
 7
8
        div = div ? (div - 1) : 0;
        cru_writel((1 << 31) | (0x1F << 24) | (1 << 15) |</pre>
    (div \ll 8),
10
    CRU_CLKSELS_CON(24));
11 #endif
```

## 2.1.4 Reset Cancellation

The final step in the MCU's implement is to perform a reset cancellation. Read the register information in the Chapter 3 CRU Rockchip RK3399 TRM . The reset cancel register of the RK3399 Perilp M0 is:

 $PMUCRU_SOFTRST_CONO(0x0110)$   $PMUCRU_SOFTRST_CONO[5:0] = 4b'0000.$ 

reference code

```
1 arch/arm/cpu/armv8/rk33xx/clock-rk3399.c
2 #ifdef CONFIG_PERILP_MCU
3    /* perilp m0 dereset */
4    cru_writel(0x00160000, CRU_SOFTRSTS_CON(11));
5 #endif
```

Tip: The configuration of the clock and reset can be placed where the MCU expected to run. The Rockchip SDK is currently placed in U-Boot.

## 2.2 Other Configuration

### 2.2.1 Enable JTAG Configuration

In MCU development, it is often necessary to use JTAG to track, debug and solve problems. The Rockchip MCU JTAG interface implements SWD (2-wire) mode and requires configuration of JTAG iomux to connect.

The iomux configuration details of the RK3399 Perilp M0 can be found in Chapter 7.3 RK3399 TRM, including the following two registers:

```
1 GRF_GPIO4B_IOMUX[9:8] = 2'b10
2 GRF_GPIO4B_IOMUX[9:8] = 2'b10
```

### 2.3 MCU Communicated with SoC

#### 2.3.1 Mailbox

The integrated mailbox on the Rockchip SoC has 4 channels triggered by interrupts. And data is passed through shared memory. Rockchip MCU can communicate with SoC via the mailbox peripheral. RK3399 Mailbox programming can be found in Chapter 21 Mailbox Rockchip RK3399 TRM; RK3368 Mailbox refer to Chapter 11 Mailbox chapter in Rockchip RK3368 TRM.

Currently in the Linux 4.4 Kernel, the upper layer of the Mailbox Driver framework uses the ARM SCPI protocol, so you need to enable "CONFIG\_RK3368\_MBOX" and "CONFIG\_RK3368\_SCPI\_PROTOCOL" in Kernel. At the same time, the MCU code also needs the mail driver and SCPI protocol support.

Kernel DTS can be configured with reference to the code below.

```
mailbox: mailbox@ff6b0000 {
 2
        compatible = "rockchip, rk3368-mbox-legacy";
 3
        reg = <0x0 \ 0xff6b0000 \ 0x0 \ 0x1000>
 4
               <0x0 0xff8cf000 0x0 0x1000>; /* the end 4k
    of sram */
 5
        interrupts = <GIC_SPI 146 IRQ_TYPE_LEVEL_HIGH>,
 6
                      <GIC_SPI 147 IRQ_TYPE_LEVEL_HIGH>,
 7
                      <GIC_SPI 148 IRQ_TYPE_LEVEL_HIGH>,
8
                      <GIC_SPI 149 IRQ_TYPE_LEVEL_HIGH>;
9
        clocks = <&cru PCLK_MAILBOX>;
        clock-names = "pclk_mailbox";
        \#mbox-cells = <1>;
11
        status = "disabled";
13
    };
14
    mailbox_scpi: mailbox-scpi {
        compatible = "rockchip,rk3368-scpi-legacy";
16
        mboxes = <&mailbox 0>, <&mailbox 1>, <&mailbox</pre>
17
    2>;
18
        chan-nums = <3>;
        status = "disabled";
19
    };
```

## 2.3.2 Share Memory

The Rockchip MCU can also communicate with SoC via sharing memory. For example, partitioning a space in INTMEM (SRAM) and configuring it to be accessible to both the master SoC and the MCU, which implements share memory communication.

The Rockchip MCU can also communicate with the master SoC via UART or other methods.

## 3 Demo Code

## 3.1 Fetch the Code

Git repository path:

- ssh://git@10.10.10.29/rk/mcu or https://github.com/frawang/rk-mc u.git.
- The code at "10.10.10.29" can refer to the rk3399-pmu-m0 branch
- the github code can refer to the rk3399-box-m0 branch.

### 3.2 Code Intruduction

## 3.2.1 Directory

```
rk-mcu>ls -R
 3 build include Makefile src
4
5
   ./build:
   arm-gcc-link.ld RK3399M0
7
8
   ./build/RK3399M0:
9
   bin obj
11
   ./build/RK3399M0/bin:
   RK3399MO.bin RK3399MO.dump RK3399MO.elf
12
    RK3399M0.map
13
    ./build/RK3399M0/obj:
14
15
    main.o startup.o
16
    ./include:
17
18
    mcu.h remotectl_pwm.h rk3399.h
19
   ./src:
    main.c main.c.bk remotectl_pwm.c startup.c
```

- build: Use to store the compiled "obj" file and "bin" file.
- include: Header files.
- src: Source files.

The "startup.c" is the M0 entry program, which mainly includes the M0 interrupt vector table and the interrupt execution function. The "main.c" is the main functions of the M0 program.

## 3.2.2 compiling

The cross-compilation tool chain implements gcc-arm-none-eabi-v4.8 or newer. The compilation method as follows:

```
rk-mcu>make help
   usage: make PLAT=<RK3399M0> <all|clean|distclean>
4 PLAT is used to specify which platform you wish to
    build.
 5 If no platform is specified in first time, PLAT
    defaults to:
6
   Supported Targets:
8
    all Build all the project
9
    clean Clean the current platform project
10
      distclean Clean the current project and delete
    .config
11
12 example: build the targets for the RK3399MO project:
13
     make PLAT=RK3399M0
```

After compiling, "build/RK3399M0/bin/RK3399M0.bin" will be generated, Then copy "RK3399M0.bin" to the directory "tools/rk\_tools/bin/rk33/" in the U-Boot and rename it to "rk3399bl30\_v1.00.bin", After that, follow 2.1.1 Boot Address instruction, recompile U-Boot, and packing M0 bin into trust.img .

## 3.2.3 Interrupt Development

M0 interrupt vector table can be seen at Cortex-M0 Devices Generic User Guide Chapter 2.3 Exception model.

The reference Demo program in "src/startup.c" has the following reference code:

```
1 /**
2  * The minimal vector table for a Cortex M3. Note
  that the proper constructs
3  * must be placed on this to ensure that it ends up
  at physical address
4  * 0x00000000.
5  */
```

```
__attribute__ ((used, section(".isr_vector")))
   void (* const g_pfnVectors[])(void) =
8
9
       /* core Exceptions */
10
        (void *)&pstack[STACK_SIZE], /* the initial stack
    pointer */
11
       reset_handler,
12
       nmi_handler,
13
       hardware_fault_handler,
14
       0,0,0,0,0,0,0,
15
       svc_handler,
16
       0,0,
17
       pend_sv_handler,
18
       systick_handler,
19
       /* external exceptions */
21
       22
       0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
23 };
```

There are 16 internal exceptions of Coretex-M0 ,which from "g\_pfnVectors[0]" to "g\_pfnVectors[15]", you can register and implement the reference exception handling function according to the actual application.

The Cortex-M0 can handle 32 external interrupts, refer to "g\_pfnVectors[16]" to "g\_pfnVectors[47]".

RK3399 Perilp M0 introduces an interrupt arbiter that extends 32 external interrupts to 256, see Chapter 7.4.6 Interrupt Source Arbiter for PERILPM0 in Rockchip RK3399 TRM. Of course, to use the arbiter, you are required to configure the reference mask bit. After receiving the interrupt, M0 needs to judge the specific interrupt source according to the corresponding mask bit.

The Arbiter's interface can be found in "include/rk3399.h"

MO\_INT\_ARB\_SET\_MASK() // Set interrupt mask MO\_INT\_ARB\_GET\_FLAG() // Get interrupt bit

About the external interrupt supported by RK3399 Perilp M0 , please refer to Chapter 2.4 System Interrupt Connection for Cortex-M0 in Rockchip RK3399 TRM.

## 4 MCU Debug

## 4.1 JTAG Debug

- Set iomux, tck, tms about JTAG in GRF, please refer to 2.2.1 Enabling JTAG configuration.
- Development board JTAG switch or "tck/tms" switch to the MCU;
- DS-5, ICE or Jlink connect to m3/m0 for debugging.

## 4.2 Console Printing

- M0 can access the UART register directly for printing and debugging.
- MCU implements the same UART as the SoC master, it is recommend to turn off M0 printing during normal operation to prevent the system being abnormal due to irregular UART access.

## 4.3 Read and Write Register

- The system can be stay to the U-Boot or Kernel command line and the MCU status register can be read via io to view the MCU status.
- It is also one method to write the running status of the MCU key point to the idle GRF register and then read its value on the U-Boot or kernel command line to judge the current running state of the MCU.

## **Reference Document**

This section lists relevant documents published by third parties:

Cortex-M0 Devices Generic User Guide Cortex-M0 Technical Reference Manual ARM Cortex-M3 Processor Technical Reference Manual Cortex-M3 Devices Generic User Guide Rockchip RK3399 TRM V0.4 Rockchip RK3368 TRM V2.0