# Азбука халтурщика-ARМатурщика основы CMSIS

учебный курс по микроконтроллерам Cortex-Mx: Миландр 1986BE, STM32F, LPC21xx

(copypasta) Понятов Д.А. <dponyatov@gmail.com>, ИКП СГАУ

20 июля 2014 г.

# Оглавление

| 0.1 | (C)    |                                   | 3  |
|-----|--------|-----------------------------------|----|
| 0.2 |        |                                   | 4  |
| 0.3 | Структ | rypa CMSIS                        | 6  |
|     | 0.3.1  | Структура файлов                  | 9  |
|     | 0.3.2  | Независимость от тулчейна         | 10 |
|     | 0.3.3  | Стандарт безопасности ПО MISRA-C  | 12 |
|     | 0.3.4  | Функции CPAL                      | 12 |
|     | 0.3.5  | Подпрограммы обработки прерываний | 12 |
|     | 0.3.6  | Другие соглашения о коде          | 13 |
|     |        | Идентификаторы                    | 14 |
|     |        | Комментарии                       | 14 |
|     |        | Типы данных                       | 15 |
|     | 0.3.7  | Отладка                           | 16 |
|     | 0.3.8  | Контроль версии CMSIS             | 17 |
| 0.4 | Урок 1 | — Первый пример                   | 17 |
|     | 0.4.1  | Пример 1 Точка входа              | 18 |
|     | 0.4.2  | Пример 2 Адаптация для CMSIS      | 20 |
| 0.5 | Урок 2 | – ITM Отладка                     | 24 |
|     |        |                                   |    |

| 0.6  | Заключени | e | <br> | <br> | 26 |
|------|-----------|---|------|------|----|
| Лите | ратура    |   |      |      | 27 |

ı

This tutorial material is part of a series to be published progressively by Doulos. Этот учебный материал является частью регулярных публикаций ДвухЛосей

You can find the full set of currently published Tutorials and register for notification of future additional at www.doulos.com/knowhow

Вы можете найти полный набор опубликованных методичек и зарегестироваться для получения оповещений о новых выпусках на www.doulos.com/knowhow.

You can also download the full source code of the examples used within the Tutorial at the same URL. По тому же URLу вы также можете скачать полную версию исходных кодов примеров.

Also check out the Doulos ARM Training and service options at www.doulos.com/arm Также посмотрите варианты учебных курсов Duolos по ARMam на www.doulos.com/arm.

Or email <info@doulos.com> for further information Или запросите дополнительную информацию по <info@doulos.com>

First published by Doulos March 2009 Первая публикация от Duolos Март 2009

Copyright 2009 Doulos. All rights reserved. All trademarks acknowledged. All information is provided "as is" without warranty of any kind.

<sup>&</sup>lt;sup>1</sup>копипаста: двойной лось http://www.doulos.com/knowhow/arm/CMSIS/CMSIS\_Doulos\_Tutorial.pdf

# 0.2 Введение

The Cortex Microcontroller Software Interface Standard (CMSIS) supports developers and vendors in creating reusable software components for ARM Cortex-M based systems.

Cortex Microcontroller Software Interface Standard (CMSIS)<sup>2</sup> обеспечивает разработчикам и производителям МК создание повторно используемых программных компонентов для систем на основе микроконтроллеров Cortex-M.

The ARM Cortex-M3 processor is the first core from ARM specifically designed for the Microcontroller market. This core includes many common features (NVIC, Timer, Debug-hardware) needed for this market. This will enable developers to port and reuse software (e.g. a real time kernel) with much less effort to Cortex-M3 based MCUs.

Процессор ARM Cortex-M3 первое ядро от компании ARM специально разработанное для рынка микроконтроллеров. Это ядро включает множество типовых блоков (NVIC, таймеры, отладочный интрефейс) необходимых на этом рынке. Это позволяет разработчикам с минимальными усилиями портировать и повторно использовать уже написанное  $\Pi O^3$  для MK семейства Cortex-M3 любых производителей.

