

#### Disclaimer

The information written in this document is assumed to be accurate without guarantee. The information in this manual is subject to change for functional or performance improvements without notice. SEGGER Microcontroller GmbH (SEGGER) assumes no responsibility for any errors or omissions in this document. SEGGER disclaims any warranties or conditions, express, implied or statutory for the fitness of the product for a particular purpose. It is your sole responsibility to evaluate the fitness of the product for any specific use.

#### **Copyright notice**

You may not extract portions of this manual or modify the PDF file in any way without the prior written permission of SEGGER. The software described in this document is furnished under a license and may only be used or copied in accordance with the terms of such a license.

© 2010-2025 SEGGER Microcontroller GmbH, Monheim am Rhein / Germany

#### Trademarks

Names mentioned in this manual may be trademarks of their respective companies.

Brand and product names are trademarks or registered trademarks of their respective holders.

#### **Contact address**

SEGGER Microcontroller GmbH

Ecolab-Allee 5 D-40789 Monheim am Rhein

Germany

| Tel.      | +49 2173-99312-0    |
|-----------|---------------------|
| Fax.      | +49 2173-99312-28   |
| E-mail:   | support@segger.com* |
| Internet: | www.segger.com      |

<sup>\*</sup>By sending us an email your (personal) data will automatically be processed. For further information please refer to our privacy policy which is available at https://www.segger.com/legal/privacy-policy/.

#### **Manual versions**

This manual describes the current software version. If you find an error in the manual or a problem in the software, please inform us and we will try to assist you as soon as possible. Contact us for further information on topics or functions that are not yet documented.

Print date: June 5, 2025

| Software | Revision | Date   | Ву | Description                                                                                                                                                                                                                                                                                                                                                                      |  |
|----------|----------|--------|----|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|
| 5.20.0.0 | 0        | 250605 | MC | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 5.18.3.0 | 0        | 240719 | MC | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 5.18.0.1 | 0        | 230307 | MC | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 5.18.0.0 | 0        | 221111 | ММ | Chapter "CPU and compiler specifics" updated.<br>Chapter "RTT and SystemView" added.                                                                                                                                                                                                                                                                                             |  |
| 5.16.1.1 | 0        | 220707 | ММ | Chapter "Libraries" and "VFP and NEON support" updated.                                                                                                                                                                                                                                                                                                                          |  |
| 5.16.1.0 | 0        | 220201 | ММ | Chapter "CPU and compiler specifics" updated.                                                                                                                                                                                                                                                                                                                                    |  |
| 5.14.0.0 | 1        | 210708 | ММ | Chapter "VFP and NEON support" updated.                                                                                                                                                                                                                                                                                                                                          |  |
| 5.14.0.0 | 0        | 210618 | ММ | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 5.8.2.1  | 0        | 200417 | MC | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 5.8.2.0  | 0        | 200304 | MM | New software version.<br>Chapter "MMU/MPU and cache support" updated.                                                                                                                                                                                                                                                                                                            |  |
| 5.06     | 0        | 191010 | MM | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 5.02     | 0        | 180710 | MC | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 4.40     | 0        | 180201 | MC | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 4.38     | 0        | 171013 | MC | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 4.36     | 0        | 170802 | MC | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 4.34     | 1        | 170707 | MC | Updated chapter "libraries" to include descriptions for both IAR EWARM 6/7 and IAR EWARM 8.                                                                                                                                                                                                                                                                                      |  |
| 4.34     | 0        | 170330 | MC | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 4.16     | 0        | 160406 | RH | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 4.14     | 0        | 151130 | TS | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 4.12b    | 0        | 150928 | TS | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 3.90     | 0        | 140310 | AW | <pre>New software version.<br/>Global enable and disable interrupts. Described in the generic manual.<br/>Chapter 4, "Debug outputs, printf" and SWI_Handler() description<br/>added.<br/>Chapter 7, "MMU and cache support", function descripition added:<br/>OS_ARM_MMU_v2p()<br/>OS_ARM_MMU_GetVirtualAddr()<br/>OS_ARM720_MMU_v2p()<br/>OS_ARM720_MMU_GetVirtualAddr()</pre> |  |
| 3.88a    | 0        | 130503 | AW | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 3.86n    | 1        | 130312 | AW | Chapter 4, "Thread safe system libraries with IAR compiler V6.4 or new-<br>er" corrected, one more required linker directive added.                                                                                                                                                                                                                                              |  |
| 3.86n    | 0        | 121210 | AW | New software version.<br>Chapter 4, "Thread safe system libraries with IAR compiler V6.4 or new-<br>er" added to describe the procedure to activate thread safe library sup-<br>port with newer IAR compiler.                                                                                                                                                                    |  |
| 3.861    | 0        | 121122 | AW | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 3.86g    | 0        | 120806 | AW | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 3.86f    | 0        | 120801 | AW | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 3.84c    | 0        | 120110 | TS | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |
| 3.84a    | 0        | 111214 | TS | New software version.                                                                                                                                                                                                                                                                                                                                                            |  |

| Software | Revision | Date   | Ву | Description                                                                                                                                                                                                 |  |
|----------|----------|--------|----|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|
|          |          |        |    | New function for VFP/Neon support in chapter 4: OS_ExtendTaskContext_NEON()                                                                                                                                 |  |
| 3.84     | 0        | 111105 | AW | New software version.                                                                                                                                                                                       |  |
| 3.82v    | 0        | 110715 | AW | <pre>New software version.<br/>New functions for thread locale storage and VFP support in chapter 4:<br/>OS_ExtendTaskContext_TLS()<br/>OS_ExtendTaskContext_TLS_VFP()<br/>OS_ExtendTaskContext_VFP()</pre> |  |
| 3.82t    | 0        | 110503 | AW | New software version.<br>Project settings and macros corrected for EWARM6.<br>New library mode OS_LIBMODE_DP with low optimization.                                                                         |  |
| 3.82m    | 0        | 101124 | AW | New software version.<br>Thread-local storage and thread safe library support for IAR compile<br>added, Chapter "Compiler specifics".                                                                       |  |
| 3.82     | 1        | 090918 | TS | New software version.                                                                                                                                                                                       |  |
| 3.80     | 0        | 090625 | SK | New software version.<br>Chapter "Using embOS ARM": Sample corrected.                                                                                                                                       |  |
| 3.62     | 2        | 090513 | SK | Chapter "ARM core version specifics": "Naming conventions for prebuild libraries compatible to IAR EW V5.x" corrected.                                                                                      |  |
| 3.62     | 1        | 081209 | SK | Chapter footer corrected.                                                                                                                                                                                   |  |
| 3.62     | 0        | 080904 | SK | New software version.<br>Chapter "C-SPY plug-in" added.                                                                                                                                                     |  |

# About this document

#### Assumptions

This document assumes that you already have a solid knowledge of the following:

- The software tools used for building your application (assembler, linker, C compiler).
- The C programming language.
- The target processor.
- DOS command line.

If you feel that your knowledge of C is not sufficient, we recommend *The C Programming Language* by Kernighan and Richie (ISBN 0--13--1103628), which describes the standard in C programming and, in newer editions, also covers the ANSI C standard.

#### How to use this manual

This manual explains all the functions and macros that the product offers. It assumes you have a working knowledge of the C language. Knowledge of assembly programming is not required.

#### Typographic conventions for syntax

This manual uses the following typographic conventions:

| Style                                                               | Used for                                                                                                                 |  |  |
|---------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------|--|--|
| Body                                                                | Body text.                                                                                                               |  |  |
| Keyword                                                             | Text that you enter at the command prompt or that appears on the display (that is system functions, file- or pathnames). |  |  |
| Parameter                                                           | Parameters in API functions.                                                                                             |  |  |
| Sample                                                              | Sample code in program examples.                                                                                         |  |  |
| Sample comment                                                      | Comments in program examples.                                                                                            |  |  |
| Reference                                                           | Reference to chapters, sections, tables and figures or other doc-<br>uments.                                             |  |  |
| <b>GUIElement</b> Buttons, dialog boxes, menu names, menu commands. |                                                                                                                          |  |  |
| Emphasis                                                            | Very important sections.                                                                                                 |  |  |

# **Table of contents**

| 1 | Using embOS10            |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        |                                  |
|---|--------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------|
|   | 1.1<br>1.2<br>1.3<br>1.4 | Installation                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           | 12<br>13                         |
| 2 | Build                    | l your own application1                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                | 8                                |
|   | 2.1<br>2.2<br>2.3<br>2.4 | Introduction                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           | 19<br>19                         |
| 3 | Libra                    | ries2                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  | 20                               |
|   | 3.1                      | Naming conventions for prebuilt libraries 2                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                            | 21                               |
| 4 | CPU                      | and compiler specifics2                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                | 23                               |
|   | 4.1<br>4.2<br>4.3<br>4.4 | IAR C-Spy stack check warning                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          | 24<br>24<br>26<br>26<br>27<br>28 |
| 5 | Stac                     | ks                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                     | 30                               |
|   | 5.1<br>5.2<br>5.3<br>5.4 | Task stack       3         System stack       3         Interrupt stack       3         Stack specifics       3                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        | 31<br>31                         |
| 6 | Inter                    | rupts3                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 | 33                               |
|   | 6.1<br>6.2<br>6.3<br>6.4 | What happens when an interrupt occurs?       Image: Constraint of the second seco | 34<br>34<br>36<br>36             |

|   |     | 6.4.1.2 OS_ARM_EnableISR()                                               |     |
|---|-----|--------------------------------------------------------------------------|-----|
|   |     | 6.4.1.3 OS_ARM_DisableISR()                                              |     |
|   |     | 6.4.1.4 OS_ARM_ISRSetPrio()                                              |     |
|   |     | 6.4.1.5 OS_ARM_ClearPendingFlag()                                        |     |
|   |     | 6.4.1.6 OS_ARM_IsPending()<br>6.4.1.7 OS ARM AssignISRSource()           |     |
|   |     | 6.4.1.7 OS_ARM_AssignISRSource()<br>6.4.1.8 OS_ARM_EnableISRSource()     |     |
|   |     | 6.4.1.9 OS_ARM_DisableISRSource()                                        |     |
|   |     | 6.4.1.10 OS_ARM_SetISRCfg()                                              |     |
|   |     | 6.4.1.11 OS_ARM_SetVBAR()                                                |     |
|   | 6.5 | Interrupt-stack switching                                                |     |
|   | 6.6 | Fast Interrupt (FIQ)                                                     | 49  |
| 7 | MML | I/MPU and cache support                                                  | 50  |
|   | 7.1 | Introduction                                                             | .51 |
|   | 7.2 | MMU and cache handling for ARM720 CPUs                                   |     |
|   |     | 7.2.1 API functions                                                      |     |
|   |     | 7.2.1.1 OS_ARM720_MMU_InitTT()                                           |     |
|   |     | 7.2.1.2 OS_ARM720_MMU_AddTTEntries()                                     |     |
|   |     | 7.2.1.3 OS_ARM720_MMU_Enable()<br>7.2.1.4 OS_ARM720_MMU_GetVirtualAddr() |     |
|   |     | 7.2.1.4 OS_ARM720_MM0_GetVirtualAddr()                                   |     |
|   |     | 7.2.1.6 OS_ARM720_CACHE_Enable()                                         |     |
|   |     | 7.2.1.7 OS_ARM720_CACHE_CleanRange()                                     |     |
|   |     | 7.2.1.8 OS_ARM720_CACHE_InvalidateRange()                                | .60 |
|   | 7.3 | MMU handling for ARMv5/ARMv7-A CPUs                                      |     |
|   |     | 7.3.1 API functions                                                      |     |
|   |     | 7.3.1.1 OS_ARM_MMU_InitTT()                                              |     |
|   |     | 7.3.1.2 OS_ARM_MMU_AddTTEntries()<br>7.3.1.3 OS ARM MMU Enable()         |     |
|   |     | 7.3.1.3 OS_ARM_MMU_Enable()<br>7.3.1.4 OS_ARM_MMU_GetVirtualAddr()       |     |
|   |     | 7.3.1.5 OS_ARM_MMU_v2p()                                                 |     |
|   | 7.4 | MPU handling for ARMv7-R CPUs                                            |     |
|   |     | 7.4.1 API functions                                                      | 68  |
|   |     | 7.4.1.1 OS_ARM_MPU_AddEntry()                                            |     |
|   |     | 7.4.1.2 OS_ARM_MPU_Enable()                                              |     |
|   |     | 7.4.1.3 OS_ARM_MPU_GetMinRegionSize()                                    | 72  |
|   |     | 7.4.1.4 OS_ARM_MPU_GetNumRegions()                                       |     |
|   | 7.5 | 7.4.1.5 OS_ARM_MPU_Init()<br>Cache handling for ARMv5/ARMv7 CPUs         |     |
|   | 7.5 | 7.5.1 API functions                                                      |     |
|   |     | 7.5.1.1 OS_ARM_ICACHE_Enable()                                           |     |
|   |     | 7.5.1.2 OS_ARM_ICACHE_Invalidate()                                       |     |
|   |     | 7.5.1.3 OS_ARM_DCACHE_Enable()                                           | 78  |
|   |     | 7.5.1.4 OS_ARM_DCACHE_Invalidate()                                       |     |
|   |     | 7.5.1.5 OS_ARM_DCACHE_Clean()                                            |     |
|   |     | 7.5.1.6 OS_ARM_DCACHE_CleanRange()                                       |     |
|   |     | 7.5.1.7 OS_ARM_DCACHE_InvalidateRange()<br>7.5.1.8 OS ARM CACHE Sync()   |     |
|   |     | 7.5.1.8 OS_ARM_CACHE_Sync()<br>7.5.1.9 OS_ARM_AddL2Cache()               |     |
|   |     | 7.5.1.10 OS_ARM_CACHE_GetLineSize()                                      |     |
|   | 7.6 | MMU and cache handling program sample                                    |     |
|   | 7.7 | MPU and cache handling program sample                                    |     |
| 8 | VFP | and NEON support                                                         | 88  |
|   | 8.1 | Introduction                                                             | .89 |
|   | 8.2 | Using embOS libraries with VFP/NEON support                              |     |
|   | 8.3 | Using the VFP/NEON unit in interrupt service routines                    |     |
|   |     | -                                                                        |     |

