# Using Freescale MQX™ RTOS on multi-core devices

| PRODUCT:         | Freescale MQX™ RTOS                                            |
|------------------|----------------------------------------------------------------|
| PRODUCT VERSION: | 3.8.0                                                          |
| DESCRIPTION:     | Using Freescale MQX™ RTOS on multi-core devices, version 3.8.0 |
| RELEASE DATE:    | February 15 <sup>th</sup> , 2012                               |
|                  |                                                                |



#### How to Reach Us:

#### Home Page:

www.freescale.com

#### Web Support:

http://www.freescale.com/support

#### **USA/Europe or Locations Not Listed:**

Freescale Semiconductor, Inc.
Technical Information Center, EL516
2100 East Elliot Road
Tempe, Arizona 85284
1-800-521-6274 or +1-480-768-2130
www.freescale.com/support

#### Europe, Middle East, and Africa:

Freescale Halbleiter Deutschland GmbH Technical Information Center Schatzbogen 7 81829 Muenchen, Germany +44 1296 380 456 (English) +46 8 52200080 (English) +49 89 92103 559 (German) +33 1 69 35 48 48 (French) www.freescale.com/support

#### Japan:

Freescale Semiconductor Japan Ltd. Headquarters ARCO Tower 15F 1-8-1, Shimo-Meguro, Meguro-ku, Tokyo 153-0064 Japan 0120 191014 or +81 3 5437 9125 support.japan@freescale.com

#### Asia/Pacific:

Freescale Semiconductor China Ltd. Exchange Building 23F No. 118 Jianguo Road Chaoyang District Beijing 100022 China +86 10 5879 8000 support.asia@freescale.com

#### For Literature Requests Only:

Freescale Semiconductor Literature Distribution Center P.O. Box 5405
Denver, Colorado 80217
1-800-441-2447 or +1-303-675-2140
Fax: +1-303-675-2150
LDCForFreescaleSemiconductor@hibbertgroup.com

Information in this document is provided solely to enable system and software implementers to use Freescale Semiconductor products. There are no express or implied copyright licenses granted hereunder to design or fabricate any integrated circuits or integrated circuits based on the information in this document.

Freescale Semiconductor reserves the right to make changes without further notice to any products herein. Freescale Semiconductor makes no warranty, representation or guarantee regarding the suitability of its products for any particular purpose, nor does Freescale Semiconductor assume any liability arising out of the application or use of any product or circuit, and specifically disclaims any and all liability, including without limitation consequential or incidental damages. "Typical" parameters that may be provided in Freescale Semiconductor data sheets and/or specifications can and do vary in different applications and actual performance may vary over time. All operating parameters, including "Typicals", must be validated for each customer application by customer's technical experts. Freescale Semiconductor does not convey any license under its patent rights nor the rights of others. Freescale Semiconductor products are not designed, intended, or authorized for use as components in systems intended for surgical implant into the body, or other applications intended to support or sustain life, or for any other application in which the failure of the Freescale Semiconductor product could create a situation where personal injury or death may occur. Should Buyer purchase or use Freescale Semiconductor products for any such unintended or unauthorized application, Buyer shall indemnify and hold Freescale Semiconductor and its officers, employees, subsidiaries, affiliates, and reasonable attorney fees arising out of, directly or indirectly, any claim of personal injury or death associated with such unintended or unauthorized use, even if such claim alleges that Freescale Semiconductor was negligent regarding the design or manufacture of the part.



Freescale™ and the Freescale logo are trademarks of Freescale Semiconductor, Inc. ARC, the ARC logo, ARCangel, ARCform, ARChitect, ARCompact, ARCtangent, BlueForm, CASSEIA, High C/C++, High C++, iCon186, MetaDeveloper, MQX, Precise Solution, Precise/BlazeNet, Precise/EDS, Precise/MFS, Precise/MQX, Precise/MQX Test Suites, Precise/RTCS, RTCS, SeeCode, TotalCore, Turbo186, Turbo86, V8 µ RISC, V8 microRISC, and VAutomation are trademarks of ARC International. High C and MetaWare are registered under ARC International.

All other product or service names are the property of their respective owners.