With a significant amount of hardware components being identical, a large portion of the Hardware Abstraction Layer (HAL) can be identical. However, reality has shown that lacking a common standard we find a variety of  $\rm HAL/driver$  libraries for different devices, which, as far as the Cortex-M3 part is concerned essentially do the same thing — just differently.

Благодаря идентичности большого колиства аппаратных компонентов, также идентичным оказывается и Hardware Abstraction Layer  $(HAL)^4$ . Тем не менее, реальность показывает что отсутствие общего стандарта приводит к множеству несовместимых версий библотек HAL и драйверов для различных MK, что не соответствует идее полной переносимости  $\Pi O$  в серии Cortex-M3.

The latest study of the development for the embedded market shows that software complexity and cost will increase over time, see figure left. Reusing Software and having a common standard to govern how to write and

<sup>&</sup>lt;sup>2</sup> стандарт программного интерфейса микроконтроллеров Cortex

<sup>&</sup>lt;sup>3</sup>например ядро ОС реального времени

<sup>&</sup>lt;sup>4</sup>программный слой аппаратной абстракции

debug the software will be essential to minimising costs for future developments.

Последние исследования разработок для рынка встраиваемого ПО показывают, что сложность программного обеспечения и его стоимость постоянно увеличиваются. Повторное использование кода и наличие общего стандарта управляющего способами написания и отладки ПО необходимы для минимизации стоимости разработки и сопровождения.



стоимости разработки

With more Cortex-M3 based MCUs about to come onto the market, ARM has recognized that after solving the diversity issue on the hardware side, there is still a need to create a standard to access these hardware components.

Анализируя ситуацию с взрывным ростом количества моделей МК Cortex-M3 на рынке, компания ARM обнаружила что полная идентичность аппаратной части недостаточна для обеспечения совместимости, и необходимо создание стандарта доступа к аппаратным компонентам.

The result of that effort is CMSIS; a framework to be extended by vendors, while taking advantage of a common API (Application Programming Interface) for core specific components and conventions that define how the device specific portions should be implemented to make developers feel right at home when they reuse code or develop new code for ARM Cortex-M based devices.

Результатом этих исследований является CMSIS: фреймворк, расширямый поставщиками МК, с со-

хранением полезных свойств общего API (Application Programming Interface)<sup>5</sup> для ядреных компонентов и соглашениями о том, как должны быть реализованы части зависимые от железа, чтобы разработчики чувствовали себя как дома при повторном использовании ил разработки нового кода для семейства Cortex-M.

# 0.3 Структура CMSIS

CMSIS can be divided into three basic function layers: CMSIS может быть поделен на три основных слоя:

• Core Peripheral Access Layer (CPAL)

The lowest level defines addresses, and access methods for common components and functionality that exists in every Cortex-M system. Access to core registers, NVIC, debug subsystem is provided by this layer. Tool specific access to special purpose registers (e.g. CONTROL, xPSR), will be provided in the form of inline functions or compiler intrinsics. This layer will be provided by ARM.

Самый нижний уровень определяет адреса, и методы доступа к общим компонентам и функциям, существующим в каждой Cortex-M-системе. Этим уровнем описывается доступ к регистрам ядра, NVIC<sup>6</sup>, подсистеме отладки. Инструментальный доступ к спецрегистрам (CONTROL,xPSR) предоставляется в форме inline-функций или интринсик компилятора. Этот уровень обеспечивается лицензиатом архитектуры — компанией ARM.

• Middleware Access Layer (MWAL)

This layer is also defined by ARM, but will be adapted by silicon vendors for their respective devices. The Middleware Access Layer defines a common API for accessing peripherals. The Middleware Access Layer is still under development and no further information is available at this point.

<sup>&</sup>lt;sup>5</sup>прикладной программный интерфейс

<sup>&</sup>lt;sup>6</sup>Nested Vector Interrupt Controller, контроллер вложенных прерываний

Этот слой также специфицируется ARM, но адаптируется производителем кристаллов для их конкретных изделий. Слой MWAL определяет общий API для доступа к периферии. Этот слой все еще находится на стадии доработки, и на текущий момент более подробная информация неступна.

#### • Device Peripheral Access Layer (DPAL)