# Chapter 1 Using embOS

# 1.1 Installation

This chapter describes how to get started with embOS. You should follow these steps to become familiar with embOS.

embOS is shipped as a zip-file in electronic form. To install it, you should extract the zip-file to any folder of your choice while preserving its directory structure (i.e. keep all files in their respective sub directories). Ensure the files are not read-only after extraction. Assuming that you are using an IDE to develop your application, no further installation steps are required.

#### Note

The projects at /Start/BoardSupport/<DeviceManufacturer>/<Board> assume a relative location for the /Start/Lib and /Start/Inc folders. If you copy a BSP folder to another location, you will need to adjust the include paths of the project accordingly.

At /Start/BoardSupport/<DeviceManufacturer>/<Board> you should find several example start projects, which you may adapt to write your application. To do so, follow the instructions of section *First Steps* on page 12.

In order to become familiar with embOS, consider using the example projects (even if you will not use the IDE for application development).

If you do not or do not want to work with an IDE, you may copy either all library files or only the library that is used with your project into your work directory. embOS does in not rely on an IDE, but may be used without an IDE just as well, e.g. using batch files or a make utility.

# 1.2 First Steps

After installation of embOS, you can create your first multitasking application. You received several ready-to-go sample workspaces and projects as well as all required embOS files inside the subfolder Start. The subfolder Start/BoardSupport contains the workspaces and projects, sorted into manufacturer- and board-specific subfolders. It is a good idea to use one of the projects as a starting point for any application development.

To get your new application running, you should:

- Create a directory for your development.
- Copy the whole Start folder from your embOS shipment into the directory.
- Clear the read-only attribute of all files in the copied Start folder.
- Open one sample workspace/project in Start/BoardSupport/<DeviceManufacturer>/<Board> with your IDE (for example, by double clicking it).
- Build the project. It should be built without any error or warning messages.

After building the project of your choice, the screen should look like this:



For additional information, you should open the ReadMe.txt file that is part of every BSP. It describes the different configurations of the project and, if required, gives additional information about specific hardware settings of the supported evaluation board(s).

## 1.3 The example application OS\_StartLEDBlink.c

The following is a printout of the example application <code>OS\_StartLEDBlink.c</code>. It is a good starting point for your application (the actual file shipped with your port of embOS may differ slightly).

What happens is easy to see:

After initialization of embOS, two tasks are created and started. The two tasks get activated and execute until they run into a delay, thereby suspending themselves for the specified time, and eventually continue execution.

```
*
              SEGGER Microcontroller GmbH
                                                *
*
                                                *
               The Embedded Experts
----- END-OF-HEADER ------
File : OS_StartLEDBlink.c
Purpose : embOS sample program running two simple tasks, each toggling
      an LED of the target hardware (as configured in BSP.c).
* /
#include "RTOS.h"
#include "BSP.h"
static OS_STACKPTR int StackHP[128], StackLP[128]; // Task stacks
                                   // Task control blocks
static OS_TASK TCBHP, TCBLP;
static void HPTask(void) {
 while (1) {
  BSP_ToggleLED(0);
  OS_TASK_Delay(50);
 }
}
static void LPTask(void) {
 while (1) {
  BSP_ToggleLED(1);
  OS_TASK_Delay(200);
 }
}
*
*
     main()
*/
int main(void) {
 OS_Init(); // Initialize embOS
 OS_InitHW(); // Initialize required hardware
BSP_Init(); // Initialize LED ports
 OS_TASK_CREATE(&TCBHP, "HP Task", 100, HPTask, StackHP);
 OS_TASK_CREATE(&TCBLP, "LP Task", 50, LPTask, StackLP);
 OS_Start(); // Start embOS
 return 0;
}
```

## 1.4 Stepping through the sample application

When starting the debugger, you will see the main() function (see example screenshot below). The main() function appears as long as project option Run to main is selected, which it is enabled by default. Now you can step through the program.

OS\_Init() is part of the embOS library and written in assembler; you can therefore only step into it in disassembly mode. It initializes the relevant OS variables.

OS\_InitHW() is part of RTOSInit.c and therefore part of your application. Its primary purpose is to initialize the hardware required to generate the system tick interrupt for embOS. Step through it to see what is done.

 $\texttt{OS\_Start()}$  should be the last line in main(), because it starts multitasking and does not return.



Before you step into  ${\tt OS\_Start()}$  , you should set two breakpoints in the two tasks as shown below.

