



# **ADuCM4x50 Device Family Pack for Keil Users Guide**

# Contents

|        |                                                      |    |
|--------|------------------------------------------------------|----|
| 1      | Introduction                                         | 4  |
| 1.1    | Purpose                                              | 4  |
| 1.2    | Scope of this Manual                                 | 4  |
| 1.3    | Acronyms and Terms                                   | 5  |
| 1.4    | Conventions                                          | 6  |
| 1.5    | References                                           | 6  |
| 1.6    | Additional Information                               | 7  |
| 1.6.1  | Manual Contents                                      | 7  |
| 2      | Product Overview                                     | 8  |
| 2.1    | Software System Overview                             | 8  |
| 2.2    | Hardware System Overview                             | 8  |
| 2.3    | ADuCM4x50 Device Family Pack Directory Structure     | 10 |
| 2.3.1  | <ADuCM4x50_root>/Boards                              | 10 |
| 2.3.2  | <ADuCM4x50_root>/Documents                           | 10 |
| 2.3.3  | <ADuCM4x50_root>/Flash                               | 10 |
| 2.3.4  | <ADuCM4x50_root>/Include                             | 10 |
| 2.3.5  | <ADuCM4x50_root>/License                             | 10 |
| 2.3.6  | <ADuCM4x50_root>/Source                              | 10 |
| 2.3.7  | <ADuCM4x50_root>/SVD                                 | 10 |
| 2.3.8  | <ADuCM4x50_root>/tools                               | 11 |
| 2.4    | Documentation                                        | 11 |
| 2.4.1  | USB serial driver Installation                       | 11 |
| 2.4.2  | Setup a Terminal Emulator program on PC              | 12 |
| 2.5    | Technical or Customer Support                        | 13 |
| 3      | Installation Components                              | 14 |
| 3.1    | Keil Project Support Files                           | 14 |
| 3.1.1  | Scatter File                                         | 15 |
| 3.1.2  | J-Link Settings File                                 | 16 |
| 3.1.3  | Flash Programming Algorithms                         | 16 |
| 3.2    | Keil Project Options                                 | 16 |
| 3.2.1  | Options for Target                                   | 16 |
| 3.2.2  | Device Options                                       | 17 |
| 3.2.3  | Output Settings                                      | 18 |
| 3.2.4  | Listing                                              | 19 |
| 3.2.5  | User Settings                                        | 20 |
| 3.2.6  | C/C++ Settings                                       | 21 |
| 3.2.7  | ASM Settings                                         | 22 |
| 3.2.8  | Linker Settings                                      | 23 |
| 3.2.9  | Debugger Settings                                    | 24 |
| 3.2.10 | Debug Settings (J-Link/J-Trace Setup and Connection) | 25 |

|        |                                      |    |
|--------|--------------------------------------|----|
| 3.2.11 | Utilities                            | 28 |
| 3.2.12 | Manage Run-Time Environment          | 29 |
| 4      | ADuCM4x50 System Overview            | 31 |
| 4.1    | Block Diagram and Driver Layout      | 31 |
| 4.2    | Boot-Time CRC Validation             | 32 |
| 4.3    | System Reset Strategy                | 33 |
| 5      | Application Configuration            | 34 |
| 5.1    | Application Initialization           | 34 |
| 5.2    | Static Pin Multiplexing              | 35 |
| 5.3    | UART Baud Rate Configuration Utility | 37 |
| 5.4    | Driver Include Files                 | 38 |
| 5.5    | Driver Configuration                 | 38 |
| 5.5.1  | Global Configuration                 | 38 |
| 5.5.2  | Configuration Defaults               | 39 |
| 5.5.3  | Configuration Overrides              | 39 |
| 5.5.4  | IVT Table Location                   | 40 |
| 5.5.5  | Interrupt Callbacks                  | 40 |
| 6      | Examples                             | 42 |
| 6.1    | Build and Run                        | 42 |
| 7      | Device Driver API Documentation      | 45 |
| 7.1    | Appendix                             | 45 |
| 7.1.1  | CMSIS                                | 45 |
| 7.1.2  | Interrupt Vector Table               | 46 |
| 7.1.3  | Startup_<Device>.c Content           | 47 |
| 7.1.4  | System_<Device>.c Content            | 47 |

# **1 Introduction**

## **1.1 Purpose**

This document describes the ADuCM4x50 Device Family Pack (DFP) for Keil uVision and its use. The ADuCM4x50 processor integrates an ARM Cortex-M4 microcontroller with various on-chip peripherals within a single package.

## **1.2 Scope of this Manual**

This document describes how to install and work with the Analog Devices ADuCM4x50 DFP. This document explains what is included with the pack and how to configure the software to run the example applications that accompanies this package.

This document is intended for engineers who integrate ADI's device driver libraries with other software to build a system based on the ADuCM4x50 processor. This document assumes background in ADI's ADuCM4x50 processor.

## 1.3 Acronyms and Terms