Hardware register addresses and other definitions, as well as device specific access functions will be defined in this layer. The Device Peripheral Access Layer is very similar to the Core Peripheral Access Layer and will be provided by the silicon vendor. Access methods provided by CPAL may be referenced and the vector table will be adapted to include device specific exception handler address.

Слой содержит адреса аппаратных регистров и другие определения, в том числе функции доступа к специфичным особенностям чипов. DPAL сильно похож на CPAL, но предоставляется поставщиком кристаллов. В CPAL могут быть описаны методы доступа и адаптированная таблица векторов, содержащая обработчики исключений, специфичные для конкретного МК.

While DPAL is intended to be extended by the silicon vendor, let's not forget about Cortex-M based FPGA products, which effectively put developers into the position of a silicon vendor.

DPAL предназначен для расширения вендором, но не стоит забывать о FPGA-продуктах с примением Cortex-M-ядер, которые ставят разработчиков в положение вендора.

The basic structure and the functional flow is illustrated in the Figure 2. below.



Figure 2 CMSIS Structure functional flow Puc.2 Функциональная структура CMSIS

As far as MCU based systems are concerned it might make sense for developers to treat the entire PCB system as monolithic block. There is no reason to differentiate between a memory mapped register inside the MCU and a memory mapped register external to the MCU, connected via external memory interface. The

benefit of applying a standard like CMSIS is that existing guidelines on how to access these devices set a clear goal on how to implement and integrate critical parts of the software. Other team members will find a familiar environment.

### 0.3.1 Структура файлов

File names in CMSIS are standardized as follows:

```
core_cm3.h
core_cm3.c
core_cm3.c
core_cm3.c
core_cm3.c
core_cm3.d
Top-level header file (device specific). To be included by application code.
Includes core_cm3.h and system_<device>.h
system_cdevice>.h
Device specific declarations
system_cdevice>.c
Device specific definitions, e.g. SystemInit()
```

Application code will only include the top-level header file which implicitly pulls in all other essential



header files. The illustration below shows the flow and dependencies of the header files **stm32.h**, **core\_cm3.h** and **system stm32.h**, which are part of CMSIS release V1P0.

```
4 #define __Vendor_SysTickConfig 0 /*!< Set to 1 if different SysTick Config isused
5
6 #include "core_cm3.h" /* Cortex-M3 processor and core peripherals */
7 #include "system_stm32.h" /* STM32 System */</pre>
```

The **<device>.h** file is the central include file and provided by the silicon vendor. The application programmer is using that as the main include file in his C source code. Note that the ARM Cortex-M3 has some optional hardware features (e.g. the MPU, number of Interrupts and the number of the NVIC priority bits) the silicon vendors may have implemented differently. The listing above shows that STM32 implements four out of eight possible priority bits. The macro \_\_NVIC\_PRIO\_BITS is set here to 4. STM32 does not offer a Memory Protection Unit (MPU). Accordingly, the macro \_\_MPU\_PRESENT has the value 0.

The next example shows the corresponding definitions for a NXP LPC17xx device. In this Cortex-M3 implementation five priority bits have been implemented and an MPU is available.

The \_\_Vendor\_SysTickConfig defined is showing in both cases the default setting. When this macro is set to 1, the SysTickConfig() function in the cm3\_core.h is excluded. In this case the file <device>.h must contain a vendor specific implementation of this function.

#### 0.3.2 Независимость от тулчейна

CMSIS exists in a three-dimensional space of the form vendor÷device÷tool chain. In order to remove one dimension (tool chain), the common files **core cm3.c** and **core cm3.h** contain all essential tool specific

declarations and definitions.

#### Example:

```
1 /* define compiler specific symbols */
2 #if defined ( CC ARM )
     #define ASM asm
                                    /*!< asm keyword for armcc */</pre>
     #define INLINE inline
                                     /*!< inline keyword for armcc */</pre>
 #elif defined ( ICCARM )
     #define ASM asm
                                     /*!< asm keyword for iarcc */
                                     /*! < inline keyword for iarcc.
     #define INLINE inline
                                     Only avaiable in High optimization mode! */
                                     /*!< no operation intrinsic in iarcc */</pre>
     #define nop no operation
10 #elif defined ( GNUC )
     #define ASM asm
                                  /*!< asm keyword for gcc */
    #define INLINE inline
                                    /*!< inline keyword for gcc */</pre>
13 #endif
```