| 🔀 IAR Embedded Workbench IDE                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                    |
|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| <u>File Edit View Project Debug</u> Disassembly <u>2</u> -Link embOS <u>I</u> ools <u>W</u> indow <u>H</u> elp                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  |
| _ D 🗳 🖬 🕼   & ha 🋍   ∽ ∽   📃 🖉 🖉 🖉 🖉 🖉 🖉 🖉 🖉 🖉                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  |
| 5 B B B B B B B B B B B B B B B B B B B                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |
| Workspace RTOSInit_LPC1000.c Start_LEDBlink.c **                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |
| Debug       37         Files       38 OS_STACKPTR int StackHP[128], StackLP[128];       /* Task stc         Start_LPC10 ✓       49         H Application       41         H Start_L.       49         H Start_L.       41         H Start_L.       41         H Start_L.       41         H Start_L.       42         H Start_L.       45         J Start_L.       6         H Start_L.       45         J Start_C.       58 DE lay (11);         J Start       58 DE lay (11);         J Start       51         J Start       52         J Start       53         J Start       54 |
| 55<br>56 /************************************                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  |
| 59 *                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                            |
|                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| × Expression Value Location Type                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |
| Image: Time         0x0000000         0x10000ACC         int           Debug Log         Build         Watch         ×                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |
| Ready                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           |

As  $\texttt{OS\_Start()}$  is part of the embOS library, you can step through it in disassembly mode only.

Click GO, step over <code>OS\_Start()</code>, or step into <code>OS\_Start()</code> in disassembly mode until you reach the highest priority task.

| 🗶 IAR Embedded Workbench IDE 📃 🛛 🗙                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                             |
|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| <u>File Edit View Project Debug Disassembly J-Link embOS Tools Window Help</u>                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| _ D 😅 🖬 🕼   & B 🛍   ∽ ∽   ▼ ♦ 🐄 🔽 🛛 🗠 📣 🏰 👫 💥 🖗 🕭                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              |
| 5 B B B B B B B B B B B B B B B B B B B                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        |
| Workspace x RTOSInit_LPC1000.c Start_LEDBlink.c **                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                             |
| Debug       37         38 OS_STACKPTR int StackHP[128], StackLP[128];       /* Task stal         Image: Start_LPC10        /*         Image: Start_L        /* |
| 56         >***********************************                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |
|                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |
| Expression         Value         Location         Type           -         Time         0x00000000         0x10000ACC         int                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              |
| Debug Log  Build  Watch                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        |
| Ready //                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |

If you continue stepping, you will arrive at the task that has lower priority:

| 🔀 IAR Embedded Workbench IDE 📃 🗖 🗙                                                                                                                                                                                                                                                                                                                                                                        |
|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| File Edit View Project Debug Disassembly 2-Link embOS Tools Window Help                                                                                                                                                                                                                                                                                                                                   |
| D 🖆 🖬 🕼   🎄   X 🛍 🛍   🕫 🕬                                                                                                                                                                                                                                                                                                                                                                                 |
| 」;;: ::::::::::::::::::::::::::::::::::                                                                                                                                                                                                                                                                                                                                                                   |
| Workspace RTOSInit_LPC1000.c Start_LEDBlink.c                                                                                                                                                                                                                                                                                                                                                             |
| Debug       37         Files       38 OS_STACKPTR int StackHP[128], StackLP[128]; /* Task stack         Image: Start_LPC10 v       40         Image: Application       40         Image: Start_L       41         Image: Start_L       42         Image: Start_L       43         Image: Start_L       45         Image: Start_L       95         Image: Start_L       45         Image: Start_L       48 |
|                                                                                                                                                                                                                                                                                                                                                                                                           |
| Start_LPC1000                                                                                                                                                                                                                                                                                                                                                                                             |
| *         Expression         Value         Location         Type                                                                                                                                                                                                                                                                                                                                          |
| Debug Log   Build Watch     X                                                                                                                                                                                                                                                                                                                                                                             |
| Ready                                                                                                                                                                                                                                                                                                                                                                                                     |

Continue to step through the program, there is no other task ready for execution. embOS will therefore start the idle-loop, which is an endless loop always executed if there is nothing else to do (no task is ready, no interrupt routine or timer executing).

You will arrive there when you step into the <code>OS\_TASK\_Delay()</code> function in disassembly mode. <code>OS\_Idle()</code> is part of <code>RTOSInit.c</code>. You may also set a breakpoint there before stepping over the delay in <code>LPTask()</code>.



If you set a breakpoint in one or both of our tasks, you will see that they continue execution after the given delay.

As can be seen by the value of embOS timer variable OS\_Global.Time, shown in the Watch window, HPTask() continues operation after expiration of the delay.



# Chapter 2 Build your own application

## 2.1 Introduction

This chapter provides all information to set up your own embOS project. To build your own application, you should always start with one of the supplied sample workspaces and projects. Therefore, select an embOS workspace as described in chapter *First Steps* on page 12 and modify the project to fit your needs. Using an embOS start project as starting point has the advantage that all necessary files are included and all settings for the project are already done.

# 2.2 Required files for an embOS

To build an application using embOS, the following files from your embOS distribution are required and have to be included in your project:

- **RTOS.h** from the directory .\Start\Inc. This header file declares all embOS API functions and data types and has to be included in any source file using embOS functions.
- **RTOSINIT\*.c** from one target specific .\Start\BoardSupport\<Manufacturer>\<MCU> subfolder. It contains hardware-dependent initialization code for embOS. It initializes the system timer interrupt but can also initialize or set up the interrupt controller, clocks and PLLs, the memory protection unit and its translation table, caches and so on.
- OS\_Error.c from one target specific subfolder .\Start\BoardSupport \<Manufacturer>\<MCU>. The error handler is used only if a debug library is used in your project.
- One **embOS library** from the subfolder .\Start\Lib.
- Additional CPU and compiler specific files may be required according to CPU.

When you decide to write your own startup code or use a low level <code>init()</code> function, ensure that non-initialized variables are initialized with zero, according to C standard. This is required for some embOS internal variables. Your <code>main()</code> function has to initialize embOS by calling <code>OS\_Init()</code> and <code>OS\_InitHW()</code> prior to any other embOS functions that are called.

# 2.3 Change library mode

For your application you might want to choose another library. For debugging and program development you should always use an embOS debug library. For your final application you may wish to use an embOS release library or a stack check library.

Therefore you have to select or replace the embOS library in your project or target:

- If your selected library is already available in your project, just select the appropriate project configuration.
- To add a library, you may add the library to the existing Lib group. Exclude all other libraries from your build, delete unused libraries or remove them from the configuration.
- Check and set the appropriate <code>OS\_LIBMODE\_\*</code> define as preprocessor option and/or modify the <code>OS\_Config.h</code> file accordingly.

# 2.4 Select another CPU

embOS contains CPU-specific code for various CPUs. Manufacturer- and CPU-specific sample start workspaces and projects are located in the subfolders of the .\Start\BoardSupport directory. To select a CPU which is already supported, just select the appropriate workspace from a CPU-specific folder.

If your CPU is currently not supported, examine all <code>RTOSInit.c</code> files in the CPU-specific subfolders and select one which almost fits your CPU. You may have to modify <code>OS\_InitH-W()</code>, the interrupt service routines for the embOS system tick timer and the low level initialization.

# **Chapter 3**

# Libraries

# 3.1 Naming conventions for prebuilt libraries

embOS is shipped with different pre-built libraries with different combinations of features. Note that not all combinations are available (e.g., there is no VFP support for Armv4).

The libraries are named as follows:

os<Architecture>\_<CpuMode><Endianness><VFP\_NEON><LibMode>.a

| Parameter    | Meaning                          | Values                                                                                                                                                                                                                                                                                                                                        |
|--------------|----------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Architecture | ARM architecture                 | 4t : ARMv4<br>5t : ARMv5, ARMv6<br>7a : ARMv7-A<br>7r : ARMv7-R                                                                                                                                                                                                                                                                               |
| CpuMode      | CPU mode                         | a : ARM<br>t : Thumb / Thumb2                                                                                                                                                                                                                                                                                                                 |
| Endianness   | Byte order                       | <ul><li>b : Big endian</li><li>I : Little endian</li></ul>                                                                                                                                                                                                                                                                                    |
| VFP_NEON     | Floating point /<br>NEON support | <ul> <li>No hardware VFP support</li> <li>v16: VFP-D16 floating-point ABI</li> <li>v32: VFP-D32 floating-point ABI</li> </ul>                                                                                                                                                                                                                 |
| LibMode      | embOS library mode               | <ul> <li>xr : Extreme Release</li> <li>r : Release</li> <li>s : Stack check</li> <li>sp : Stack check + Profiling</li> <li>d : Debug</li> <li>dp : Debug + Profiling + Stack check</li> <li>dt : Debug + Profiling + Stack check + Trace</li> <li>dpl : Debug + Profiling + Stack check built</li> <li>with low optimization level</li> </ul> |

#### Example

os7a\_tlv32dp.a is the IAR EWARM V8/V9 library for an ARMv7-A core, thumb2 mode, little endian mode, support for VFP/NEON D32, with debug and profiling support.

The Debug and Profiling library built with low optimization level may be used during development to show a more detailed call stack when using the CSpy plugin.

#### Note

With embOS V5.06 for ARM and IAR, the names of the ARMv7-A libraries were changed. In previous versions, these libraries were erroneously named with an  $os7t_$  prefix in the shipment, even though the manual correctly denoted them to have an  $os7a_$  prefix. Starting with embOS V5.06 for ARM and IAR, the shipped libraries use the  $os7a_$  prefix as described in the manual.

#### Note

When updating from an earlier embOS version you might need to update to an embOS library with VFP/NEON support. For example, if you use  $libos_v7a_t_le_i_dp.a$  for an ARMv7-A CPU with VFP/NEON unit you will get a linker error message like 'no definition for "os\_Init\_VFPD32".

This check avoids that the project and the used embOS library use different VFP/NEON settings. If your project settings allow the compiler to generate VFP/NEON instructions, an embOS library with VFP/NEON support like  $os7a_tlv32dp.a$  must be used.

# Chapter 4 CPU and compiler specifics

# 4.1 IAR C-Spy stack check warning

IAR's C-Spy debugger provides a stack check feature which throws a warning when the stack pointer does not point to memory within the CSTACK scope anymore. This renders the C-Spy stack check useless, as C-Spy is not aware of any task stacks the application is using. Depending on the IAR version used, this warning can be disabled by removing the check mark for Tools -> Options... -> Stack -> 'Warn when stack pointer is out of bounds' Or Project -> Options... -> Debugger -> Plugins -> Stack.

# 4.2 IAR C-Spy RTOS plugin

SEGGER's embOS plug-in for the IAR Embedded Workbench provides embOS awareness during debugging sessions. This enables you to inspect the state of several embOS primitives such as the task list, semaphores, mailboxes, and software timers.

SEGGER's embOS plug-in is already shipped with IAR EWARM but the most recent version can be downloaded from <u>segger.com/products/rtos/embos/tools/plug-ins/iar-embed-ded-workbench</u>.

# 4.3 Interrupt and thread safety

Using embOS with specific calls to standard library functions (e.g. heap management functions) may require thread-safe system libraries if these functions are called from several tasks or interrupts. IAR's system libraries provide functions, which can be overwritten to implement a locking mechanism making the system library functions thread-safe.

The Setup directory in each embOS BSP contains the file <code>OS\_ThreadSafe.c</code> which overwrites these functions. By default they disable and restore embOS interrupts to ensure thread safety in tasks, embOS interrupts, <code>OS\_Idle()</code> and software timers. Zero latency interrupts are not disabled and therefore unprotected. If you need to call e.g. malloc() also from within a zero latency interrupt additional handling needs to be added. If you don't call such functions from within embOS interrupts, <code>OS\_Idle()</code> or software timers, you can instead use thread safety for tasks only. This reduces the interrupt latency because a mutex is used instead of disabling embOS interrupts.

You can choose the safety variant with the macro <code>os\_interrupt\_safe</code>.

- When defined to 1 thread safety is guaranteed in tasks, embOS interrupts, OS\_Idle() and software timers.
- When defined to 0 thread safety is guaranteed only in tasks. In this case you must not call e.g. heap functions from within an ISR, <code>os\_Idle()</code> or embOS software timers.

### 4.3.1 Enabling thread-safe IAR system libraries

By default, IAR does not use thread-safe system libraries. As a result the implemented hook functions are not linked into the application. For more information on IAR's multithread support, please refer to the IAR Embedded Workbench manuals.

To use the thread-safe system libraries the option "Enable thread support in library" must be set in Project -> Options... -> General Options -> Library Configuration. Alternatively, the option --threaded\_lib can be passed to the linker. Additionally the function OS\_INIT\_SYS\_LOCKS() must be called.

With older IAR Embedded Workbench versions, neither the IDE option nor the linker option are available. In this case, the linker has to be told to explicitly link the hook functions by redirecting them to another symbol.

Activate the checkbox "Use command line options" in the dialog Project -> Options... -> Linker -> Extra Options. Then, in the "Command line options:" field, add the following lines:

<sup>--</sup>redirect \_\_iar\_Locksyslock=\_\_iar\_Locksyslock\_mtx

```
--redirect __iar_Unlocksyslock=__iar_Unlocksyslock_mtx
--redirect __iar_Lockfilelock=__iar_Lockfilelock_mtx
--redirect __iar_Unlockfilelock=__iar_Unlockfilelock_mtx
--keep __iar_Locksyslock_mtx
```

#### C++ thread safety

To enable thread-safe C++ constructors and destructors the option  $--guard_calls$  needs to be passed to the compiler.

# 4.4 Thread-Local Storage TLS

The DLib for IAR supports usage of thread-local storage. Several library objects and functions need local variables which have to be unique to a thread. Thread-local storage will be required when these functions are called from multiple threads.

embOS for IAR is prepared to support the tread-local storage, but does not use it per default. This has the advantage of no additional overhead as long as thread-local storage is not needed by the application. The embOS implementation of thread-local storage allows activation of TLS separately for each task.

Only tasks that are accessing TLS variables, for instance by calling functions from the system library, need to initialize their TLS by calling an initialization function when the task is started. For each task that uses TLS the memory for the thread-local storage is allocated by the IAR runtime environment on the heap. Therefore, thread-safe heap management should be used together with TLS. For information on thread safety, please refer to *Interrupt and thread safety* on page 24.

When the task terminates by a call of  $OS_{TASK}$  terminate(), the memory used for TLS is automatically freed and put back into the free heap memory.

Library objects that need thread-local storage when used in multiple tasks are for example:

- error functions errno, strerror.
- locale functions localeconv, setlocale.
- time functions asctime, localtime, gmtime, mktime.
- multibyte functions mbrlen, mbrtowc, mbsrtowc, mbtowc, wcrtomb, wcsrtomb, wctomb.
- rand functions rand, srand.
- etc functions atexit, strtok.
- C++ exception engine.

### 4.4.1 API functions

| Routine                          | Description                                                                                        | main | Priv Task | Unpriv Task | ISR | SW Timer |
|----------------------------------|----------------------------------------------------------------------------------------------------|------|-----------|-------------|-----|----------|
| OS_TLS_Set()                     | Initializes the thread-local storage for the current task.                                         |      | •         |             |     |          |
| OS_TLS_SetTaskContextExtension() | Initializes the thread-local storage and sets the TLS task context extension for the current task. |      | •         |             |     |          |

### 4.4.1.1 OS\_TLS\_Set()

#### Description

Initializes the thread-local storage for the current task.

#### Prototype

void OS\_TLS\_Set(void);

#### **Additional information**

 ${\tt OS\_TLS\_Set()}$  shall be the first function called from a task when TLS should be used in this task.

The required memory for the thread-local storage is allocated from the heap. If  $OS\_TLS\_Set()$  was already called for this task or if there is not enough memory on the heap, then embOS will call  $OS\_Error()$  with the error code  $OS\_ERR\_TLS\_INIT$ .

This function has to be used only in combination with OS\_TASK\_AddContextExtension() or OS\_TASK\_SetContextExtension() and OS\_TLS\_ContextExtension as argument to these functions. When OS\_TLS\_SetTaskContextExtension() is used, OS\_TLS\_Set() will be called automatically.

#### Example

```
static void Task(void) {
  OS_TLS_Set();
  OS_TASK_SetContextExtension(&OS_TLS_ContextExtension);
  while (1) {
  }
}
```

#### 4.4.1.2 OS\_TLS\_SetTaskContextExtension()

#### Description

Initializes the thread-local storage and sets the TLS task context extension for the current task.

#### Prototype

void OS\_TLS\_SetTaskContextExtension(void);

#### Additional information

 ${\tt OS\_TLS\_SetTaskContextExtension()}$  shall be the first function called from a task when TLS should be used in this task.

The required memory for the thread-local storage is allocated from the heap. If  $OS\_TLS\_Set()$  was already called for this task or if there is not enough memory on the heap, then embOS will call  $OS\_Error()$  with the error code  $OS\_ERR\_TLS\_INIT$ .

If the task already contains a task context extension, <code>OS\_TLS\_SetTaskContextExtension()</code> cannot be used. Instead, <code>OS\_TASK\_AddContextExtension()</code> needs to be called with <code>OS\_TLS\_ContextExtension</code> as argument. Furthermore, <code>OS\_TLS\_Set()</code> needs to be called to initialize TLS for this task.

#### Example

The following printout demonstrates the usage of task specific TLS in an application.

```
#include "RTOS.h"
static OS_STACKPTR int StackHP[128], StackLP[128]; // Task stacks
                                                          // Task control blocks
static OS_TASK
                         TCBHP, TCBLP;
static void HPTask(void) {
  OS_TLS_SetTaskContextExtension();
  while (1) {
    errno = 42; // errno specific to HPTask
    OS_TASK_Delay(50);
  }
}
static void LPTask(void) {
  OS_TLS_SetTaskContextExtension();
  while (1) {
    errno = 1; // errno specific to LPTask
    OS_TASK_Delay(200);
  }
}
int main(void) {
 errno = 0; // errno not specific to any task
OS_Init(); // Initialize embOS
OS_InitHW(); // Initialize required hardware
  OS_TASK_CREATE(&TCBHP, "HP Task", 100, HPTask, StackHP);
  OS_TASK_CREATE(&TCBLP, "LP Task", 50, LPTask, StackLP);
  OS_Start(); // Start embOS
  return 0;
}
```

# 4.5 Debug output

The IAR Embedded Workbench and compiler allow debug output via e.g. printf() which are redirected to the IAR terminal in the debugger.

There are two different modes for debug outputs available under the "Library low-level interface implementation" settings in the "Library Configuration" tab under "General Options" of the project options dialog. Using "Semihosted" triggers an SWI interrupt during execution of printf(). The default SWI handler in the IAR startup code is implemented as an endless loop. The debugger sets a breakpoint at the SWI handler entry address and monitors the code execution. When the breakpoint is reached, the debugger starts the data exchange with the target application.

When the same code runs stand alone without connection to the debugger, the application runs into the default SWI handler and gets stuck in the endless loop. To avoid this situation, every embOS start project comes with an SWI handler that is delivered in source code in the file SWI\_Handler.c in the Setup folder. The SWI\_Handler() from this file function replaces the weak SWI\_Handler() from the startup code in the system library. The SWI\_Handler() function has no functionality, it just returns. If the SWI handler shall fulfill another purpose than just enabling printf() output, a custom SWI handler has to be used.

# **Chapter 5**

# Stacks

## 5.1 Task stack

Each task uses its individual stack. The stack pointer is initialized and set every time a task is activated by the scheduler. The task stack needs to be able to accommodate the stack content of any (sub-)function plus the basic stack size.

The basic stack size is the size of memory required to store the context of the task on the stack. The minimum basic task stack size is 72 bytes for CPUs without a VFP/NEON unit and up to 332 bytes for CPUs with a VFP/NEON unit. We recommend at least 256 bytes stack as a start for CPUs without a VFP/NEON unit and 512 bytes for CPUs with a VFP/NEON unit.

#### Note

Stacks for ARM devices need to be 8-byte aligned. embOS ensures that task stacks are properly aligned. If an unaligned stack was aligned, the first few bytes up to the aligned address will not be used. Thus, the application should ensure that task stacks are properly aligned. This can be achieved by defining an array using a 64-bit data type like OS\_U64.

## 5.2 System stack

The embOS scheduler executes in supervisor (SVC) mode. However, embOS doesn't use the dedicated SVC stack symbol in order to initialize the SVC stack pointer. After <code>os\_Start()</code> was called embOS uses the stack symbol of the system stack which was used in the <code>main()</code> routine. This avoids the need to allocate dedicated SVC stack space.

The minimum system stack size required by embOS is about 160 bytes (stack check & profiling build, no VFP/NEON unit). Since the system stack is also used by the application before the start of multitasking (the call to  $os_{start()}$ ), and because software timers and C-level interrupt handlers also use the system stack, the actual stack requirements depend on the application.

The size of the system stack can be changed by modifying the stack size definition in your linker file or within the project settings. We recommend a minimum stack size of 512 bytes for the system stack for CPUs without a VFP/NEON unit and 1024 bytes for CPUs with a VFP/NEON unit.

### 5.3 Interrupt stack

If a normal hardware exception occurs, the ARM core switches to IRQ mode, which has a separate stack pointer. After saving the scratch registers as well as LR\_irq and SPSR\_irq (and FPSCR, if VFP/NEON unit is present) onto the IRQ stack embOS switches to SVC mode. Only the previously mentioned registers are saved onto the IRQ stack. Thus, every interrupt requires 32 bytes on the IRQ stack. The maximum IRQ stack size required by the application can be calculated as "Maximum interrupt nesting level \* 32 bytes". For the interrupt routine itself, the system stack is used, because they're executed in SVC mode.

The size of the IRQ stack can be changed by modifying the stack size definition in your linker file or within the project settings.

## 5.4 Stack specifics

There are two stacks which have to be declared in the linker script file or project settings:

- The system stack.
- The IRQ stack.

The system stack is used by the startup, the main() routine, embOS internal functions, and C-level interrupt handlers.

The IRQ stack is used when an interrupt exception is triggered. The exception handler saves some registers and then performs a mode switch which then uses the system stack for further execution.

When the CPU starts, it runs in supervisor mode. Then the startup code initializes the various stack pointer registers for each mode with their assigned stack and finally jumps into the main() routine. embOS expects the main() routine to use the system stack, no matter in which CPU mode.

When  $OS\_Init()$  is called, embOS initializes the supervisor stack pointer to point to the system stack. After embOS is started with  $OS\_Start()$ , the embOS scheduler runs in supervisor mode using the system stack while each task is running in system mode using its own dedicated stack.

#### Note

Stacks for ARM devices need to be 8-byte aligned.

# **Chapter 6**

# Interrupts

## 6.1 What happens when an interrupt occurs?

- The CPU-core receives an interrupt request.
- As soon as the interrupts are enabled, the interrupt is executed.
- The CPU switches to the IRQ mode which uses the IRQ stack.
- The CPU saves PC and flags in registers LR\_irq and SPSR\_irq.
- The CPU jumps to offset 0x18 in the exception vector table which contains an instruction to branch to the embOS low-level IRQ\_Handler().
- embOS IRQ\_Handler(): Saves scratch registers as well as LR\_irq, SPSR\_irq and FPSCR on the IRQ stack.
- embOS IRQ\_Handler(): Switches to supervisor mode and system stack.
- embOS IRQ\_Handler(): Saves scratch VFP registers on the system stack.
- embOS IRQ\_Handler(): Executes OS\_irq\_handler() (defined in RTOSInit\_\*.c).
- embOS os\_irg\_handler(): Checks for interrupt source and executes the according ISR handler. The implementation of this functions depends on the implemented interrupt controller.
- embOS IRQ\_Handler(): Restores scratch VFP registers from the system stack.
- embOS IRQ\_Handler(): Switches to IRQ mode and IRQ stack.
- embOS IRQ\_Handler(): Restores scratch registers as well as LR\_irq, SPSR\_irq and FPSCR from the IRQ stack.
- embOS IRQ\_Handler(): Returns from interrupt.

#### Note

FPSCR and VFP registers are only preserved by embOS libraries with VFP support.

## 6.2 Defining interrupt handlers in C

Interrupt handlers called from the default C interrupt handler <code>OS\_irq\_handler()</code> located in <code>RTOSInit\*.c</code> are just normal functions which do not take parameters and do not return any value. <code>OS\_irq\_handler()</code> first calls <code>OS\_INT\_Enter()</code> or <code>OS\_INT\_EnterNestable()</code> to inform embOS that interrupt code is running. Then this handler examines the source of interrupt and calls the related interrupt handler function. Finally, <code>OS\_irq\_handler()</code> calls <code>OS\_INT\_Leave()</code> or <code>OS\_INT\_LeaveNestable()</code> and returns to the primary low level interrupt handler <code>IRQ\_Handler()</code>.

Depending on the interrupting source, it may be required to reset the interrupt pending condition of the related peripherals.

#### Example

Simple interrupt routine:

```
void Timer_IRQHandler(void) {
  static unsigned long Time = 0;
  //
  // Handle timer IRQ
  //
  Time++;
}
```

# 6.3 Interrupt handling without vectored interrupt controller

When using an ARM CPU without implementation of a vectored interrupt controller, the application is responsible to examine which interrupting source triggered the IRQ.

The reaction to an interrupt is as follows:

- IRQ\_Handler() calls OS\_irq\_handler().
- OS\_irq\_handler() informs embOS that interrupt code is running by calling OS\_INT\_Enter().
- OS\_irq\_handler() determines the interrupt sources and handles all pending IRQs.
- OS\_irq\_handler() informs embOS that interrupt handling ended by calling OS\_INT\_Leave().
- IRQ\_Handler() returns to IRQ\_Handler().

#### Example

Simple interrupt routine:

```
void OS_irq_handler(void) {
   OS_INT_Enter();
   if (Timer_IsPending()) { // Interrupt pending?
      Timer_IRQHandler(); // Handle interrupt
   }
   if (UART_IsPending()) { // Interrupt pending ?
      UUART_IRQHandler(); // Handle interrupt
   }
   OS_INT_Leave();
}
```

During interrupt processing, you should not re-enable interrupts, as this would lead in recursion.

# 6.4 Interrupt handling with vectored interrupt controller

For ARM derivatives with built in vectored interrupt controller, embOS uses a different interrupt handling procedure and delivers additional functions to install and setup interrupt handler functions. You should not program the interrupt controller for IRQ handling directly. You should use the functions delivered with embOS.

The reaction to an interrupt with vectored interrupt controller is as follows:

- IRQ\_Handler() **calls** OS\_irq\_handler().
- OS\_irq\_handler() examines the interrupting source by reading the interrupt vector from the interrupt controller.
- OS\_irq\_handler() informs embOS that interrupt code is running by calling OS\_INT\_Enter().
- OS\_irq\_handler() calls the interrupt handler function which is addressed by the interrupt vector.
- OS\_irq\_handler() resets the interrupt controller to re-enable acceptance of new interrupts.
- OS\_irq\_handler() informs embOS that interrupt handling ended by calling OS\_INT\_Leave().
- OS\_irq\_handler() returns to IRQ\_Handler().

#### Note

Different ARM CPUs may have different versions of vectored interrupt controller hardware, and usage of embOS supplied functions varies depending on the type of interrupt controller. Refer to the samples delivered with embOS which are used in the CPU specific RTOSInit module.

### 6.4.1 API functions

To handle interrupts with vectored interrupt controller, embOS offers the following functions:

| Function                   | Description                                                           |
|----------------------------|-----------------------------------------------------------------------|
| OS_ARM_InstallISRHandler() | Installs an interrupt handler                                         |
| OS_ARM_EnableISR()         | Enables an interrupt                                                  |
| OS_ARM_DisableISR()        | Disables an interrupt                                                 |
| OS_ARM_ISRSetPrio()        | Sets the priority of an interrupt                                     |
| OS_ARM_ClearPendingFlag()  | Clears an interrupt pending flag                                      |
| OS_ARM_IsPending()         | Checks if an interrupt is pending                                     |
| OS_ARM_AssignISRSource()   | Assigns a hardware interrupt channel to an inter-<br>rupt vector      |
| OS_ARM_EnableISRSource()   | Enables an interrupt channel of a VIC type interrupt controller       |
| OS_ARM_DisableISRSource()  | Disables an interrupt channel of a VIC type inter-<br>rupt controller |
| OS_ARM_SetISRCfg()         | Sets the interrupt configuration.                                     |
| OS_ARM_SetVBAR()           | Writes the vector table address register.                             |

# 6.4.1.1 OS\_ARM\_InstallISRHandler()

#### Description

OS\_ARM\_InstallISRHandler() is used to install a specific interrupt vector when ARM CPUs with vectored interrupt controller are used.

#### Prototype

```
OS_ISR_HANDLER* OS_ARM_InstallISRHandler(int ISRIndex,
OS_ISR_HANDLER* pISRHandler);
```

#### Parameters

| Parameter   | Description                                                         |  |
|-------------|---------------------------------------------------------------------|--|
| ISRIndex    | Index of the interrupt source, usually the interrupt vector number. |  |
| pISRHandler | Address of the interrupt handler function.                          |  |

#### **Return Value**

 ${\tt OS\_ISR\_HANDLER*}$  : The address of the interrupt handler that was previously installed with the addressed interrupt source.

#### **Additional Information**

This function just installs the interrupt vector but does not modify the priority and does not automatically enable the interrupt.

# 6.4.1.2 OS\_ARM\_EnableISR()

#### Description

OS\_ARM\_EnableISR() is used to enable interrupt acceptance of a specific interrupt source in a vectored interrupt controller.

#### Prototype

void OS\_ARM\_EnableISR(int ISRIndex);

#### Parameters

| Parameter | Description                                            |  |
|-----------|--------------------------------------------------------|--|
| ISRIndex  | Index of the interrupt source which should be enabled. |  |

#### **Additional Information**

This function just enables the interrupt inside the interrupt controller. It does not enable the interrupt of any peripherals. This has to be done elsewhere.

#### Note

For ARM CPUs with VIC type interrupt controller, this function just enables the interrupt vector itself. To enable the hardware assigned to that vector, you have to call OS\_ARM\_EnableISRSource() also.

# 6.4.1.3 OS\_ARM\_DisableISR()

#### Description

OS\_ARM\_DisableISR() is used to disable interrupt acceptance of a specific interrupt source in a vectored interrupt controller which is not of the VIC type.

#### Prototype

void OS\_ARM\_DisableISR(int ISRIndex);

#### Parameters

| Parameter | Description                                             |  |
|-----------|---------------------------------------------------------|--|
| ISRIndex  | Index of the interrupt source which should be disabled. |  |

#### **Additional Information**

This function just disables the interrupt controller. It does not disable the interrupt of any peripherals. This has to be done elsewhere.

#### Note

When using an ARM CPU with built in interrupt controller of VIC type, use <code>OS\_AR-M\_DisableISRSource()</code> to disable a specific interrupt.

# 6.4.1.4 OS\_ARM\_ISRSetPrio()

#### Description

OS\_ARM\_ISRSetPrio() is used to set or modify the priority of a specific interrupt source by programming the interrupt controller.

#### Prototype

#### Parameters

| Parameter | Description                                                  |  |
|-----------|--------------------------------------------------------------|--|
| ISRIndex  | Index of the interrupt source which should be modified.      |  |
| Prio      | The priority which should be set for the specific interrupt. |  |

#### **Return Value**

Previous priority which was assigned before the call of OS\_ARM\_ISRSetPrio().

#### **Additional Information**

This function sets the priority of an interrupt channel by programming the interrupt controller. Refer to CPU-specific manuals about allowed priority levels.

# 6.4.1.5 OS\_ARM\_ClearPendingFlag()

#### Description

 ${\tt OS\_ARM\_ClearPendingFlag()}$  is used to clear an interrupt pending flag

#### Prototype

void OS\_ARM\_ClearPendingFlag(int ISRIndex);

#### Parameters

| Parameter | Description                                           |  |
|-----------|-------------------------------------------------------|--|
| ISRIndex  | Index of the interrupt source which should be cleared |  |

#### **Additional Information**

This function just clears the interrupt pending flag inside the interrupt controller. It does not clear the interrupt pending flag in any peripheral.

# 6.4.1.6 OS\_ARM\_IsPending()

#### Description

 ${\tt OS\_ARM\_IsPending()}$  is used to check if an interrupt is pending

#### Prototype

unsigned int OS\_ARM\_IsPending(int ISRIndex);

#### Parameters

| Parameter | Description                                           |
|-----------|-------------------------------------------------------|
| ISRIndex  | Index of the interrupt source which should be checked |

#### Return value

- = 0 Interrupt is not pending.
- = 1 Interrupt is pending.

#### **Additional Information**

This function just checks the interrupt pending flag inside the interrupt controller. It does not check the interrupt pending flag in any peripheral.

# 6.4.1.7 OS\_ARM\_AssignISRSource()

#### Description

OS\_ARM\_AssignISRSource() is used to assign a hardware interrupt channel to an interrupt vector in an interrupt controller of VIC type.

#### Prototype

#### Parameters

| Parameter | Description                                                                           |  |
|-----------|---------------------------------------------------------------------------------------|--|
| ISRIndex  | Index of the interrupt source which should be modified.                               |  |
| Source    | The source channel number which should be assigned to the specified interrupt vector. |  |

#### **Additional Information**

This function assigns a hardware interrupt line to an interrupt vector of VIC type only. It cannot be used for other types of vectored interrupt controllers. The hardware interrupt channel number of specific peripherals depends on specific CPU derivatives and has to be taken from the hardware manual of the CPU.

#### Example

```
/* Install UART interrupt handler */
OS_ARM_InstallISRHandler(UART_ID, &COM_ISR); // UART interrupt vector
OS_ARM_ISRSetPrio(UART_ID, UART_PRIO); // UART interrupt priority
OS_ARM_EnableISR(UART_ID); // Enable UART interrupt
/* Install UART interrupt handler with VIC type interrupt controller*/
OS_ARM_InstallISRHandler(UART_INT_INDEX, &COM_ISR); // UART interrupt vector
OS_ARM_EnableISR(UART_INT_INDEX); // Enable UART interrupt vector
OS_ARM_EnableISRSource(UART_INT_SOURCE);
OS_ARM_EnableISRSource(UART_INT_SOURCE); // Enable UART interrupt source
```

## 6.4.1.8 OS\_ARM\_EnableISRSource()

#### Description

OS\_ARM\_EnableISRSource() is used to enable an interrupt input channel of an interrupt controller of VIC type.

#### Prototype

void OS\_ARM\_EnableISRSource(int SourceIndex);

#### **Parameters**

| Parameter   | Description                                             |  |
|-------------|---------------------------------------------------------|--|
| SourceIndex | Index of the interrupt channel which should be enabled. |  |

#### **Additional Information**

This function enables a hardware interrupt input of a VIC-type interrupt controller. It cannot be used for other types of vectored interrupt controllers. The hardware interrupt channel number of specific peripherals depends on specific CPU derivatives and has to be taken from the hardware manual of the CPU.

#### Example



# 6.4.1.9 OS\_ARM\_DisableISRSource()

#### Description

OS\_ARM\_DisableISRSource() is used to disable an interrupt input channel of an interrupt controller of VIC type.

#### Prototype

void OS\_ARM\_DisableISRSource(int SourceIndex);

#### Parameters

| Parameter   | Description                                              |  |
|-------------|----------------------------------------------------------|--|
| SourceIndex | Index of the interrupt channel which should be disabled. |  |

#### **Additional Information**

This function disables a hardware interrupt input of a VIC-type interrupt controller. It cannot be used for other types of vectored interrupt controllers. The hardware interrupt channel number of specific peripherals depends on specific CPU derivatives and has to be taken from the hardware manual of the CPU.

# 6.4.1.10 OS\_ARM\_SetISRCfg()

#### Description

OS\_ARM\_SetISRCfg() sets the interrupt configuration.

#### Prototype

#### Parameters

| Parameter | Description                                                                                     |  |
|-----------|-------------------------------------------------------------------------------------------------|--|
| ISRIndex  | ndex of the interrupt source.                                                                   |  |
| Cfg       | 0: Corresponding interrupt is level-sensitive.<br>1: Corresponding interrupt is edge-triggered. |  |

# 6.4.1.11 OS\_ARM\_SetVBAR()

#### Description

OS\_ARM\_SetVBAR() writes the vector table address register.

#### Prototype

void OS\_ARM\_SetVBAR(OS\_U32 Addr);

#### Parameters

| Parameter | Description                  |
|-----------|------------------------------|
| Addr      | Address of the vector table. |

# 6.5 Interrupt-stack switching

Because ARM core based controllers have a separate stack pointer for interrupts, there is no need for explicit stack-switching in an interrupt routine. The routines  $OS_{INT\_En-terIntStack()}$  and  $OS_{INT\_LeaveIntStack()}$  are supplied for source compatibility to other processors only and have no functionality.

The ARM interrupt stack is used for the low-level interrupt handler  $\tt IRQ\_Handler()$  in the embOS library only.

# 6.6 Fast Interrupt (FIQ)

The FIQ interrupt cannot be used with embOS functions, it is reserved for high speed user functions.

Note the following:

- FIQ is never disabled by embOS.
- Never call any embOS function from an FIQ handler.
- Do not assign any embOS interrupt handler to FIQ.

#### Note

When you decide to use FIQ, ensure the FIQ stack is initialized during startup and that an interrupt vector for FIQ handling is included in your application.

# Chapter 7 MMU/MPU and cache support

# 7.1 Introduction

This chapter describes the MMU/MPU and cache support for ARM CPUs. With the ARM core the MMU is part of the Virtual Memory System Architecture (VMSA) and the MPU is part of the Protected Memory System Architecture (PMSA). embOS comes with functions to support the MMU/MPU and cache of ARMv4, ARMv5 and ARMv7-A/ARMv7-R CPUs.

# 7.2 MMU and cache handling for ARM720 CPUs

ARM720 CPUs with MMU have a unified cache for data and instructions. embOS delivers the following functions to setup and handle the MMU and cache.

# 7.2.1 API functions

| Function                          | Description                                           |
|-----------------------------------|-------------------------------------------------------|
| OS_ARM720_MMU_InitTT()            | Initialize the MMU translation table.                 |
| OS_ARM720_MMU_AddTTEntries()      | Add address entries to the table.                     |
| OS_ARM720_MMU_Enable()            | Enable the MMU.                                       |
| OS_ARM720_MMU_GetVirtualAddr()    | Translates a physical address into a virtual address. |
| OS_ARM720_MMU_v2p()               | Translates a virtual address into a physical address. |
| OS_ARM720_CACHE_Enable()          | Enable the cache.                                     |
| OS_ARM720_CACHE_CleanRange()      | Clean the cache.                                      |
| OS_ARM720_CACHE_InvalidateRange() | Invalidate the cache.                                 |

# 7.2.1.1 OS\_ARM720\_MMU\_InitTT()

#### Description

OS\_ARM720\_MMU\_InitTT() is used to initialize an MMU translation table which is located in RAM. The table is filled with zero, thus all entries are marked invalid initially.

#### Prototype

void OS\_ARM720\_MMU\_InitTT (unsigned int\* pTranslationTable);

#### Parameters

| Parameter         | Description                                          |
|-------------------|------------------------------------------------------|
| pTranslationTable | Points to the base address of the translation table. |

#### **Additional information**

This function does not need to be called, if the translation table is located in ROM.

# 7.2.1.2 OS\_ARM720\_MMU\_AddTTEntries()

#### Description

OS\_ARM720\_MMU\_AddTTEntries() is used to add entries to the MMU address translation table. The start address of the virtual address, physical address, area size and cache modes are passed as parameter.

#### Prototype

#### **Parameters**

| Parameter         | Description                                                                                                                                                                                                                                                                                                                                  |
|-------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| pTranslationTable | Points to the base address of the translation table.                                                                                                                                                                                                                                                                                         |
| CacheMode         | Specifies the cache operating mode which should be used for<br>the selected area. May be one of the following modes:<br>OS_ARM_CACHEMODE_NC_NB<br>non-cacheable, non-bufferable<br>OS_ARM_CACHEMODE_C_NB<br>cacheable, non-bufferable<br>OS_ARM_CACHEMODE_NC_B<br>non-cacheable, bufferable<br>OS_ARM_CACHEMODE_C_B<br>cacheable, bufferable |
| VIndex            | Virtual address index, which is the start address of the virtu-<br>al memory address range with MBytes resolution.<br>VIndex = (virtual address >> 20)                                                                                                                                                                                       |
| PIndex            | Physical address index, which is the start address of the physical memory area range with MBytes resolution.<br>PIndex = (physical address >> 20)                                                                                                                                                                                            |
| NumEntries        | Specifies the size of the memory area in MBytes.                                                                                                                                                                                                                                                                                             |

#### **Additional information**

This function does not need to be called, if the translation table is located in ROM. The function adds entries for every section of one MegaByte size into the translation table for the specified memory area.

# 7.2.1.3 OS\_ARM720\_MMU\_Enable()

#### Description

 $\tt OS\_ARM720\_MMU\_Enable()$  is used to enable the MMU which will then perform the address mapping.

#### Prototype

void OS\_ARM720\_MMU\_Enable (unsigned int\* pTranslationTable);

#### **Parameters**

| Parameter         | Description                                          |
|-------------------|------------------------------------------------------|
| pTranslationTable | Points to the base address of the translation table. |

#### **Additional information**

As soon as the function was called, the address translation is active. The MMU table has to be setup before calling <code>OS\_ARM720\_MMU\_Enable()</code>.

# 7.2.1.4 OS\_ARM720\_MMU\_GetVirtualAddr()

#### Description

OS\_ARM720\_MMU\_GetVirtualAddr() is used to translate a physical address into a virtual address with specified cache mode.

#### Prototype

#### Parameters

| Parameter | Description                                                                                                                                                                                                           |
|-----------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| PAddr     | The physical address as unsigned long.                                                                                                                                                                                |
| CacheMode | The cache mode of the requested virtual address<br>May be one of the defined cache modes:<br>OS_ARM_CACHEMODE_NC_NB<br>OS_ARM_CACHEMODE_C_NB<br>OS_ARM_CACHEMODE_NC_B<br>OS_ARM_CACHEMODE_C_B<br>OS_ARM_CACHEMODE_ANY |

#### Return value

#### Additional information

The function may be useful to examine an address of memory mapped to a virtual address with specific cache mode.

For the CPU it may be necessary to write into a specific memory in uncached mode. This can be done by setting up the MMU table with different virtual address for the same physical memory with different cache modes. For efficiency reasons, the CPU should access the memory fully cached for normal operation.

When a peripheral or DMA accesses the same memory for reading, for example an LCD controller accesses the diplay buffer, or an Ethernet MAC access a transfer buffer, the CPU has to write the data uncached into this memory, or has to clean the cache after writing.

The function <code>OS\_ARM720\_MMU\_GetVirtualAddress()</code> can be used to find the address for uncached access.

The MMU table has to be setup before the function is called.

# 7.2.1.5 OS\_ARM720\_MMU\_v2p()

#### Description

 $OS\_ARM720\_MMU\_v2p()$  is used to translate a virtual address into a physical address.

#### Prototype

unsigned long OS\_ARM720\_MMU\_v2p (void\* pVAddr);

#### Parameters

| Parameter | Description                                   |
|-----------|-----------------------------------------------|
| pVAddr    | Pointer which represents the virtual address. |

#### Return value

The physical address which is mapped to the virtual address passed as parameter.

#### **Additional information**

The function can be used to examine the physical address of memory. The CPU normally operates with virtual addresses which may differ from the physical address of the memory. When a peripheral or DMA has to be programmed to access the same memory, the peripheral has to be programmed to access the physical memory. The function <code>os\_ARM720\_M-MU\_v2p()</code> can be used to find the physical address of a memory area. The MMU table has to be setup before the function is called.

# 7.2.1.6 OS\_ARM720\_CACHE\_Enable()

#### Description

OS\_ARM720\_CACHE\_Enable() is used to enable the data cache of the CPU.

#### Prototype

void OS\_ARM720\_CACHE\_Enable(void);

#### **Additional information**

As soon as the function was called, the unified cache is active. The MMU has to be set up and has to be enabled before the cache is enabled.

# 7.2.1.7 OS\_ARM720\_CACHE\_CleanRange()

#### Description

OS\_ARM720\_CACHE\_CleanRange() is used to clean a range in the data cache memory to ensure that the data is written from the data cache into the memory.

#### Prototype

#### Parameters

| Parameter | Description                                                           |
|-----------|-----------------------------------------------------------------------|
| р         | Points to the base address of the memory area that should be updated. |
| NumBytes  | Number of bytes which have to be written from cache to memory.        |

#### **Additional information**

Cleaning the data cache is needed, when data should be transferred by a DMA or other BUS master that does not use the data cache. When the CPU writes data into a cacheable area, the data might not be written into the memory immediately. When then a DMA cycle is started to transfer the data from memory to any other location or peripheral, the wrong data will be written.

Before starting a DMA transfer, a call of OS\_ARM720\_CACHE\_CleanRange() ensures, that the data is transferred from the data cache into the memory and the write buffers are drained.

The cache is cleaned line by line. Cleaning one cache line takes approximately 10 CPU cycles. As each cache line covers 32 bytes, the total time to invalidate a range may be calculated as:

t = (NumBytes / 32) \* (10 [CPU clock cycles] + Memory write time).

The real time depends on the content of the cache. If data in the cache is marked as dirty, the cache line has to be written to memory. The memory write time depends on the memory BUS clock and memory speed. If data has to be written to memory, the required cycles for this memory operation has to be added to the 10 CPU clock cycles for every 32 bytes to be cleaned.

## 7.2.1.8 OS\_ARM720\_CACHE\_InvalidateRange()

#### Description

OS\_ARM720\_CACHE\_InvalidateRange() is used to invalidate a memory area in the data cache. Invalidating means, mark all entries in the specified area as invalid. Invalidation forces re-reading the data from memory into the cache, when the specified area is accessed again.

#### Prototype

#### Parameters

| Parameter   | Description                                              |
|-------------|----------------------------------------------------------|
| SourceIndex | Index of the interrupt channel which should be disabled. |

#### Additional information

This function is needed, when a DMA or other BUS master is used to transfer data into the main memory and the CPU has to process the data after the transfer.

To ensure, that the CPU processes the updated data from the memory, the cache has to be invalidated. Otherwise the CPU might read invalid data from the cache instead of the memory.

Special care has to be taken, before the data cache is invalidated. Invalidating a data area marks all entries in the data cache as invalid. If the cache contained data which was not written into the memory before, the data gets lost. Unfortunately, only complete cache lines can be invalidated.

Therefore, it is requires, that the base address of the memory area has to be located at a 32 byte boundary and the number of bytes to be invalidated has to be a multiple of 32 bytes.

The debug version of embOS will call <code>OS\_Error()</code> with error code <code>OS\_ERR\_NON\_ALIGNED\_IN-VALIDATE</code>, if one of these restrictions is violated.

The cache is invalidated line by line. Invalidating one cache line takes approximately 10 CPU cycles. As each cache line covers 32 bytes, the total time to invalidate a range may be calculated as:

t = (NumBytes / 32) \* 10 [CPU clock cycles].

# 7.3 MMU handling for ARMv5/ARMv7-A CPUs

The MMU allows virtual-to-physical address mapping with sections of one MByte and cache control. The MMU requires a translation table which can be located in any data area, RAM or ROM, but has to be aligned at a 16Kbyte boundary. A translation table in RAM has to be set up during run time. embOS delivers API functions to set up this table.

CHAPTER 7

# 7.3.1 API functions

| Function                    | Description                                                |
|-----------------------------|------------------------------------------------------------|
| OS_ARM_MMU_InitTT()         | Initialize the MMU translation table.                      |
| OS_ARM_MMU_AddTTEntries()   | Add address entries to the table.                          |
| OS_ARM_MMU_Enable()         | Enable the MMU.                                            |
| OS_ARM_MMU_GetVirtualAddr() | Translates a physical address into a virtual ad-<br>dress  |
| OS_ARM_MMU_v2p()            | Translates a virtual address into a physical ad-<br>dress. |

# 7.3.1.1 OS\_ARM\_MMU\_InitTT()

#### Description

OS\_ARM\_MMU\_InitT() is used to initialize an MMU translation table which is located in RAM. The table is filled with zeroes, thus all entries are marked as OS\_ARM\_MMU\_UNMAPPED initially.

#### Prototype

void OS\_ARM\_MMU\_InitTT(unsigned int\* pTranslationTable);

#### Parameters

| Parameter         | Description                                          |
|-------------------|------------------------------------------------------|
| pTranslationTable | Points to the base address of the translation table. |

# 7.3.1.2 OS\_ARM\_MMU\_AddTTEntries()

#### Description

OS\_ARM\_MMU\_AddTTEntries() is used to add entries to the MMU address translation table. The start address of the virtual address, physical address, area size and cache modes are passed as parameter.

CHAPTER 7

#### Prototype

#### **Parameters**

| Parameter         | Description                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |
|-------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| pTranslationTable | Points to the base address of the translation table.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| CacheMode         | Specifies the cache operating mode and memory access permis-<br>sions which should be used for the selected area.<br>May be one of the following modes:<br>ARMv4/ARMv5:<br>OS_ARM_MMU_UINMAPPED:<br>The associated MVA is unmapped, and attempting to access it<br>generates a translation fault<br>OS_ARM_CACHEMODE_NC_NB:<br>non-cacheable, non-bufferable<br>OS_ARM_CACHEMODE_C_NE:<br>cacheable, non-bufferable<br>OS_ARM_CACHEMODE_C_NE:<br>cacheable, bufferable<br>OS_ARM_CACHEMODE_C_B:<br>cacheable, bufferable<br>OS_ARM_CACHEMODE_C_B:<br>cacheable, bufferable<br>OS_ARM_CACHEMODE_C_B:<br>cacheable, bufferable<br>ARMv7-A:<br>OS_ARM_CACHEMODE_STRONGLY_ORDERED:<br>Strongly ordered<br>OS_ARM_CACHEMODE_STRONGLY_ORDERED:<br>Strongly ordered<br>OS_ARM_CACHEMODE_STRONGLY_ORDERED:<br>Shareable Device<br>OS_ARM_CACHEMODE_MRITE_THROUGH:<br>Outer and Inner Write-Through, no Write-Allocate<br>OS_ARM_CACHEMODE_WRITE_THROUGH:<br>Outer and Inner Write-Back, no Write-Allocate<br>OS_ARM_CACHEMODE_MRITE_BACK_MO_ALLOC:<br>Outer and Inner Write-Back, no Write-Allocate<br>OS_ARM_CACHEMODE_NNITE_BACK_ALLOC:<br>Outer and Inner Write-Back, Mrite-Allocate<br>OS_ARM_CACHEMODE_NNITE_BACK_ALLOC:<br>Outer and Inner Write-Back, Write-Allocate<br>OS_ARM_CACHEMODE_NNITE_BACK_ALLOC:<br>Outer and Inner Write-Back, Write-Allocate<br>OS_ARM_CACHEMODE_NNITE_BACK_ALLOC:<br>Outer and Inner Write-Back, Write-Allocate<br>OS_ARM_CACHEMODE_NNITE_BACK_ALLOC:<br>Outer and Inner Write-Back, Mrite-Allocate<br>OS_ARM_CACHEMODE_NNITE_BACK_ALLOC:<br>Outer and Inner Write-Back, Write-Allocate<br>OS_ARM_CACHEMODE_NNITE:<br>Non-shareable Device<br>OS_ARM_MMU_NOACCESS:<br>All accesses generate Permission faults<br>OS_ARM_MMU_READONLY:<br>Read-only access<br>OS_ARM_MMU_READONLY:<br>Read-only access<br>OS_ARM_MMU_EXECUTE_NEVER:<br>Determines whether the memory region is executable |

| Parameter  | Description                                                                                                                                       |
|------------|---------------------------------------------------------------------------------------------------------------------------------------------------|
| VIndex     | Virtual address index, which is the start address of the virtual memory address range with MBytes resolution.<br>VIndex = (virtual address >> 20) |
| PIndex     | Physical address index, which is the start address of the physical memory area range with MBytes resolution.<br>PIndex = (physical address >> 20) |
| NumEntries | Specifies the size of the memory area in MBytes.                                                                                                  |

CHAPTER 7

#### **Additional information**

The function adds entries for every section of one MegaByte size into the translation table for the specified memory area.

The macros for normal memory, i.e. <code>OS\_ARM\_CACHEMODE\_WRITE\_THROUGH</code>, <code>OS\_AR-M\_CACHEMODE\_WRITE\_BACK\_NO\_ALLOC</code>, <code>OS\_ARM\_CACHEMODE\_NON\_CACHEABLE</code> and <code>OS\_AR-M\_CACHEMODE\_WRITE\_BACK\_ALLOC</code>, can be OR-red with <code>OS\_ARM\_MMU\_SHAREABLE</code> to mark normal memory as shareable.

OS\_ARM\_MMU\_NOACCESS, OS\_ARM\_MMU\_READWRITE, OS\_ARM\_MMU\_READONLY and OS\_ARM\_M-MU\_EXECUTE\_NEVER can be used in combination with the cache attribute defines. If no memory access permissions are set full memory access is allowed per default.

OS\_ARM\_MMU\_InitT() sets all entries to OS\_ARM\_MMU\_UNMAPPED. The MMU table does not need to define such entries.

## 7.3.1.3 OS\_ARM\_MMU\_Enable()

#### Description

 ${\tt OS\_ARM\_MMU\_Enable()}$  is used to enable the MMU which will then perform the address mapping.

#### Prototype

void OS\_ARM\_MMU\_Enable(unsigned int\* pTranslationTable);

#### Parameters

| Parameter         | Description                                          |
|-------------------|------------------------------------------------------|
| pTranslationTable | Points to the base address of the translation table. |

#### **Additional information**

As soon as the function was called, the address translation is active. The MMU table has to be setup before calling  $OS_{ARM_MU_Enable()}$ .

OS\_ARM\_MMU\_Enable() also enables the branch prediction unit of Cortex-A CPUs.

# 7.3.1.4 OS\_ARM\_MMU\_GetVirtualAddr()

#### Description

OS\_ARM\_MMU\_GetVirtualAddr() is used to translate a physical address into a virtual address with specified cache mode.

#### Prototype

#### Parameters

| Parameter          | Description                                                                                                                                                                                                                                                                                                                                                                 |
|--------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| PAddr              | The physical address as unsigned long.                                                                                                                                                                                                                                                                                                                                      |
| PAddr<br>CacheMode | The physical address as unsigned long.The cache mode of the requested virtual addressMay be one of the defined cache modes:ARMv4/ARMv5:OS_ARM_CACHEMODE_NC_NBOS_ARM_CACHEMODE_C_NBOS_ARM_CACHEMODE_NC_BOS_ARM_CACHEMODE_C_BOS_ARM_CACHEMODE_C_BOS_ARM_CACHEMODE_ANYARMv7-A:OS_ARM_CACHEMODE_STRONGLY_ORDEREDOS_ARM_CACHEMODE_SHAREABLE_DEVICEOS_ARM_CACHEMODE_WRITE_THROUGH |
|                    | OS_ARM_CACHEMODE_WRITE_BACK_NO_ALLOC                                                                                                                                                                                                                                                                                                                                        |
|                    | OS_ARM_CACHEMODE_NON_CACHEABLE                                                                                                                                                                                                                                                                                                                                              |
|                    | OS_ARM_CACHEMODE_WRITE_BACK_ALLOC                                                                                                                                                                                                                                                                                                                                           |
|                    | OS_ARM_CACHEMODE_NON_SHAREABLE_DEVICE                                                                                                                                                                                                                                                                                                                                       |
|                    | OS_ARM_CACHEMODE_ANY                                                                                                                                                                                                                                                                                                                                                        |

#### **Return value**

void\* to the first virtual address found. A value of  ${\tt 0xFFFFFFF}$  indicates that no entry was found.

#### **Additional information**

The function may be useful to examine an address of memory mapped to a virtual address with specific cache mode. For the CPU it may be necessary to write into a specific memory in uncached mode. This can be done by setting up the MMU table with different virtual address for the same physical memory with different cache modes. For efficiency reasons, the CPU should access the memory fully cached for normal operation. When a peripheral or DMA accesses the same memory for reading, for exaplme an LCD controller accesses the diplay buffer, or an Ethernet MAC access a transferbuffer, the CPU has to write the data uncached into this memory, or has to clean the cache after writing. The function OS\_ARM\_MMU\_GetVirtualAddress() can be used to find the address for uncached access. The MMU table has to be setup before the function is called.

#### OS\_ARM\_MMU\_v2p() 7.3.1.5

### Description

 $OS_ARM_MU_v2p()$  is used to translate a virtual address into a physical address.

## Prototype

unsigned long OS\_ARM\_MMU\_v2p(void\* pVAddr);

#### **Parameters**

| Parameter | Description                                   |
|-----------|-----------------------------------------------|
| pVAddr    | Pointer which represents the virtual address. |

#### **Return value**

The physical address which is mapped to the virtual address passed as parameter.

#### Additional information

The function can be used to examine the physical addresss of memory. The CPU normally operates with virtual addresses which may differ from the physical address of the memory. When a peripheral or DMA has to be programmed to access the same memory, the peripheral has to be programmed to access the physical memory. The function OS\_ARM\_M- $MU_v2p()$  can be used to find the physical address of a memory area. The MMU table has to be setup before the function is called.

# 7.4 MPU handling for ARMv7-R CPUs

The ARMv7-R MPU is used to set cache and access settings for memory regions.

# 7.4.1 API functions

| Function                      | Description                                          |
|-------------------------------|------------------------------------------------------|
| OS_ARM_MPU_AddEntry()         | Sets an ARMv7-R PMSA MPU memory region.              |
| OS_ARM_MPU_Enable()           | Enables the ARMv7-R PMSA MPU.                        |
| OS_ARM_MPU_GetMinRegionSize() | Returns the ARMv7-R PMSA minimum memory region size. |
| OS_ARM_MPU_GetNumRegions()    | Returns the number of available memory re-<br>gions. |
| OS_ARM_MPU_Init()             | Initializes the ARMv7-R PMSA MPU.                    |

# 7.4.1.1 OS\_ARM\_MPU\_AddEntry()

#### Description

OS\_ARM\_MPU\_AddEntry() sets an ARMv7-R PMSA MPU memory region.

#### Prototype

OS\_U32 Size, OS\_U32 Permissions, OS\_U32 Attributes);

#### Parameters

| Parameter   | Description                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                     |
|-------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Region      | Region index                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                    |
| BaseAddr    | Memory region address                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           |
| Size        | Memory region size in bytes                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                     |
| Permissions | OS_ARM_MPU_NOACCESS:<br>No read or write access<br>OS_ARM_MPU_READWRITE:<br>Read and write access<br>OS_ARM_MPU_READONLY:<br>Read access only<br>OS_ARM_MPU_EXECUTE_NEVER:<br>No code execution allowed                                                                                                                                                                                                                                                                                                                                                                                                         |
| Attributes  | OS_ARM_CACHEMODE_STRONGLY_ORDERED:<br>Strongly ordered<br>OS_ARM_CACHEMODE_SHAREABLE_DEVICE:<br>Shareable Device<br>OS_ARM_CACHEMODE_WRITE_THROUGH:<br>Outer and Inner Write-Through, no Write-Allocate<br>OS_ARM_CACHEMODE_WRITE_BACK_NO_ALLOC:<br>Outer and Inner Write-Back, no Write-Allocate<br>OS_ARM_CACHEMODE_NON_CACHEABLE:<br>Outer and Inner Non-cacheable<br>OS_ARM_CACHEMODE_WRITE_BACK_ALLOC:<br>Outer and Inner Write-Back, Write-Allocate<br>OS_ARM_CACHEMODE_WRITE_BACK_ALLOC:<br>Outer and Inner Write-Back, Write-Allocate<br>OS_ARM_CACHEMODE_NON_SHAREABLE_DEVICE:<br>Non-shareable Device |

#### **Additional information**

The region index starts at zero for the first region. The number of available regions can be read with  $OS\_ARM\_MPU\_GetNumRegions()$ . The regions size must be aligned to the PMSA regions size which can be read with  $OS\_ARM\_MPU\_GetMinRegionSize()$ .

The macros for normal memory, i.e. <code>OS\_ARM\_CACHEMODE\_WRITE\_THROUGH</code>, <code>OS\_ARM\_CACHEMODE\_WRITE\_BACK\_NO\_ALLOC</code>, <code>OS\_ARM\_CACHEMODE\_NON\_CACHEABLE</code> and <code>OS\_ARM\_CACHEMODE\_WRITE\_BACK\_ALLOC</code>, can be <code>OR-red</code> with <code>OS\_ARM\_MPU\_SHAREABLE</code> to mark normal memory as shareable.

#### Example

```
int __low_level_init(void) {
    //
    // Sets access and cache settings for 256 Bytes at address 0x00
    //
    OS_ARM_MPU_AddEntry(0u,(void*)0x0000000, 256,
```

OS\_ARM\_MPU\_READWRITE, OS\_ARM\_CACHEMODE\_WRITE\_BACK\_ALLOC);

return 1;

}

# 7.4.1.2 OS\_ARM\_MPU\_Enable()

#### Description

OS\_ARM\_MPU\_Enable() enables the ARMv7-R PMSA MPU.

#### Prototype

void OS\_ARM\_MPU\_Enable(void);

#### Additional information

OS\_ARM\_MPU\_Enable() has to be called after the the MPU was initialized and configured.

# 7.4.1.3 OS\_ARM\_MPU\_GetMinRegionSize()

#### Description

 $\texttt{OS}\_\texttt{ARM}\_\texttt{MPU}\_\texttt{GetMinRegionSize()}$  returns the ARMv7-R PMSA minimum memory region size.

#### Prototype

OS\_U32 OS\_ARM\_MPU\_GetMinRegionSize(void);

#### **Return value**

Minimum memory region size which can be used with this PMSA implementation.

## 7.4.1.4 OS\_ARM\_MPU\_GetNumRegions()

#### Description

OS\_ARM\_MPU\_GetNumRegions() returns the number of available memory regions.

#### Prototype

OS\_U32 OS\_ARM\_MPU\_GetNumRegions(void);

#### **Return value**

Number of available memory regions.

## 7.4.1.5 OS\_ARM\_MPU\_Init()

#### Description

OS\_ARM\_MPU\_Init() initializes the ARMv7-R PMSA MPU.

#### Prototype

void OS\_ARM\_MPU\_Init(void);

# 7.5 Cache handling for ARMv5/ARMv7 CPUs

ARM CPUs with MMU/MPU and cache have separate data and instruction caches. embOS delivers the following functions to setup and handle the MMU and caches.

CHAPTER 7

# 7.5.1 API functions

| Function                        | Description                                           |
|---------------------------------|-------------------------------------------------------|
| OS_ARM_ICACHE_Enable()          | Enable the instruction cache.                         |
| OS_ARM_ICACHE_Invalidate()      | Invalidates the complete instruction cache.           |
| OS_ARM_DCACHE_Enable()          | Enable the data cache.                                |
| OS_ARM_DCACHE_Invalidate()      | Invalidates the complete data cache.                  |
| OS_ARM_DCACHE_Clean()           | Clean data cache.                                     |
| OS_ARM_DCACHE_CleanRange()      | Clean data cache range.                               |
| OS_ARM_DCACHE_InvalidateRange() | Invalidate the data cache.                            |
| OS_ARM_CACHE_Sync()             | Syncs data and instruction cache.                     |
| OS_ARM_AddL2Cache()             | Sets 2nd level cache API table.                       |
| OS_ARM_CACHE_GetLineSize()      | Returns cache line size of the specified cache level. |

## 7.5.1.1 OS\_ARM\_ICACHE\_Enable()

#### Description

OS\_ARM\_ICACHE\_Enable() is used to enable the instruction cache of the CPU.

#### Prototype

void OS\_ARM\_ICACHE\_Enable(void);

#### Additional information

As soon as the function was called, the instruction cache is active. It is CPU implementation defined whether the instruction cache works without MMU. Normally, the MMU should be setup before activating instruction cache.

## 7.5.1.2 OS\_ARM\_ICACHE\_Invalidate()

#### Description

OS\_ARM\_ICACHE\_Invalidate() invalidates the complete instruction cache. Invalidating means, mark all entries in the specified area as invalid. Invalidation forces re-reading the code from memory into the cache, when the specified area is accessed again.

#### Prototype

```
void OS_ARM_ICACHE_Invalidate(void);
```

## 7.5.1.3 OS\_ARM\_DCACHE\_Enable()

#### Description

OS\_ARM\_DCACHE\_Enable() is used to enable the data cache of the CPU.

#### Prototype

void OS\_ARM\_DCACHE\_Enable(void);

#### **Additional information**

The function must not be called before the MMU translation table was set up correctly and the MMU was enabled. As soon as the function was called, the data cache is active, according to the cache mode settings which are defined in the MMU translation table. It is CPU implementation defined whether the data cache is a write through, a write back, or a write through/write back cache. Most modern CPUs will have implemented a write through/ write back cache.

## 7.5.1.4 OS\_ARM\_DCACHE\_Invalidate()

#### Description

OS\_ARM\_DCACHE\_Invalidate() invalidates the complete data cache. Invalidating means, mark all entries in the specified area as invalid. Invalidation forces re-reading the data from memory into the cache, when the specified area is accessed again.

#### Prototype

```
void OS_ARM_DCACHE_Invalidate(void);
```

### 7.5.1.5 OS\_ARM\_DCACHE\_Clean()

#### Description

 ${\tt OS\_ARM\_DCACHE\_Clean()}$  is used to clean the data cache memory without invalidating the instruction cache.

#### Prototype

void OS\_ARM\_DCACHE\_Clean(void);

#### **Additional information**

Cleaning the data cache is needed, when data should be transferred by a DMA or other BUS master that does not use the data cache. When the CPU writes data into a cacheable area, the data might not be written into the memory immediately. When then a DMA cycle is started to transfer the data from memory to any other location or peripheral, the wrong data will be written.

The cache is cleaned line by line. Cleaning one cache line takes approximately 10 CPU cycles. The total time to invalidate a range may be calculated as:

t = (NumBytes / Cache line size) \* (10 [CPU clock cycles] + Memory write time).

The real time depends on the content of the cache. If data in the cache is marked as dirty, the cache line has to be written to memory. The memory write time depends on the memory BUS clock and memory speed. If data has to be written to memory, the required cycles for this memory operation has to be added to the 10 CPU clock cycles for every cache line to be cleaned.

## 7.5.1.6 OS\_ARM\_DCACHE\_CleanRange()

#### Description

 $OS\_ARM\_DCACHE\_CleanRange()$  is used to clean a range in the data cache memory to ensure that the data is written from the data cache into the memory.

#### Prototype

#### Parameters

| Parameter | Description                                                           |
|-----------|-----------------------------------------------------------------------|
| р         | Points to the base address of the memory area that should be updated. |
| NumBytes  | Number of bytes which have to be written from cache to memory.        |

#### **Additional information**

Cleaning the data cache is needed, when data should be transferred by a DMA or other BUS master that does not use the data cache. When the CPU writes data into a cacheable area, the data might not be written into the memory immediately. When then a DMA cycle is started to transfer the data from memory to any other location or peripheral, the wrong data will be written.

Before starting a DMA transfer, a call of  $OS\_ARM\_DCACHE\_CleanRange()$  ensures, that the data is transferred from the data cache into the memory and the write buffers are drained.

The cache is cleaned line by line. Cleaning one cache line takes approximately 10 CPU cycles. The total time to invalidate a range may be calculated as:

t = (NumBytes / Cache line size) \* (10 [CPU clock cycles] + Memory write time).

The real time depends on the content of the cache. If data in the cache is marked as dirty, the cache line has to be written to memory. The memory write time depends on the memory BUS clock and memory speed. If data has to be written to memory, the required cycles for this memory operation has to be added to the 10 CPU clock cycles for every cache line to be cleaned.

#### Note

Unfortunately, only complete cache lines can be cleaned. Therefore, it is required, that the base address of the memory area has to be located at a cache line size byte boundary and the number of bytes to be cleaned has to be a multiple of the cache line size. The debug version of embOS will call <code>OS\_Error()</code> with error code <code>OS\_ERR\_NON\_ALIGNED\_INVALIDATE</code>, if one of these restrictions is violated.

## 7.5.1.7 OS\_ARM\_DCACHE\_InvalidateRange()

#### Description

OS\_ARM\_DCACHE\_InvalidateRange() is used to invalidate a memory area in the data cache. Invalidating means, mark all entries in the specified area as invalid. Invalida- tion forces rereading the data from memory into the cache, when the specified area is accessed again.

#### Prototype

#### Parameters

| Parameter | Description                                                           |  |
|-----------|-----------------------------------------------------------------------|--|
| р         | Points to the base address of the memory area that should be updated. |  |
| NumBytes  | Number of bytes which have to be written from cache to memory.        |  |

#### **Additional information**

This function is needed, when a DMA or other BUS master is used to transfer data into the main memory and the CPU has to process the data after the transfer.

To ensure, that the CPU processes the updated data from the memory, the cache has to be invalidated. Otherwise the CPU might read invalid data from the cache instead of the memory.

Special care has to be taken, before the data cache is invalidated. Invalidating a data area marks all entries in the data cache as invalid. If the cache contained data which was not written into the memory before, the data gets lost. The cache is invalidated line by line. Invalidating one cache line takes approximately 10 CPU cycles. The total time to invalidate a range may be calculated as: t = (NumBytes / Cache line size) \* 10 [CPU clock cycles]. Notes

#### Note

Unfortunately, only complete cache lines can be invalidated. Therefore, it is required, that the base address of the memory area has to be located at a cache line size byte boundary and the number of bytes to be invalidated has to be a multiple of the cache line size. The debug version of embOS will call <code>OS\_Error()</code> with error code <code>OS\_ERR\_NON\_ALIGNED\_INVALIDATE</code>, if one of these restrictions is violated.

## 7.5.1.8 OS\_ARM\_CACHE\_Sync()

#### Description

 ${\tt OS\_ARM\_CACHE\_Sync()}$  cleans the data cache and invalidates the instruction cache to to ensure cache coherency.

#### Prototype

void OS\_ARM\_CACHE\_Sync(void);

#### **Additional information**

This function is for example needed, when code is copied into RAM and code is then executed from RAM.

### 7.5.1.9 OS\_ARM\_AddL2Cache()

#### Description

 $\texttt{OS}\_\texttt{ARM}\_\texttt{AddL2Cache()}$  is used to add.

#### Prototype

#### **Parameters**

| Parameter | Description                                                  |
|-----------|--------------------------------------------------------------|
| pCacheAPI | Pointer to 2nd level Cache API table.                        |
| pParam    | Additional parameter (e.g. base address or cache registers). |

#### Additional information

This function s needed to enable the L2 cache. Nothing else is necessary to do since the actual L2 cache routines are automatically called by the L1 cache routines. For example OS\_ARM\_DCACHE\_InvalidateRange() calls also internally the according L2 cache routine.

#### Example

#define L2CACHE\_BASE\_ADDR 0x3FFFF000u
//
// Set API functions and base address for L2 Cache
//
OS\_ARM\_AddL2Cache(&OS\_L2CACHE\_L2C310, (void\*)L2CACHE\_BASE\_ADDR);

## 7.5.1.10 OS\_ARM\_CACHE\_GetLineSize()

#### Description

OS\_ARM\_CACHE\_GetLineSize() returns the cache line size of the specified cache level.

#### Prototype

OS\_U32 OS\_ARM\_CACHE\_GetLineSize(OS\_U32 CIndex);

#### Parameters

| Parameter | Description                                                              |
|-----------|--------------------------------------------------------------------------|
| CIndex    | Index of the cache level of which the cache line size shall be returned. |

#### Additional information

The returned cache line size can be used to calculate the alignment and number of bytes passed to the <code>OS\_ARM\_DCACHE\_InvalidateRange()</code> and <code>OS\_ARM\_DCACHE\_CleanRange()</code> functions.

# 7.6 MMU and cache handling program sample

The MMU and cache handling has to be set up before the data segments are initial-ized. Otherwise a virtual address mapping would not work. The startup code must call a \_\_low\_lev-el\_init() function before sections are initialized.

It is a good idea to initialize memory access, the MMU table and the cache control during \_\_low\_level\_init() . The following sample is an excerpt from one \_\_low\_level\_init() function which is part of an RTOSInit.c file:

```
*
* MMU and cache configuration
*
* The MMU translation table has to be aligned to 16KB boundary
*
  and has to be located in uninitialized data area
*/
#pragma data_alignment=16384
 _no_init static unsigned int _TranslationTable [0x1000]; // OS_INTERWORK int
int __low_level_init(void) {
 11
  // Init MMU and caches
 11
 OS_ARM_MMU_InitTT (&_TranslationTable[0]);
 11
  // Internal SRAM, the first MB remapped to 0,
  // cacheable, bufferable, region not executable
  11
 OS_ARM_MMU_AddTTEntries ( &_TranslationTable[0],
                          OS_ARM_CACHEMODE_C_B | OS_ARM_MMU_EXECUTE_NEVER,
                          0x000, 0x200, 0x001);
  11
 // Internal SRAM, original address, NON cachable, NON bufferable
  11
 OS_ARM_MMU_AddTTEntries ( &_TranslationTable[0],
                          OS_ARM_CACHEMODE_NC_NB,
                          0x200, 0x200, 0x001);
 OS_ARM_MMU_Enable (&_TranslationTable[0]);
 OS_ARM_ICACHE_Enable();
 OS_ARM_DCACHE_Enable();
 return 1;
}
```

Other samples are included in the CPU specific RTOSInit\*.c files delivered with embOS.

# 7.7 MPU and cache handling program sample

```
int __low_level_init(void) {
 11
  // Enable MPU, Caches and branch prediction unit
 11
 OS_ARM_DCACHE_Enable();
 OS_ARM_ICACHE_Enable();
 OS_ARM_MPU_Init();
 11
 // Add MPU regions to set cache settings for different memory sections
 11
 OS_ARM_MPU_AddEntry(0u, (void*)0x00000000, 0x00140000,
   OS_ARM_MPU_READONLY, OS_ARM_CACHEMODE_WRITE_BACK_ALLOC); // FLASH
 OS_ARM_MPU_AddEntry(1u, (void*)0x08000000, 0x00030000,
   OS_ARM_MPU_READWRITE, OS_ARM_CACHEMODE_WRITE_BACK_ALLOC); // RAM
 OS_ARM_MPU_Enable();
 return 1;
}
```

# Chapter 8 VFP and NEON support

# 8.1 Introduction

Some ARM MCUs come with integrated Arm VFP and NEON units.

When activating the VFP or NEON support in the project options, the compiler and linker will add efficient code which uses the VFP/NEON register bank and VFP/NEON instructions where possible in the application.

With embOS, the VFP/NEON registers are automatically saved and restored when preemptive or cooperative task switches are performed. embOS also automatically saves and restores VFP/NEON registers for all embOS interrupt routines.

The VFP register bank consists of either 16 or 32 double-precision registers. The VFP register bank is also shared between the VFP and NEON units. If a NEON unit is implemented the VFP register bank consists of 32 64-bit double-precision registers. For a VFP unit with 32 double-precision registers or NEON unit all 32 double-precision registers are preserved, while for a VFP unit with 16 double-precision registers only the 16 double-precision registers need to be preserved.

embOS comes with libraries which preserve 16 double-precision registers D0-D15, 32 double-precision registers D0-D31 or none. Please have a look in the chapter *Libraries* on page 20 for more details.

#### Note

embOS ARM until V5.16.1.0 used task context extensions and ISR macros to preserve VFP/NEON registers. These API functions and macros are kept for compatibility but have no functionality anymore.

# 8.2 Using embOS libraries with VFP/NEON support

When VFP/NEON support is selected as project option, one of the embOS libraries with VFP/NEON support have to be used in the project. The embOS libraries with VFP/NEON support require that the VFP/NEON unit is switched on during startup and remains switched on during program execution. When the VFP/NEON unit is not switched on, the embOS scheduler will fail. Using your own startup code, ensure that the VFP/NEON unit is switched on during startup.

The debug version of embOS checks in <code>OS\_Init()</code> whether the VFP/NEON unit is switched on. If not, embOS calls <code>OS\_Error()</code> with the error code <code>OS\_ERR\_HW\_NOT\_AVAILABLE</code>.

# 8.3 Using the VFP/NEON unit in interrupt service routines

Using the VFP/NEON unit in embOS interrupt service routines does not require any additional functions to save and restore the VFP/NEON registers. embOS automatically saves and restores these registers.

VFP/NEON registers are not automatically saved and restored in zero latency interrupts. If the VFP/NEON unit is used in zero latency interrupts, it is the user's responsibility to preserve these registers.

# Chapter 9 RTT and SystemView

# 9.1 SEGGER Real Time Transfer

With SEGGER's Real Time Transfer (RTT) it is possible to output information from the target microcontroller as well as sending input to the application at a very high speed without affecting the target's real time behavior. SEGGER RTT can be used with any J-Link model and any supported target processor which allows background memory access.

RTT is included with many embOS start projects. These projects are by default configured to use RTT for debug output. Some IDEs, such as SEGGER Embedded Studio, support RTT and display RTT output directly within the IDE. In case the used IDE does not support RTT, SEGGER's J-Link RTT Viewer, J-Link RTT Client, and J-Link RTT Logger may be used instead to visualize your application's debug output.

For more information on SEGGER Real Time Transfer, refer to <u>segger.com/jlink-rtt</u>.

# 9.2 SEGGER SystemView

SEGGER SystemView is a real-time recording and visualization tool to gain a deep understanding of the runtime behavior of an application, going far beyond what debuggers are offering. The SystemView module collects and formats the monitor data and passes it to RTT.

SystemView is included with many embOS start projects. These projects are by default configured to use SystemView in debug builds. The associated PC visualization application, SystemView, is not shipped with embOS. Instead, the most recent version of that application is available for download from our website.

SystemView is initialized by calling SEGGER\_SYSVIEW\_Conf() on the target microcontroller. This call is performed within OS\_InitHW() of the respective RTOSInit\*.c file. As soon as this function was called, the connection of the SystemView desktop application to the target can be started. In order to remove SystemView from the target application, remove the SEGGER\_SYSVIEW\_Conf() call, the SEGGER\_SYSVIEW\_h include directive as well as any other reference to SEGGER\_SYSVIEW\_\* like SEGGER\_SYSVIEW\_TickCnt.

For more information on SEGGER SystemView and the download of the SystemView desktop application, refer to <u>segger.com/systemview</u>.

#### Note

SystemView uses embOS timing API to get at start the current system time. This requires that OS\_TIME\_ConfigSysTimer() was called before SEGGER\_SYSVIEW\_Start() is called or the SystemView PC application is started.

# Chapter 10 Technical data

# 10.1 Resource Usage

The memory requirements of embOS (RAM and ROM) differs depending on the used features, CPU, compiler, and library model. The following values are measured using embOS library mode  $OS\_LIBMODE\_XR$ .

| Module                       | Memory type | Memory requirements |
|------------------------------|-------------|---------------------|
| embOS kernel                 | ROM         | ~1700 bytes         |
| embOS kernel                 | RAM         | ~136 bytes          |
| Task control block           | RAM         | 36 bytes            |
| Software timer               | RAM         | 20 bytes            |
| Task event                   | RAM         | 0 bytes             |
| Event object                 | RAM         | 12 bytes            |
| Mutex                        | RAM         | 16 bytes            |
| Semaphore                    | RAM         | 8 bytes             |
| RWLock                       | RAM         | 28 bytes            |
| Mailbox                      | RAM         | 24 bytes            |
| Queue                        | RAM         | 32 bytes            |
| Watchdog                     | RAM         | 12 bytes            |
| Fixed Block Size Memory Pool | RAM         | 32 bytes            |