|               |                                                    |
|---------------|----------------------------------------------------|
| <b>ADI</b>    | Analog Devices, Inc.                               |
| <b>API</b>    | Application Programming Interface                  |
| <b>ARM</b>    | Advanced RISC Microcontroller                      |
| <b>CMSIS</b>  | Cortex Microcontroller Software Interface Standard |
| <b>Cortex</b> | A series of ARM microcontroller core designs       |
| <b>CRC</b>    | Cyclic Redundancy Check                            |
| <b>HRM</b>    | Hardware Reference Manual                          |
| <b>ISR</b>    | Interrupt Service Routine                          |
| <b>IVT</b>    | Interrupt Vector Table                             |
| <b>JTAG</b>   | Joint Test Action Group                            |
| <b>MDK</b>    | Microcontroller Development Kit                    |
| <b>NVIC</b>   | Nested Vectored Interrupt Controller               |
| <b>RISC</b>   | Reduced Instruction Set Computer                   |
| <b>RTE</b>    | Run-Time Environment                               |
| <b>RTOS</b>   | Real-Time Operating System                         |
| <b>TRACE</b>  | Debugging with TRACE access port                   |

## 1.4 Conventions

Throughout this document, we refer to two important installation locations: the ADuCM4x50 DFP, ARM CMSIS, and the Keil toolchain installation root. Each of these packages can be installed in various places, which are referred to as follows:

- <dfp\_version>
  - The version of the ADuCM4x50 Device Family Pack, e.g. 1.0.0, 3.0.0.
- <Keil\_root>
  - The default KEIL Pack installer places the product at location **C:/Keil\_v5**, but the install location may vary depending on user preferences.
  - The default Analog Devices packs are placed at location <Keil\_root>/ARM/Pack/AnalogDevices. There will be the following folder for ADuCM4x50 within that location called **ADuCM4x50\_DFP**.
- <ADuCM4x50\_root>
  - The directory <Keil\_root>/ARM/Pack/AnalogDevices/**ADuCM4x50\_DFP**/**<dfp\_version>** which contains the content of the Analog Devices ADuCM4x50 DFP.
- <ARM\_CMSIS\_version>
  - The version of the ARM CMSIS pack, e.g. 4.5.0 or 5.0.1.
- <ARM\_CMSIS\_root>
  - The directory <Keil\_root>/ARM/Pack/ARM/CMSIS/<ARM\_CMSIS\_version> which contains the content of the ARM CMSIS pack.

## 1.5 References

1. Analog Devices : <ADuCM4x50\_root>/Documents
  - a. ADuCM4x50\_DFP\_<dfp\_version>\_Release\_Notes.pdf
  - b. ADuCM4x50\_DFP\_Device\_Drivers\_UsersGuide.pdf
  - c. ADuCM4x50\_DFP\_UsersGuide\_Keil.pdf (this document)
  - d. ADuCM4x50 Device Drivers API Reference Manual (Documents/html and hyperlinked)