The remaining parts of CMSIS can now simply use the macro \_\_INLINE to define an inline function.

Остальная часть CMSIS теперь может просто использовать макрос \_\_INLINE для определения инлайнфункций.

Currently three of the most important C-compilers are supported: ARM RealView (armcc), IAR EWARM (iccarm), and GNU Compiler Collection (gcc). This is expected to cover the majority of tool chains.

На настоящий момент поддерживаются три наиболее применяемых Си-компилятора: ARM RealView (armcc), IAR EWARM (iccarm), и GNU Compiler Collection (gcc).

# 0.3.3 Стандарт безопасности ПО MISRA-C

Besides defining an API for Cortex-M core peripherals and guidelines on how to support device peripherals, CMSIS defines some coding guidelines and conventions. Most important is that the CMSIS code base is MISRA-C 2004 compliant, which implies that every extension should be compliant, too. MISRA-C is a set of safety rules established by the "Motor Industry Software Reliability Association" for the C programming language. Maintaining MISRA compliance can be tricky, in particular when implementing driver level software. Therefore, pragma-like exceptions in PCLint style are scattered across the source code. Be aware that other tools, e.g. MISRA checker in IAR EWARM, might flag errors. Each exception is accompanied with a comment explaining why this exception was made.

# 0.3.4 Функции CPAL

All functions in the Core Peripheral Access Layer are reentrant and can be called from different interrupt service routines (ISR). CPAL functions are also non-blocking<sup>7</sup> in the sense that they do not contain any wait-loops.

The majority of functions in the CPAL have been implemented in the header file **core\_cm3.h** as **static** inline functions. This allows the compiler to optimize the function calls by placing the instructions that make up the called function along with other code from which the function was called.

#### 0.3.5 Подпрограммы обработки прерываний

Exception handlers will get a name suffix \_Handler, while (external) interrupt handlers get the suffix \_IRQHandler. There must be a default handler for each interrupt, which executes an infinite loop. Tool specific configuration must make sure that this default handler will be used as fall-back if no user-provided handler exists. It done Through \_\_weak declaration in EWARM and RVCT armcc, \_\_attribute\_\_((weak)) in GCC and RVCT armcc and [WEAK] export in RVCT/armasm.

<sup>&</sup>lt;sup>7</sup>Memory barriers are exempt from that rule although they might stall the processor for a few cycles.

Given that the Cortex-M NVIC provides byte-arrays and bit-strings to configure priorities and interrupt source en-/disable, an enumerated type IRQn\_t with an element for each exception/interrupt position with the suffix \_IRQn must be defined for each interrupt (<device>.h). The system handler names are common for all devices and must not be changed.

Listing shows the generic part of the (**<device>.h**) file.

```
NonMaskableInt IRQn = -14, /*!< 2 Non Maskable Interrupt */
  MemoryManagement IRQn = -12, /*!< 4 Cortex-M3 Memory Mgmt Interrupt */
  BusFault IRQn = -11,
                         /*!< 5 Cortex-M3 Bus Fault Interrupt */</pre>
  UsageFault IRQn = -10, /*!< 6 Cortex-M3 Usage Fault Interrupt */
  SVCall IRQn = -5,
                    /*!< 11 Cortex-M3 SV Call Interrupt */
  DebugMonitor IRQn = -4, /*!< 12 Cortex-M3 Debug Monitor Interrupt */
  PendSV IRQn = -2,
                 /*! < 14 Cortex-M3 Pend SV Interrupt */
  SysTick IRQn = -1,
                /*!< 15 Cortex-M3 System Tick Interrupt */
 /*!< Example Interrupt */</pre>
12 UART IRQn = 0,
13 } IRQn Type;
```

All system handlers have negative virtual slot numbers so that they can be distinguished in functions that abstract from the differences between system handlers and external interrupt handlers. External interrupt handlers start at the index 0.