© Freescale Semiconductor, Inc. 2011. All rights reserved.

Rev. 08 12/2011

# **Table of Contents**

| Using Freescale MQX™ RTOS on multi-core devices         | i |
|---------------------------------------------------------|---|
| 1 Support for multi-core operation in MQX               | 2 |
| 1.1 Introduction                                        | 2 |
| 1.2 Execution modes of multi-core MPX parts             | 2 |
| 2 Principle of operation                                | 3 |
| 2.1 Memory map                                          |   |
| 3 Design and implementation of a multi-core application |   |
| 3.1 Entry point (main function) of a multi-core BSP     | 4 |
| 3.2 Selecting tasks to start on each core               | 4 |
| 3.3 Assigning HW devices to particular core             | 5 |
| 3.4 Inter-core synchronization                          | 5 |
| 3.4.1 SEMA4 peripheral                                  | 5 |
| 3.4.2 Description of core_mutex API functions           | 5 |
| 3.4.3 Example of core_mutex usage                       | 6 |
| 3.5 Inter-processor communication                       | 7 |
| 3.5.1 Example of IPC setup on a dual-core device        | 7 |

## 1 Support for multi-core operation in MQX

## 1.1 Introduction

The MQX 3.8 introduces support for multi-core operation on PowerPC platforms of MPX family having multiple cores of the same type.

## 1.2 Execution modes of multi-core MPX parts

The multi-core parts of MPX family may operate either in lock-step mode (LSM) or decoupled parallel mode (DPM). In LSM, both cores execute the same code at the same time (redundancy for safety applications) while in DPM the cores execute code independently on each other, so there are effectively two processors available. The mode is selected by LSM/DPM bit in shadow FLASH area. The mode cannot be changed in runtime, thus proper mode needs to be set prior running application on the MCU, please refer to part specific notes in the getting started document.

From the application point of view, part configured in LSM behaves like it was just a single processor so there is virtually no difference to running MQX on a single core part (besides early initialization handled by startup code in the BSP). This document rather deals with the other case, the DPM which requires the application to be designed accordingly to take advantage of multi-core execution. The following text presumes a dual-core system is used (e.g. PXS30).

## 2 Principle of operation

Current multi-core operation of MQX involves starting multiple instances of MQX, one on each core. Each instance of MQX may use of different set of drivers, typically services distinct set of peripherals and different set of tasks.

## 2.1 Memory map

Both instances of MQX are started from a single image, thus they share the same code memory, while the data memory is separate for each instance. The principle of linking process requires the data memory to be mapped to the same address space on both cores. That means the global variables in both instances of MQX are located at the same (virtual) address but in fact they occupy different locations of physical memory. This is achieved by proper configuration of MMUs (memory management units) which takes place in the startup code and BSP initialization.

Figure 1 illustrates the memory mapping. Please note that the illustration depicts the basic concept and does not cover details like shared memory for inter-processor communication (IPC), uncached memory for peripheral buffers, etc.



Figure 1: multi-core memory mapping

## 3 Design and implementation of a multi-core application

## 3.1 Entry point (main function) of a multi-core BSP

The following code snippet shows the main function which is part of BSP supporting multi-core operation

```
int main(void)
{
    extern const MQX_INITIALIZATION_STRUCT MQX_init_struct;
    extern const MQX_INITIALIZATION_STRUCT MQX_init_struct_1;

    /* Start MQX */
    if (_psp_core_num() == 0) {
        _mqx( (MQX_INITIALIZATION_STRUCT_PTR) &MQX_init_struct );
    } else {
        _mqx( (MQX_INITIALIZATION_STRUCT_PTR) &MQX_init_struct_1 );
    }
    return 0;
}
```

Compared to a single-core operation the main function contains conditional statement testing the core number the code is running on and multiple calls to \_mqx function with pointers to different initialization structures. Only one of the calls to \_mqx function takes place on each core.

## 3.2 Selecting tasks to start on each core

On a single core device one or more tasks are defined upon MQX start according to MQX\_template\_list array:

```
const TASK_TEMPLATE_STRUCT MQX_template_list[] =
{
   /*
    Task Index, Function, Stack, Priority, Name, Attributes, Param, Time Slice
   */
    { 1, test_task, 1500, 8, "test_task", MQX_AUTO_START_TASK, 0, 0 },
};
```