2. For Keil <Keil\_root>/ARM/Hlp [<http://www.keil.com>]
  - a. Keil MDK for Cortex-M micro controller.
  - b. Release notes.
3. *The Definitive Guide to ARM Cortex-M3 and Cortex-M4 Processors*, Joseph Yiu, 3<sup>rd</sup> edition.
  - Every Cortex programmer's bible; a must-have reference.
4. Micrium [<http://micrium.com>]
  - a. uC/OS-II RTOS for ARM Cortex-M4
  - b. uC/OS-II User's Manual
5. SEGGER J-Link Emulator [<http://www.segger.com>]
6. ARM CMSIS pack [[www.keil.com/cmsis/pack](http://www.keil.com/cmsis/pack)]

## 1.6 Additional Information

For more information on the latest ADI processors, silicon errata, code examples, development tools, system services and devices drivers, technical support and any other additional information, please visit our website at [www.analog.com/processors](http://www.analog.com/processors).

### 1.6.1 Manual Contents

- Product Overview
- Installation Components
- ADuCM4x50 System Overview
- Build Configurations
- Examples
- Device Driver API Documentation

## 2 Product Overview

### 2.1 Software System Overview

The ADuCM4x50 Device Family Pack (DFP) provides files which are needed to write application software for the ADuCM4x50 processor. The product consists of a boot kernel, startup, system and driver source code, driver configuration settings, driver libraries, sample applications and associated documentation (*see Figure 1. Software Overview*).

The ADuCM4x50 DFP is designed to work with KEIL uVision in CMSIS pack format for ARM.

**Figure 1. Software Overview**



### 2.2 Hardware System Overview

The examples provided with the ADuCM4x50 DFP run on the Analog Devices' ADuCM4x50-EZ-Kit evaluation board. The evaluation board is connected to the host computer using a Segger J-Link lite emulator over the evaluation board's JTAG or TRACE debug port interface connectors. External I/O signals and system hardware are connected to the evaluation board connectors as shown in *Figure 3. ADuCM4x50 EZ-Board*.

**Figure 2. Hardware Overview**



**Figure 3. ADuCM4x50 EZ-Board**



## 2.3 ADuCM4x50 Device Family Pack Directory Structure

Keil toolchain support files are placed with the Keil installation folder (e.g. **C:\Keil\_v5\ARM\Pack\AnalogDevices\ADuCM4x50\_DFP**). This also includes files for flash loading, debugging, etc.

The ADuCM4x50 Device Family Pack software (startup code, device drivers, libraries, examples, tools, documentation, etc.) is placed at **<ADuCM4x50\_root>**.

The following list details the content of each DFP directory, **<ADuCM4x50\_root>**, which contains the content of the ADuCM4x50 CMSIS-Pack distribution.

### 2.3.1 <ADuCM4x50\_root>/Boards

This directory contains a collection of example projects.

### 2.3.2 <ADuCM4x50\_root>/Documents

This directory contains complete HTML documentation for all the ADuCM4x50 Device Drivers and API, as well as the device driver *Release Notes*, *Getting Started Guide* and *Software User's Guide*.

### 2.3.3 <ADuCM4x50\_root>/Flash

This directory contains the flash programmer algorithm.

### 2.3.4 <ADuCM4x50\_root>/Include

This directory contains the ADuCM4x50 Device Driver include files and default driver configuration files.

### 2.3.5 <ADuCM4x50\_root>/License

This directory contains the software license agreement.

### 2.3.6 <ADuCM4x50\_root>/Source

This directory contains the ADuCM4x50 CMSIS startup and system source files, as well as the ADuCM4x50 device driver source files.

### 2.3.7 <ADuCM4x50\_root>/SVD

This directory contains the ADuCM4x50 CMSIS System View Description (SVD) file.

## 2.3.8 <ADuCM4x50\_root>/tools

This directory contains the ADuCM4x50 tool extensions, such as the pin multiplexing configuration utility and a utility which provides the optimum values that need to be programmed into the UART controller register for a given baud rate.

## 2.4 Documentation

The following documentation is provided for this release in the main product documentation directory:

- ADuCM4x50\_DFP\_<dfp\_version>\_Release\_Notes.pdf
- ADuCM4x50\_DFP\_Device\_Drivers\_UsersGuide.pdf
- ADuCM4x50\_DFP\_UsersGuide\_Keil.pdf (this document)

The *ADuCM4x50 Device Drivers API Reference Manual* is published in hyperlinked HTML format under the html directory.

Launch the "<ADuCM4x50\_root>\Documents\html\index.html" file to browse the API documentation interactively.

The *ADuCM4x50 Device Drivers API Reference Manual* contains complete driver documentation, including API descriptions, data types, structures, parameters, return values, etc.

### 2.4.1 USB serial driver Installation

ADuCM4050 EZ-KIT Rev 1.x boards use the SEGGER J-Link LITE emulator. The J-Link software package can be downloaded from <http://www.segger.com/jlink-software.html>.

Please download the J-Link software and documentation pack for Windows, as shown in the screen below



The installation will ask you to choose optional components to install. Please ensure to select "Install USB Driver for J-Link", as shown in the screen below.



## 2.4.2 Setup a Terminal Emulator program on PC

Setup a Terminal Emulator program, such as HyperTerminal or TeraTerm to interact with the target over UART. To setup a Terminal Emulator session, configure the session on the PC as follows.

Select the appropriate communications channel COMx. You may use the Device Manager to locate the "USB Serial Port (COMx)" device under the Device Manager "Com\_Ports" section, where "x" corresponds to the target communications serial Port.

Then set the following attributes:

Communications Settings:

- Baud rate : 57600
- Data : 8 bit
- Parity : none
- Stop : 1 bit
- Flow control : none

## **2.5 Technical or Customer Support**

**Submit your questions online at:**

<http://www.analog.com/support>

**E-mail your Processors and DSP applications and processor questions to:**

[processor.support@analog.com](mailto:processor.support@analog.com) OR

[processor.china@analog.com](mailto:processor.china@analog.com) (Greater China support)

**For Keil toolchain support please visit**

<http://www.keil.com>

## 3 Installation Components

The Keil MDK <http://www2.keil.com/mdk5> or later must be purchased and installed prior to installing the ADuCM4x50 Software Package. Follow the instructions in the Keil MDK for ARM product installation procedure (Keil uVision Full license version).

Keil toolchain support files are placed with the Keil installation folder (e.g. **C:/Keil\_v5/ARM/Pack/AnalogDevices/ADuCM4x50\_DFP**). This also includes files for flash loading, debugging, etc.

The ADuCM4x50 Device Family Pack files (startup code, device drivers, libraries, examples, tools, documentation, etc.) are placed at <ADuCM4x50\_root>.

**Figure 4. Example of Installation Directory Structure for ADuCM4x50\_DFP version 1.0.0.**



### 3.1 Keil Project Support Files

This section documents the Keil-specific details of the ADuCM4x50 Device Family Pack installer. A working knowledge of the Keil toolchain and environment is assumed. See the Keil reference materials for details of installing, configuring and using the Keil tools. The following are the list of important files added by the ADuCM4x50 Device Family Pack installer, necessary to build applications in the Keil uVision environment.

- Scatter File (.sct)
- J-Link debugger setting file (JLinkSettings.ini)
- Flash Loader Algorithm (.FLM)

### 3.1.1 Scatter File

An scatter file contains the memory configuration of the ADuCM4x50 core, this file can be customized as per the application needs.

A scatter file can be stored in the same path as a Keil project file. The scatter file in the project can be used to define and allocate various memory regions:

- Internal SRAM size
- FLASH size
- Placement of all code and data blocks
- Reserves memory for post-link processing (CRC checksums, parity, etc)

The scatter file can be used to:

- Define Memory “Regions” (size, location, alignment, etc.).
- FLASH Area, Internal SRAM Code, SRAM Data, Special-Purpose, etc.
- Internal SRAM Bank Partition.
- Define “Blocks” for Specific Tasks
  - Runtime Stack, Heap Space, etc.
  - Size and Alignment.
- Specify Runtime “Initialization” Sections
  - Linker and C-Runtime Startup Collaboration.
  - Compress Code/Data for Expansion into Internal SRAM at Startup.
- Manage Code and Data “Placements” within Regions
  - Explicit Interrupt Vector Table (IVT) Placement.
  - Read-Only, Read-Write Attribute.
  - Special Section Handling.
  - Default Flash, Code and Data Placements.
  - Explicit Stack and Heap Block Placements.

### **3.1.2 J-Link Settings File**

The J-Link setting file (`JLinkSettings.ini`) is present in each example project folder, this file helps the J-Link debugger to retain the device configuration every time a debugger session is initiated. If this file is not present the user has to manually select the device in the J-Link settings.

### **3.1.3 Flash Programming Algorithms**

Flash Programming Algorithms are a piece of software to erase or download applications to the on-chip flash over the debug port (using the emulator). The application may then be executed directly from flash.

The flash algorithm is stored in the following path in the Pack Installation `<ADuCM4x50_root>/Flash`.

Upon successfully connecting to the device, the flash algorithm should be selected to download the image into the flash device (Refer to Section *J-Link/J-Trace - Setup and Connection*).

## **3.2 Keil Project Options**

### **3.2.1 Options for Target**

The *Target* tab in the projects *Options for Target...* allows the user to set various project configuration described in the following section (See *Figure 5. Target Options*).

**Figure 5. Target Options**



### 3.2.2 Device Options

The *Device* tab in the *Options for Target...* allows the user to choose the target processor variant for which the project is being built. This selection is very important because it drives a number of other project settings, such as selecting the correct flash downloader from the pack file (see *Figure 6. Device Selection*).

**Figure 6. Device Selection**



### 3.2.3 Output Settings

The *Output* tab in the *Options for Target...* allows the user to set the executable name or create a library file, and select the output file format that can be generated by the ARM CC compiler as shown in the *Figure 7. Output Settings*.

**Figure 7. Output Settings**



### 3.2.4 Listing

The *Listing* tab in the *Options for Target...* allows the user to select the Assembler listing and the linker map file for the Keil ARM CC compiler as shown in the *Figure 8. Listing*.

**Figure 8. Listing**



### 3.2.5 User Settings

The *User* tab in the *Options for Target...* allows the user to set any user command for a pre and post build for the Keil ARM CC compiler as shown in the *Figure 9. User Settings* below.

**Figure 9. User Settings**



### 3.2.6 C/C++ Settings

The *C/C++* tab in the *Options for Target...* allows the user to set any compiler flags, defines, include search paths, optimization levels into the build for the Keil ARM CC compiler as shown in the *Figure 10. C/C++ Settings* below.

**Figure 10. C/C++ Settings**



### 3.2.7 ASM Settings

The *ASM* tab in the *Options for Target...* allows the user to set any Assembler flags, defines, include search paths, PIP modes into the build for the Keil ARM CC compiler as shown in the *Figure 11. ASM Settings* below.

**Figure 11. ASM Settings**



### 3.2.8 Linker Settings

The *Linker* tab in the *Options for Target...* allows the user to set the path for the memory configuration file (SCT file) or to set custom memory configuration in the tab itself for the Keil ARM CC compiler as shown in the *Figure 12. Linker Settings* below.

**Figure 12. Linker Settings**



### 3.2.9 Debugger Settings

The *Debug* tab in the *Options for Target...* allows the user to configure the Segger J-Link and its parameters. See *Figure 13. Debugger Settings* below.

**Figure 13. Debugger Settings**



### 3.2.10 Debug Settings (J-Link/J-Trace Setup and Connection)

The *Settings* under the main *Debug* tab in the *Options for Target...* contains three sub settings (Debug, Trace, Flash Download) these allow the user to configure the Segger J-Link debugger and its modes (SWD/JTAG). *Figure 14. J-Link/J-Trace Settings* below shows the settings for a sample project.

**Figure 14. J-Link/J-Trace Settings**



Trace options can be set-up on a project basis too. See *Figure 15. Trace Settings* below.

**Figure 15. Trace Settings**



The Flash Algorithm File has to be added in the *Flash Download* tab on a per project basis. After adding the algorithm the config window will look as shown in the *Figure 16. Flash Download Settings*.

**Figure 16. Flash Download Settings**



### 3.2.11 Utilities

The *Utilities* tab in the *Options for Target...* allows the user to set debugger parameters and choose custom flash utilities (if any) as shown in the *Figure 17. Utilities* below.

**Figure 17. Utilities**



### 3.2.12 Manage Run-Time Environment

The software components can be managed by the *Manage Run-Time Environment* as *Figure 18. Manage Run-Time Environment*.

CMSIS-CORE and Startup as well as Global Configuration, which is required by Startup, should be added to the project manually in order to build successfully.

Retarget along with the UART, DMA and Power driver components are required to redirect output to UART so the output can be printed to console.

**Figure 18. Manage Run-Time Environment**



## 4 ADuCM4x50 System Overview

### 4.1 Block Diagram and Driver Layout

The Peripheral Device Drivers and System Services installed with this software package are used to configure and use various ADuCM4x50 on-chip peripherals. *Figure 19. Peripherals and Driver Source* illustrates the available peripherals and interconnects on the ADuCM4x50 processor and corresponding source files in the `<ADuCM4x50_root>/Source` directory.

In general, the driver sources are located in the `<ADuCM4x50_root>/Source` directory. The include file path `<ADuCM4x50_root>/Include` must also be specified in the project's compiler/preprocessor options (see section *C/C++ Settings* in *Installation Components* of this documentation).

**Figure 19. Peripherals and Driver Source**



## 4.2 Boot-Time CRC Validation

The ADuCM4x50 system reset interrupt vector is hard-coded on-chip to execute a built-in pre-boot kernel that performs a number of critical housekeeping tasks before executing the user provided reset vector. Some of those tasks include initializing the JTAG/Serial-Wire debug interface and validating flash integrity.

One of the primary tasks of the pre-boot kernel is to validate the integrity of the Flash0/Page0 region (first 2k of flash). This is done by comparing a pre-generated CRC code (embedded in the executable code image at build-time) against a boot-time-generated CRC value of Flash0/Page0 using the on-chip CRC hardware. The Page0 embedded CRC signature is stored at reserved location 0x0000007FC.

ADuCM4x50 Keil support does not currently compute and implant the CRC signature into the executable during the target build process (as a post-link build command). Therefore at boot time, when the kernel computes a run-time Flash0/Page0 CRC value using the on-chip CRC hardware and compares it against the value at the last CRC page, by default applications are built to omit the CRC check.

### 4.3 System Reset Strategy

All projecta Uui art thr Reses Stratelt the Reset t"normal"re orderet tenutablt themulatoret tgy

# 5 Application Configuration

Application initialization and configuration will vary depending on the chosen operating mode. The modes of operation include:

- Non-RTOS
  - The application is built without an RTOS.
- RTOS
  - The application is built with an RTOS. In this mode of operation the drivers can be RTOS-Aware or RTOS-Unaware.
  - RTOS-Aware Drivers
    - In this C-Macro controlled mode of operation the driver's source code will include the following features:
      - Interrupt Service Routines (ISR) with RTOS API calls used to potentially cause a task context switch.
      - Semaphores control communication between task-level code and ISR level code.
      - Mutexes control access to access to shared resources.
  - RTOS-Unaware Drivers.
    - In this C-Macro controlled mode of operation the driver's source code will not include the features listed above.

## 5.1 Application Initialization

The functions `SystemInit()` and `adi_initpinmux()` are used to initialize an application. `SystemInit()` is required to initialize the ARM Cortex CMSIS infrastructure and `adi_initpinmux()` initializes the peripheral pin multiplexing, if static pin multiplexing is used (see section *Static Pin Multiplexing*). *Figure 21. Application Initialization* below shows these functions being called from the user application `main()`.

**Figure 21. - Application Initialization**

```
int main()
{
    /* Initialize system (required) */

    SystemInit(); /* system_<Device>.c */
```

```

    /* Initialize Pin Multiplexing (optional) */

    adi_initpinmux(); /* Auto-generated source file using PinMux
Utility */

    ...

    return 0;
}

```

## 5.2 Static Pin Multiplexing

Included with the ADuCM4x50 DFP is a Pin Multiplexing application which is capable of generating code to set all the port MUX registers statically for all peripherals in a single call. The Pin Multiplexing application is a Java application locates in <ADuCM4x50\_root>/tools /PinMuxUI, which can be run from a command prompt:

There are two versions of the Java application included with the Board Support Package (a 32-bit and 64-bit version). You will need to use the correct `java.exe` executable to run the application.

- **32-bit version:**

```
java -jar PinMuxUI_1.0.0.x_x86.jar
```

To run the 32-bit PinMux Stand-alone Utility, you should use the `java.exe` that is installed in `C:/Program Files (x86)/Java/jre8/bin` (assuming that you have installed Java version 8).

- **64-bit version:**

```
java -jar PinMuxUI_1.0.0.x_x86_64.jar
```

To run the 64-bit PinMux Stand-alone Utility, you should use the `java.exe` that is installed in `C:/Program Files/Java/jre8/bin` (assuming that you have installed Java version 8).

After starting the application, you must first select the correct processor type from the top-right drop-down list-box. The ADuCM4x50 processors have two package variants: the BCBZ and BCPZ, each with different peripheral configurations. You are then able to select the desired peripherals to be enabled. The application will not allow conflicting peripherals to be selected. The Generate

Code button will create a C source file that sets the GPIO port configuration registers based on the peripherals and functions selected. The C source file should be manually added to your project. The C source file has a function `adi_initpinmux()` which can be called from the application source (see *Figure 22. Pin Multiplexing Application*) to set up the port MUX registers.

**Figure 22. Pin Multiplexing Application**



*Note: The pinmux code generator is the preferred method of configuring port multiplexing. It avoids multiple dynamic calls to each driver and allows all pin multiplexing to be done once, which reduces both footprint and run-time overhead.*

## 5.3 UART Baud Rate Configuration Utility

Included with the ADuCM4x50 DFP is a `UartDivCalculator` utility which computes the baudrate configuration values for a specified clock. This helps you to statically configure the baudrate. The utility can also provide the baudrate configuration values for a set of baudrates.

The output can be used in the UART baudrate configuration API, as

```
ADI_UART_RESULT adi_uart_ConfigBaudRate( ADI_UART_HANDLE const  
hDevice, uint16_t nDivC, uint8_t nDivM, uint16_t nDivN, uint8_t nOSR  
);
```

Where:

- **hDevice** is the device handle to UART device obtained when an UART device opened successfully
- **nDivC** is DIV-C in the output of `UartDivCalculator` utility
- **nDivM** is DIV-M in the output of `UartDivCalculator` utility
- **nDivN** is DIV-N in the output of `UartDivCalculator` utility
- **nOSR** is OSR in the output of `UartDivCalculator` utility

To produce the baudrate configuration values for a specified clock and the whole set of baudrates:

```
UartDivCalculator.exe input_clock
```

To produce the baudrate configuration values for a specific clock and a particular baudrate:

```
UartDivCalculator.exe input_clock baudrate
```

For example, to get the baudrate configuration values for input clock 16 MHz and baudrate 9600, run the following command line:

```
UartDivCalculator.exe 16000000 9600
```

and you will get

| CALCULATING UART DIV REGISTER VALUE FOR THE INPUT CLOCK: 16000000 |       |       |       |      |      |
|-------------------------------------------------------------------|-------|-------|-------|------|------|
| BAUDRATE                                                          | DIU-C | DIU-M | DIU-N | OSR  | DIFF |
| 00009600                                                          | 0014  | 0003  | 1475  | 0003 | 0000 |

## 5.4 Driver Include Files

The C/C++ tab in the Options for Target is used to define the additional include directories needed to build the project. The device drivers only require the following search paths

`<ADuCM4x50_root>/Include`

`<ARM\CMSIS_root>/CMSIS/Include`

to be added from the DFP installation. The required include directories, including the ones described above, are added to the Tool Settings automatically by the CMSIS components. See the *Installation Components - RTE Configuration* section in this document for more information. Applications may need to augment the preprocessor search path with their own requirements.

## 5.5 Driver Configuration

Most of the drivers are statically configurable. Their configuration is controlled via C/C++ pre-processor macros that are managed in a common area.

Static initialization is preferred, as it offers two advantages over dynamic (API) initializations:

1. It reduces the run-time driver startup time (and complexity) of initializing each driver through various driver configuration APIs.
2. It allows programmers to bypass most of the driver configuration APIs altogether, thereby allowing linker elimination to remove unused driver APIs, thereby reducing overall footprint.

### 5.5.1 Global Configuration

There is a single, global configuration file `<ADuCM4x50_root>/Include/config/adi_global_config.h`, that is added to your project's RTE directory automatically by adding the Global Configuration CMSIS component via the `system.rteconfig` file.

We recommend using the same approach for overriding the driver-specific configuration files as described below to override the global feature set-up. For example, to overwrite the RTOS feature, set the corresponding macro in `adi_global_config.h` as shown in *Figure 23. Global Configuration File Contents* below:

**Figure 23. Global Configuration File Contents**

```
/*! Configure the RTOS required across the project.  
It can be configured to one of the following macros:  
- #ADI_CFGRTOS_NO_OS  
- #ADI_CFGRTOS_MICRIUM_II  
- #ADI_CFGRTOS_MICRIUM_III  
- #ADI_CFGRTOS_FREERTOS  
*/  
#define ADI_CFGRTOS      ADI_CFGRTOS_NO_OS
```

## 5.5.2 Configuration Defaults

Two distinct types of configurations are managed in the driver configuration files: feature/function enable/disable (such as removing unneeded code for slave-mode operation, DMA support, etc.) and default values for the peripheral control registers. Each device driver uses these macros to control feature inclusion and set controller registers during driver initialization.

Factory default driver configuration files (one per driver) are located in the `<ADuCM4x50_root>/Include/config` directory, which you can edit for global changes or override them within your project, for localized changes.

It is recommended that the default configuration files are not modified, but overridden by modifying the local copies within your project found in your project's RTE directory. These copies are added by the CMSIS device driver components.

## 5.5.3 Configuration Overrides

In summary, there are two ways to override the default static configuration values:

1. Locally by editing the configuration files in your project's RTE directory.
2. Dynamically by using the dynamic APIs to modify the configuration at run time. The configuration APIs may be called at run-time to alter a driver's configuration. Static configuration is preferred, however, as it will save both footprint and run-time cycles.

Please note that a combination of static and dynamic configuration is possible.

## 5.5.4 IVT Table Location

The Cortex-M4 processor core allows the Interrupt Vector Table (IVT) to be relocated. In this release, we support a default placement of the IVT in ROM (FLASH) and allow it to be moved from ROM to RAM during system startup. The pre-processor macro RELOCATE\_IVT is used to enable IVT relocation.

The default, statically-linked IVT placement in ROM is preferred as it will avoid wasting RAM space and startup time to relocate the table. The static IVT cannot be used if the application needs to alter the IVT content.

Alternatively, the IVT may be dynamically relocated during system startup from ROM to RAM for applications that need to modify the IVT content. For example, by dynamically hooking/replacing interrupt handlers or running an RTOS that requires patching interrupt handlers through a common interrupt dispatcher. To support dynamic IVT relocation, add the RELOCATE\_IVT macro to the compiler pre-processor option tab. Doing so causes the IVT to be relocated during system reset (see startup.c: ResetISR() handler).

The default static IVT is always present in ROM and is optionally copied to RAM under control of the RELOCATE\_IVT macro. See relevant source code in system files startup.c and system.c (enclosed within the RELOCATE\_IVT macro) for implementation details of the relocated IVT memory allocation, relocation address and alignment attributes, physically copying the IVT and updating the *interrupt vector table offset register* (VTOR) within the Cortex-M4 core System Control Block (SCB). Once the IVT is copied and VTOR is written with the new address, the relocated interrupt vectors are active and can then be modified dynamically.

## 5.5.5 Interrupt Callbacks

In general, the device drivers take ownership of the various device interrupt handlers in order to drive communication protocols, manage DMA data pumping, capture events, etc. Most device drivers also offer application-level interrupt callbacks by giving the application an opportunity to receive event notifications or perform some application-level task related to device interrupts.

Application callbacks are optional. They may be an integral component of an event-based system or they may just tell the application when something happened. Application callbacks are always made in response to device interrupts and are *executed in context of the originating interrupt*.

To receive interrupt callbacks, the application defines a callback handler function and registers it with the device driver. The callback registration tells the device driver what application function call to make as it processes device interrupts. Each driver has unique event notifications which are passed back with the callback and describe what caused the interrupt. Some device drivers support event filtering that allows the application to specify a subset of events upon which to receive callbacks.

To use callbacks, the application defines a callback handler with the following prototype:

```
void cbHandler (void *pcbParam, uint32_t Event, void *pArg);
```

Where:

- **pcbParam** is an application-defined parameter that is given to the device driver as part of the callback registration,
- **Event** is a device-specific identifier describing the context of the callback, and
- **pArg** is an optional device-specific argument further qualifying the callback context (if needed).

The application will then call into the device driver callback registration API to register the callback, as:

```
ADI_xxx_RESULT_TYPE adi_xxx_RegisterCallback (ADI_xxx_DEV_HANDLE const  
t hDevice, ADI_CALLBACK const pfCallback, void *const pcbParam);
```

Where:

- **xxx** is the particular device driver,
- **hDevice** is the device driver handle,
- **ADI\_CALLBACK** is a typedef (see `adi_int.h`), describing the callback handler prototype (`cbHandler`, in this case),
- **pfCallback** is the function address of the application's callback handler (`cbHandler`), and
- **pcbParam** is an application-defined parameter that is passed back to the application when the application callback is dispatched. This parameter is used however the application dictates, it is simple passed back through the callback to the application by the device driver as-is. It may be used to differentiate device drivers (e.g., the device handle) if multiple drivers or driver instances are sharing a common application callback.

*Note: Application callbacks occur in context of the originating device interrupt, so extended processing at the application level will impact interrupt dispatching. Typically, extended application-level processing is done by some task after the callback is returned and the interrupt handler has exited.*

The ADuCM4x50 processors have two package variants: the BCBZ and BCPZ, each with different peripheral configurations.

# 6 Examples

The ADuCM4x50 Device Family Pack contains examples illustrating how to use the ADuCM4x50 device and software device drivers. Use these example programs to explore the hardware and recommended use of the device drivers. Each project has a Readme.txt file that describes any required switch settings, hardware setups, build/run steps, and expected outcome. Use a HyperTerminal session (or equivalent console application for hardware-based UART output) to capture example output.

The examples are located in the <ADuCM4x50\_root>/Boards/ADuCM4050-EZ-KIT directory. The device driver sources, include files, example projects, etc., are all device-specific and ported to the specific evaluation board, which are located under the <ADuCM4x50\_root> directory. The evaluation board is the only target on which the examples are supported.

## 6.1 Build and Run

Each example may be launched into the toolchain IDE by selecting the Pack Installer and choosing to copy the example project. See *Figure 24. Opening an Example*.

**Figure 24. Opening an Example**



You are prompted to copy the example and optionally to launch the Keil IDE.



Build the project and launch the debugger to download the executable to the target, check the example output using a HyperTerminal session (or equivalent console application for hardware-based UART output).



The example project release-mode build is configured with optimizations enabled and no debug information.

# 7 Device Driver API Documentation

Complete documentation for the DFP is listed in the references section, at the top of this document. Most of the documentation is provided in PDF format.

The API documentation for the device drivers is also available in HTML format as shown in *Figure 25. Device Driver Documentation*. The HTML documentation is located in the `<ADuCM4x50_root>/Docs/html` folder.

To open the HTML documentation, double-click on the `index.html` file.

**Figure 25. Device Driver Documentation**



## 7.1 Appendix

### 7.1.1 CMSIS

The ADuCM4x50 Device Family Pack is compliant with the Cortex Microcontroller Software Interface Standard (CMSIS). CMSIS prescribes a number of software organization aspects. One of the more convenient aspects of the CMSIS compliance is the availability of various CMSIS run-time library functions provided by the compiler vendor that implement many Cortex core access functions. These CMSIS access functions are used throughout the ADuCM4x50 DFP device driver implementation.

By wrapping up these Cortex core access functions into a compiler vendor library, the device drivers and application programmer are able to access the Cortex core implementation in a safe and reliable way. Examples of the CMSIS library access functions include functions to manage the NVIC (Nested Vectored Interrupt Controller) interrupt priority, priority grouping, interrupt enables, pending interrupts, active interrupts, etc.

Other CMSIS access functions include defining system startup, system clock and system timer functions, functions to access processor core registers, "intrinsic" functions to generate Cortex code that cannot be generated by ISO/IEC C, exclusive memory access functions, debug output functions for ITM messaging, etc. CMSIS also defines a number of naming conventions and various typedefs that are used throughout the ADuCM4x50 DFP.

Please consult *The Definitive Guide to ARM Cortex-M3 and Cortex-M4 Processors* mentioned in the *Introduction > References* section in this documentation or the [www.arm.com](http://www.arm.com) website for complete CMSIS details.

### 7.1.2 Interrupt Vector Table

The IVT is a 32-bit wide table containing mostly interrupt vectors. It consists of two regions:

- The first sixteen (16) locations contain exception handler addresses. The highest location of these addresses have fixed (pre-determined) priorities.
- The balance of the IVT contains peripheral interrupt handler addresses which are not considered exceptions. Each of the peripheral interrupts has an individually programmable interrupt priority and they are therefore sometimes referred to as "programmable" interrupts, in contrast to the non-programmable (fixed-priority) exception handlers.

The IVT is declared and initialized in the `startup_<Device>.s` file. The organization of the first 16 locations (0:15) of the IVT is prescribed for ARM Cortex M-class processors as follows:

- IVT[0] = Initial Main Stack Pointer Value (MSP register)

The very first 32-bit value contained in the IVT is not an interrupt handler address at all. It is used to convey an initial value for the processor's main stack pointer (MSP) to the system start code. It must point to a valid RAM area in which the various reset function calls may have a valid stacking area (C-Runtime Stack).

- IVT[1] = Hardware Reset Interrupt Vector

The second 32-bit value of the IVT is defined to hold the system reset vector. This is also defined in `startup_<Device>.c`. The location is initialized with the reset interrupt handler function. When the system starts up, it calls the function pointed to by this location (once the boot kernel is complete).

- IVT[2:15] = Non-Programmable System Exception Handlers

These locations contain various exception handlers, e.g., NMI, Hard Fault, Memory Manager Fault, Bus Fault, etc. All of these handlers are given weak default bindings within the startup.c file, insuring all exceptions have a safe "trapping" implementation.

- Balance of IVT Contains Interrupt Vectors for Programmable Interrupts

The remaining IVT entries are mapped by the manufacturer to the peripherals. In the case of the ADuCM4x50 processor, there are 72 (0-71) such peripheral interrupts. Each peripheral interrupt has a dedicated interrupt priority register that may be programmed at run-time to manage interrupt dispatching.

### 7.1.3 Startup\_<Device>.c Content

The <ADuCM4x50\_root>/Source/ARM/startup\_<Device>.s file is required for every ADuCM4x50 application. This file is largely defined by the CMSIS standard and contains:

- Stack and Heap set-up
- Interrupt Vector Table

### 7.1.4 System\_<Device>.c Content

The file <ADuCM4x50\_root>/Source/system\_<Device>.c is another CMSIS prescribed file implementing a number of required CMSIS APIs (SystemInit())

The system\_<Device>.c file is a required and integral component for every ADuCM4x50 application.

- SystemInit()

This is a prescribed CMSIS startup function which is called by Reset Handler.

The first and most critical task performed during SystemInit() is the activation of the (potentially) relocated IVT. Any IVT relocation is done during the system reset handler under control of the **RELOCATE\_IVT** macro. If the IVT has been moved, it must then be activated during SystemInit() by setting the Cortex core "Interrupt Vector Table Offset Register" in the Cortex Core System Control Block (SCB->VTOR) to the address of the new IVT.

Until the VTOR is reset, the default FLASH-based IVT remains active. The relocated IVT activation must be done *before* the application starts activating peripherals, but *after* the relocated IVT data has been copied.

Other important tasks performed during SystemInit() include bringing the clocks into a known state, configuring the PLL input source, and making the initial call to SystemCoreClockUpdate() (below), which must always be done (even by the application) after making any clock changes.

- SystemCoreClockUpdate()

This is another prescribed CMSIS API. The task performed here is to update the internal clock state variables within `system_<Device>.c` after making any clock changes. This insures that subsequent application calls to `SystemGetClockFrequency()` can return the correct frequency to device drivers attempting to configure themselves for serial BAUD rate, etc., or otherwise query the current system clock rate. `SystemCoreClockUpdate()` should always be called after any system clock changes.