# 0.3.6 Другие соглашения о коде

The CMSIS documentation recommends a few more things regarding capitalization of identifiers, commenting code.

#### Идентификаторы

- Capital names to identify Core Registers, Peripheral Registers, and CPU Instructions.
  - E.g.: NVIC->AIRCR, GPIOB, LDMIAEQ
- "CamelCase" (mix of upper- and lower-case letters) names to identify peripherals access functions and interrupts.
  - E.g.: SysTickConfig(), DebugMonitor\_IRQn
- Peripheral prefix (<name>\_) to identify functions that belong to specific peripherals.
  - E.g.: ITM\_SendChar(), NVIC\_SystemReset()

#### Комментарии

CMSIS uses <u>Doxygen</u> style comments for all definitions and encourages developers to do the same. In particular, the comment for each function definition should at least contain

- one-line brief function overview. (Tag: @brief)
- detailed parameter explanation. (Tag: @param)
- detailed information about return values. (Tag: @return)
- detailed description of the actual function.

The example below shows the beginning of a function definition:

```
1 /**
2 * @brief Enable Interrupt in NVIC Interrupt Controller
3 *
4 * @param IRQn_Type IRQn specifies the interrupt number
```

```
* @return none

* * @return none

* * Enable a device specific interupt in the NVIC interrupt controller.

* The interrupt number cannot be a negative value.

9 */

10 static __INLINE void NVIC_EnableIRQ(IRQn_Type IRQn)

11 {
12 // . . .
13 }
```