This basic concept is followed the same way also in the case of a multi-core system. For starting task on second core, there is a separate TASK\_TEMPLATE\_STRUCT with identifier MQX\_template\_list\_1 defining tasks to run in the same fashion as for the first core.

If there is no MQX\_template\_list\_1 defined then no tasks will be started on the second core (besides the idle task). However, the second instance of MQX itself will be started in any case.

Freescale MQX Getting Started

The sets of tasks executed on the cores need not to be distinct. Multiple instances of the same task may be running on multiple cores, if the task is designed for such operation (e.g. handles concurrent access to HW devices, etc.). Please note that tasks running on different cores do not share the same global data (see chapter 2). Inter-process communication (IPC) may be used to pass data between tasks running on different cores.

## 3.3 Assigning HW devices to particular core

Typically it is desired that the instances of MQX running on different cores use distinct set of peripherals. Initialization of peripherals in BSP is performed conditionally according to defined macros containing masks of cores the particular peripheral should be used on. The test whether the peripheral is enabled or disabled on particular core is done by calling PSP function \_psp\_core\_peripheral\_enable, which returns true only if executed on core for which corresponding bit of the parameter is set to 1.

#### Example:

```
if (_psp_core_peripheral_enabled(CORECFG_SPI_0)) {
     _dspi_polled_install("spi0:", &_bsp_dspi0_init);
}
```

#### Whereas CORECFG SPI 0 is defined:

or in special cases, if the same peripheral is to be used on both cores:

```
#define CORECFG SPI 0 (CORE 0 | CORE 1)
```

The values of CORECFG may be defined in user\_config.h, otherwise default values from PSP are used.

The test using the \_psp\_core\_peripheral\_enabled function may be also done in BSP specific module of a driver. Typically this is the case when it is needed to conditionally enable access to given peripheral in the peripheral bridge (e.g. \_bsp\_dspi\_enable\_access function).

# 3.4 Inter-core synchronization

Synchronization of tasks running on different cores is supported by the core\_mutex driver providing mutual exclusion mechanism between taks running on different cores.

#### 3.4.1 SEMA4 peripheral

MPX family devices feature SEMA4 units containing gates with mutual exclusion mechanism and ability to notify core(s) by an interrupt when a gate is unlocked which provides with efficient way to unblock a waiting task without necessity of a busy loop checking for locked/unlocked status.

The SEMA4 unit is used as an underlying device by the core\_mutex driver. There may be several SEMA4 units (one per core) each having multiple gates with mutual exclusion mechanism.

#### 3.4.2 Description of core mutex API functions

Full prototypes of the functions may be found in source/io/core\_mutex/core\_mutex.h

#### \_core\_mutex\_install(const CORE\_MUTEX\_INIT\_STRUCT \* init\_ptr)

The function performs initial installation of the device. This is done only once on each core, typically upon system initialization in BSP.

#### \_core\_mutex\_create(uint\_32 core\_num, uint\_32 mutex\_num, uint\_32 policy)

The function allocates core\_mutex structure and returns a handle to it. The mutex to be created is identified by core and mutex (gate) number. The handle is used to reference the created mutex in calls to other core\_mutex API functions. This function is to be called only once for each mutex. The policy parameter determines behavior of task queue associated with the mutex.

### \_core\_mutex\_create\_at(CORE\_MUTEX\_PTR mutex\_ptr, ....)

Function similar to \_core\_mutex\_create but it does not use dynamic allocation of CORE\_MUTEX structure. A pointer to pre-allocated memory area is passed by the caller instead.

## \_core\_mutex\_get(uint\_32 core\_num, uint\_32 mutex\_num)

Returns handle to an already created mutex.

#### \_core\_mutex\_lock(CORE\_MUTEX\_PTR core\_mutex\_ptr)

The function attempts to lock a mutex. If the mutex is already locked by another task the function blocks and waits until it is possible to lock the mutex for the calling task.

## \_core\_mutex\_trylock(CORE\_MUTEX\_PTR core\_mutex\_ptr)

The function attempts to lock a mutex. If the mutex is successfully locked for the calling task, TRUE is returned. If the mutex is already locked by another task the function does not block but rather returns FALSE immediately.

## \_core\_mutex\_unlock(CORE\_MUTEX\_PTR core\_mutex\_ptr)

Unlocks the given mutex.

#### \_core\_mutex\_owner(CORE\_MUTEX\_PTR core\_mutex\_ptr)

Returns number of core number currently "owning" the mutex.

#### 3.4.3 Example of core mutex usage

The following code snippet shows usage of core\_mutex API. The code presumes that \_core\_mutex\_install is already called (which typically takes place during BSP initialization).

```
void test_task(uint_32 initial_data)
{
    CORE_MUTEX_PTR cm_ptr;

cm_ptr = _core_mutex_create( 0, 1, MQX_TASK_QUEUE_FIFO );

while (1) {
    _core_mutex_lock(cm_ptr);
    /* mutex locked here */
    printf("Core%d mutex locked\n", _psp_core_num());
    _time_delay((uint_32)rand() % 20 );
    _core_mutex_unlock(cm_ptr);
    /* mutex unlocked here */
    printf("Core%d mutex unlocked\n", _psp_core_num());
    _time_delay((uint_32)rand() % 20 );
}
```

## 3.5 Inter-processor communication

The inter-processor communication (IPC) may be used to pass data between tasks running on different cores. For actual data transport a PCB (packet control block) device is used, particularly PCB using shared memory area. For generic information concerning IPC operation, please refer to MQX User's guide.

## 3.5.1 Example of IPC setup on a dual-core device

There are two instances of routing table, one per core:

PCB over shared memory will be used as transport layer for IPC, thus init structures are needed. Structure describing shared memory comes first.

Generic PCB init structure references shared memory init structure.

```
const IPC PCB INIT STRUCT pcb init =
  /* IO PORT NAME */
                                "pcb shmem:",
  /* DEVICE INSTALL? */
                                 io pcb shm install,
  /* DEVICE INSTALL PARAMETER*/ (pointer) &pcb shm init,
  /* IN MESSAGES MAX SIZE */ sizeof(THE MESSAGE),
  /* IN MESSAGES TO ALLOCATE */ 8,
  /* IN MESSAGES TO GROW */ 8,
  /* IN MESSAGES MAX ALLOCATE */ 16,
  /* OUT_PCBS_INITIAL */
                               8,
  /* OUT PCBS TO GROW */
                                8,
  /* OUT PCBS MAX */
                                16
};
```

Freescale Semiconductor 7

Freescale MQX Getting Started

Protocol init structure defines that PCB will be used as a transport layer for IPC.

```
const IPC_PROTOCOL_INIT_STRUCT ipc_init_table[] =
{
     { _ipc_pcb_init, (pointer)&pcb_init, "core_ipc_pcb", QUEUE_TO_REMOTE },
     { NULL, NULL, NULL, 0}
};
```

There are two instances of IPC init structure, one per core.

```
const IPC_INIT_STRUCT core0_ipc_init = {
    core0_routing_table,
    ipc_init_table
};

const IPC_INIT_STRUCT core1_ipc_init = {
    core1_routing_table,
    ipc_init_table
};
```

The \_ipc\_task has to be an autostart task on both cores. Pointer to IPC initialization structure for given core is passed as CREATION\_PARAMETER to \_ipc\_task.

Now tasks running on different cores may communicate with each other by passing messages.

```
my_qid = _msgq_open(MAIN_QUEUE,0);
msgpool = _msgpool_create(sizeof(THE_MESSAGE), 8, 8, 0);
msg_ptr = (THE_MESSAGE_PTR)_msg_alloc(msgpool);

if (msg_ptr != NULL) {
    msg_ptr->HEADER.TARGET_QID =
        _msgq_get_id((_processor_number)dest_core,REMOTE_QUEUE);
    msg_ptr->HEADER.SOURCE_QID = my_qid;
    msg_ptr->DATA = 0;
    msgq_send(msg_ptr);
}

...

msg_ptr = _msgq_receive(MSGQ_ANY_QUEUE,0);
```

See "Messages" section of MQX User's manual for details concerning message queues.