The tags can be parsed by the documentation tool Doxygen, which is used to create cross-referenced source code documentation in various formats (http://www.stack.nl/~dimitri/doxygen/index.html). The tag syntax is rather minimalistic and does not impair readability of the source code. Please consult the Doxygen Documentation for details about tag syntax.

As an alternative to regular C block comments (/\* \*/) CMSIS explicitly allows line comments (//, so called C++-comments). If you are concerned about MISRA compliance, be aware though, that MISRA-C 2004 doesn't allow line comments according to rule 2.2.

CMSIS предлагает альтернативу обычным сишным /\* блочным комментариям \*/ в виде // строчных комментариев, так называмых  $C^{++}$ -комментариев. Если вас беспокоит выполнение превил MISRA, учтите что MISRA-C:2004 не разрешает использовать строчные комментарии согласно правилу 2.2.

#### Типы данных

All data types referenced by CMSIS are based on those defined in the standard C header file **stdint.h**. Data structures for core registers are defined CMSIS header file **core\_cm3.h**, along with macros for qualifying registers according to their access permissions. The rationale is that tools might be able to automatically extract that information for debug purposes.

| 2 #define | 0          | volatile | /*!< defines | <pre>'write only' permissions */</pre> |
|-----------|------------|----------|--------------|----------------------------------------|
| 3 #define | $^{-1}$ IO | volatile | /*!< defines | 'read / write' permissions */          |

#### 0.3.7 Отладка

A common requirement in software development is some sort of terminal output for debugging. Text/graphics displays in embedded devices cannot be assumed to be at hand (or might be in use), which previously left the developer with essentially two choices:

- 1. Use one of the ubiquitous UARTs and connect a terminal
  Issues: All UARTs might be in use, access to UART signals might not be possible for reasons that include
  pin-sharing, PCB layout, etc.
- 2. Use the semihosting mechanism

Issue: Significant software overhead on target CPU, might not be supported in the same way by all tool chains, potential impact on timing behavior.

With Cortex-M3 the preferred method makes use of the Instrumentation Trace Macrocell (ITM), which is part of the processor macro cell and thus always present.  $^8$ 

A Serial Wire Viewer (SWV) capable debug adapter can receive ITM data through the SWO (Serial Wire Out) debug pin. ITM implements 32 general purpose data channels. CMSIS builds on top of this and declares channel 0 to be used for terminal output, along with a function called ITM\_SendChar() which can be used as low-level driver function for printf-style output. A second channel (31) has been reserved for OS aware debugging, which means that a kernel can use it to transmit kernel specific data which could then be interpreted by the debug tool.

With this standardization, tool vendors have it much easier to implement specific debug features, such as e.g. terminal emulation for data received via ITM channel 0. Developers on the other hand can rely on this

<sup>&</sup>lt;sup>8</sup> Always present in Cortex-M3 rev1 cores. Cortex-M3 rev2 makes ITM an optional feature.

feature to dump state information, without having to configure UARTs and external terminal emulators. See our Tutorial 2 later in this document.

Access privilege can be configured for groups of ITM channels. In order to use ITM channel 0, unprivileged access must be granted, whereas ITM channel 31 is in a different group and may allow privileged access only.

#### 0.3.8 Контроль версии CMSIS

The CMSIS developers have taken care to provide macros indicating the CMSIS version used in a project. That way provisions can be made to prevent code to be used with a different CMSIS version than originally intended.

Разработчики CMSIS позаботились о предоставлении макросов версии CMSIS, использованной в проекте. Это способ для защиты от использования кода, предназначенного для другой версии CMSIS.

# 0.4 Урок 1 – Первый пример

In order to explain application of CMSIS in real projects, we are going to look at a simple example of a Cortex-M3 application. The program compiles for STM32 processors and a project file for the Keil  $\mu$ Vision IDE has been provided. Porting the example to other tool chains, such as IAR EWARM is straight forward and the IAR EWARM version is provided as well. A great number of CMSIS function definitions can be found in **core\_cm3.h** as "static inline" functions. Depending on the compiler optimization level, this helps getting very efficient instruction sequences rather than actual function calls, while ensuring a certain level of type safety.

After clock and GPIO initialization, the SysTick timer is configured to a period of 0.5 seconds. Whenever the handler executes it toggles the state of GPIOB[15].

#### 0.4.1 Пример 1 Точка входа

Initially, the program was implemented using STMicroelectronics' FWLib, a library that provides access to Cortex-M3 internals and STM32 peripherals. Near to medium term, firmware libraries such as FWLib will be based on CMSIS. Parts of FWLib that will eventually form the DPAL (see above).

```
1 #include < stdint.h>
  |#include <stm32f10x |lib.h>
  GPIO InitTypeDef GPIOB InitStruct = {
      .GPIO Pin = GPIO Pin All,
      . GPIO Speed = GPIO Speed 2MHz,
      .GPIO Mode = GPIO Mode Out PP
10
11 int main() {
      ErrorStatus HSEStartUpStatus;
12
13
      RCC ClocksTypeDef Clocks;
15
      /*
16
       * Clock initialization
17
       * /
18
      RCC HSEConfig(RCC HSE ON);
      HSEStartUpStatus = RCC WaitForHSEStartUp();
19
```

```
(HSEStartUpStatus != SUCCESS) {
22
          while (1);
23
      RCC SYSCLKConfig(RCC SYSCLKSource HSE);
25
      RCC HCLKConfig(RCC SYSCLK Div1);
26
      RCC PCLK2Config(RCC HCLK Div1);
28
      RCC APB2PeriphClockCmd(RCC APB2Periph GPIOB, ENABLE);
29
30
31
       * NVIC initialization
32
33
      NVIC PriorityGroupConfig(NVIC PriorityGroup 3);
      NVIC SystemHandlerPriorityConfig(SystemHandler SysTick, 7, 0);
34
35
      /*
36
37
         GPIOB initialization
38
      GPIO Init (GPIOB, &GPIOB InitStruct);
39
      GPIO WriteBit (GPIOB, GPIO Pin All, Bit RESET);
40
41
42
      /*
43
       * SysTick initialization
44
      SysTick CLKSourceConfig(SysTick CLKSource HCLK);
45
46
      RCC GetClocksFreq(&Clocks);
47
48
      SysTick SetReload ((Clocks. HCLK Frequency) / 2);
```

```
SysTick_ITConfig(ENABLE);
SysTick_CounterCmd(SysTick_Counter_Enable);
while(1);
```

```
static BitAction toggle = Bit_SET;

GPIO_WriteBit(GPIOB, GPIO_Pin_15, toggle);

if (toggle == Bit_SET) {
    toggle = Bit_RESET;
} else {
    toggle = Bit_SET;
}

by
}
```

The two listings above show the contents of main.c, and **stm32f10x\_it.c**. This latter file contains interrupt handler templates from ST's FWLib, which have to be adapted to implement project specific functionality.

# 0.4.2 Пример 2 Адаптация для CMSIS

void SysTickHandler(void) {

In a second step, the program has been converted to using CMSIS. The CMSIS version used is V1P10 as downloaded via the link above. We want to make sure to use the same CMSIS that has been used to develop the program and check the version number.

```
1 #include < stdint.h>
```

```
#include <stm32.h> // *** CMSIS change ***

#if (__CM3_CMSIS_VERSION != 0x00010010)
# error "__CM3_CMSIS_VERSION:_Unexpected_CMSIS_version_detected"

#endif
```

Initial support for STM32 MCU is part of CMSIS and is pulled in by including the header file **stm32.h**. At the point of writing this tutorial a fully CMSIS complaint FWLib was not available so some compromises and hand adjustments hand to be made. Fore this reason, we will include both, FWLib and CMSIS files. Until vendors have full adopted CMSIS small issues will have to be dealt with when combining CMSIS with a vendor library.

In this case simply including the FWLib main header file **stm32f10x\_lib.h** in addition to **stm32.h** triggers a number of error messages caused by multiple definitions of functions and macros. To avoid this, we will have to selectively include individual FWLib headers (see below). All FWLib headers depend on definitions in the files **cortexm3\_core.h** and **stm32f10x\_map.h**. Most of the definitions in these two header files have already been defined by CMSIS (**core\_cm3.h** and **system\_stm32.h**) and we have to pretend to FWLib that both header files had been included already.

```
// Prevent interference with FWLib

#define __STM32F10x_MAP_H

#include <stm32f10x_type.h>

#include <stm32f10x_gpio.h>

#include <stm32f10x_rcc.h>
```

Actual system initialization will be encapsulated by the CMSIS function SystemInit(), which has to be implemented by the silicon vendor. As a minimal requirement, this function would initialize the MCU's clock system. In case of the reference implementation in system\_stm32.c, SystemInit() also initializes the Flash memory interface. CMSIS defines a single system variable, SystemFrequency, which is supposed to reflect the frequency of both core and SysTick timer in Hz. This concept is sufficient for a minimal implementation

but will likely have to be extended for actual MCU as demonstrated by CMSIS' **system\_stm32.c**, in which several variables have been defined to hold the frequency values of different clock domains in the STM32 MCU. SysTick timer and core could have different frequencies and care must be taken when using SystemFrequency in a program.

Current CMSIS does not initialize peripheral clocks and it is arguable whether it should. In any case we use the corresponding FWLib function to enable GPIOB clock.

```
// Initialization moved to SystemInit() in system_stm32.c. Clock
// configuration now handled by #defines. Use uVision
// configuration wizard or text editor to change.
SystemInit(); // *** CMSIS change ***
// APB peripherals still have to be enabled individually.
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
```

NVIC group- and sub-priority configuration is handled by the function NVIC\_SetPriorityGrouping(), where the direct encoding of the PRIGROUP field in SCB->AIRCR is used. We choose the value 4, which represents 3 bits for group (preempting) and 5 bits for the sub priority. A general formula to calculate the proper value is PRIGROUP = bitssub-1.

```
1 // priority configuration: 3.5
2 NVIC_SetPriorityGrouping(4); // *** CMSIS change ***
```

Different from the initial version of the example, the CMSIS variant does not at this point set the SysTick handler priority. This is part of the SysTick initialization and will be covered later.

```
1 GPIO_Init (GPIOB, &GPIOB_InitStruct);
2 GPIO_WriteBit (GPIOB, GPIO_Pin_All, Bit_RESET);
```

Following NVIC set-up, we use plain FWLib functions to configure GPIO port B.

```
1 SysTick_Config(SystemFrequency/2); // *** CMSIS Change ***
```

```
2
3 // SysTick_Config() hardcodes priority. We will overwrite this.
4 NVIC_SetPriority(SysTick_IRQn, 14); // *** CMSIS change ***
```

SysTick\_Config(), provided by CMSIS, programs the reload register with the parameter value. The function also selects HCLK (core clock) as clock source, enables SysTick interrupts and starts the counter. The function also fixes the SysTick handler priority to the lowest priority in the system, which is the recommended SysTick priority for use in an RTOS scheduler for instance. In our example, however, we prefer a different priority and override the hard coded value with an additional call to NVIC\_SetPriority(). This function abstracts from the difference between Cortex-M3 system handlers and external interrupt handlers. All configurable system exceptions will be identified by negative IRQ numbers (see above).

The SysTick handler code above does not need any modification. FWLib naming conventions is complying with CMSIS, in that the names of all internal exception handlers must end in \_Handler. Names of external interrupt handlers must end in \_IRQHandler. The handler implementation accesses the GPIO port via FWLib functions and definitions.

# 0.5 Урок 2 – ITM Отладка

To exercise some CMSIS debug functionality for our first example from tutorial 1, we add debug output messages via calls to puts(). We will now redirect character output to Instrumentation Trace Macrocell (ITM), (remember that CMSIS reserves ITM channel 0 for this), using the CMSIS function ITM\_SendChar(). A mechanism called retargeting enables us to provide our own implementation of a system function.

```
1 // retarget fputc() for debug output via ITM
2 int fputc(int c, FILE *stream) {
3    return (int)ITM_SendChar((uint32_t)c);
4 }
```

The standard C function fputc(), which will eventually be called by puts() in our SysTick handler, will be re-implemented, taking advantage of the function ITM\_SendChar(). The result of this retarget can now be easily monitored in  $\mu$ Vision ITM viewer as shown in the screenshot below.

```
Pin state is ON
Pin state is OFF
Pin state is ON
Pin state is OFF
Pin state is ON
```

If you are going to use the IAR EWARM tool for this example it is not necessary to manually retarget this function. Instead, in the project options dialog under "General Options" there is a tab "Library Configuration" tab, which offers checkboxes to enable this functionality. The screenshot below shows which settings are required to redirect standard output.



#### 0.6 Заключение

The CMSIS will reduce the learning-curve for the application programmers by providing a consistent software framework, ensuring consistent documentation and easy deployment of boilerplate code across various compiler vendors. The consequent use and implementation of CMSIS across many silicon and middleware software partners will simplify the verification and certification process and therefore reduce future project risk. The adapted common programming techniques though CMSIS will simplify the long term maintenance due to easier to understand source code. The silicon vendors can focus on there added value and device features. All reasons together will reduce software development cost and time to get new products to the market.

# Литература

- [1] https://github.com/ponyatov/CortexMx Азбука халтурщика-ARMатурщика
- [2] Getting started with CMSIS http://www.doulos.com/knowhow/arm/CMSIS/CMSIS\_Doulos\_Tutorial.pdf
- [3] Ю.С. Магда Программирование и отладка C/C++ приложений для микроконтроллеров ARM. М.: ДМК Пресс, 2012. 168 с.: ил.
- [4] © Quantum  $^{\textcircled{R}}L^{e}aPs$
- [5] http://www.state-machine.com/arm/Building\_bare-metal\_ARM\_with\_GNU.pdf Quantum  $^{\circledR}L^eaPs$  Building Bare-Metal ARM Systems with GNU
- [6] http://milandr.ru/ ЗАО «ПКК Миландр»
- [7] http://git-scm.com/book/ru перевод: Scott Chacon Pro Git
- [8] http://habrahabr.ru/post/114239/ хабра: Quantum $^{\circledR}L^eaPs$  QP и диаграммы состояний в UML
- [9] http://www.state-machine.com/ Quantum $^{\circledR}L^eaPs$  State Machines & Tools

- [10] http://makesystem.net/?p=988 Изучаем ARM. Собираем свою IDE для ARM
- [11] http://makesystem.net/?p=2146 Изучаем ARM. Отладка ARM приложений в Eclipse IDE
- [12] Львовский С.М. Набор и вёрстка в пакете  $\mbox{\sc IAT}_{\mbox{\footnotesize E\!X}}$