

## Standalone (v 6.0)

## **Summary**

Standalone is the lowest layer of software modules used to access processor specific functions. Standalone is used when an application accesses board/processor features directly and is below the operating system layer.

This document contains the following sections:

- MicroBlaze Processor API
- Cortex A9 Processor API
- Cortex R5 Processor API
- Cortex A53 Processor API
- Xilinx Hardware Abstraction Layer
- Program Profiling
- Configuring the Standalone OS
- MicroBlaze MMU Example

## MicroBlaze Processor API

UG647 October 5, 2016

The following list is a summary of the MicroBlaze<sup>™</sup> processor API sections. You can click on a link to go directly to the function section.

- MicroBlaze Processor Interrupt Handling
- MicroBlaze Processor Exception Handling
- MicroBlaze Processor Instruction Cache Handling
- MicroBlaze Processor Data Cache Handling
- MicroBlaze Processor Fast Simplex Link (FSL) Interface Macros
- MicroBlaze Processor FSL Macro Flags
- MicroBlaze Processor Pseudo-asm Macro Summary
- MicroBlaze Processor Version Register (PVR) Access Routine and Macros
- MicroBlaze Processor File Handling
- MicroBlaze Processor Errno

## **MicroBlaze Processor Interrupt Handling**

The interrupt handling functions help manage interrupt handling on MicroBlaze processor devices. To use these functions, include the header file mb\_interface.hin your source code.

## **MicroBlaze Processor Interrupt Handling Function Descriptions**

void microblaze\_enable\_interrupts(void)

Enable interrupts on the MicroBlaze processor. When the MicroBlaze processor starts up, interrupts are disabled. Interrupts must be explicitly turned on using this function.

© 2013-2016 Xilinx, Inc. XILINX, the Xilinx logo, Virtex, Spartan, ISE, SDK, Vivado, and other designated brands included herein are trademarks of Xilinx in the United States and other countries. All other trademarks are the property of their respective owners.





## void microblaze\_disable\_interrupts(void)

Disable interrupts on the MicroBlaze processor. This function can be called when entering a critical section of code where a context switch is undesirable.

```
void microblaze_register_handler(XInterruptHandler
Handler, void *DataPtr)
```

Register the interrupt handler for the MicroBlaze processor. This handler is invoked in turn, by the first level interrupt handler that is present in Standalone.

The first level interrupt handler saves and restores registers, as necessary for interrupt handling, so that the function you register with this handler can be dedicated to the other aspects of interrupt handling, without the overhead of saving and restoring registers.

## **MicroBlaze Processor Exception Handling**

This section describes the exception handling functionality available on the MicroBlaze processor. This feature and the corresponding interfaces are not available on versions of the MicroBlaze processor older than v3.00.a.

**Note:** These functions work correctly only when the parameters that determine hardware exception handling are configured appropriately in the MicroBlaze Microprocessor Hardware Specification (MHS) hardware block. For example, you can register a handler for divide by zero exceptions only if hardware divide by zero exceptions are enabled on the MicroBlaze processor. Refer to the *MicroBlaze Processor Reference Guide (UG081)* for information on how to configure these cache parameters. A link to that document can be found in "MicroBlaze Processor API," page 1.

### **MicroBlaze Processor Exception Handler Function Descriptions**

The following functions help manage exceptions on the MicroBlaze processor. You must include the mb\_interface.h header file in your source code to use these functions.

```
void microblaze_disable_exceptions(void)
```

Disable hardware exceptions from the MicroBlaze processor. This routine clears the appropriate "exceptions enable" bit in the model-specific register (MSR) of the processor.

```
void microblaze_enable_exceptions(void)
```

Enable hardware exceptions from the MicroBlaze processor. This routine sets the appropriate "exceptions enable" bit in the MSR of the processor.

```
void microblaze_register_exception_handler(u8
```

ExceptionId, XExceptionHandler Handler, void \*DataPtr)

Register a handler for the specified exception type. Handler is the function that handles the specified exception.

DataPtr is a callback data value that is passed to the exception handler at run-time. By default the exception ID of the corresponding exception is passed to the handler.

Send Feedback

2



Table 1 describes the valid exception IDs, which are defined in the microblaze\_exceptions\_i.h file.

Table 1: Valid Exception IDs

| Exception ID                 | Value | Description                                                                                                 |
|------------------------------|-------|-------------------------------------------------------------------------------------------------------------|
| XEXC_ID_FSL                  | 0     | FSL bus exceptions.                                                                                         |
| XEXC_ID_UNALIGNED_ACCESS     | 1     | Unaligned access exceptions.                                                                                |
| XEXC_ID_ILLEGAL_OPCODE       | 2     | Exception due to an attempt to execute an illegal opcode.                                                   |
| XEXC_ID_M_AXI_I_EXCEPTION(1) | 3     | Exception due to a timeout from the Instruction side system bus.                                            |
| XEXC_ID_M_AXI_D_EXCEPTION(1) | 4     | Exception due to a timeout on the Data side system bus.                                                     |
| XEXC_ID_DIV_BY_ZERO          | 5     | Divide by zero exceptions from the hardware divide.                                                         |
| XEXC_ID_FPU                  | 6     | Exceptions from the floating point unit on the MicroBlaze processor.                                        |
|                              |       | <b>Note:</b> This exception is valid only on v4.0 and later versions of the MicroBlaze processor.           |
| XEXC_ID_MMU                  | 7     | Exceptions from the MicroBlaze processor MMU. All possible MMU exceptions are vectored to the same handler. |
|                              |       | <b>Note:</b> This exception is valid only on v7.00.a and later versions of the MicroBlaze processor.        |

By default, Standalone provides empty, no-op handlers for all the exceptions *except* unaligned exceptions. A default, fast, unaligned access exception handler is provided by Standalone.

An unaligned exception can be handled by making the corresponding aligned access to the appropriate bytes in memory. Unaligned access is transparently handled by the default handler. However, software that makes a significant amount of unaligned accesses will see the performance effects of this at run-time. This is because the software exception handler takes much longer to satisfy the unaligned access request as compared to an aligned access.

In some cases you might want to use the provision for unaligned exceptions to just trap the exception, and to be aware of what software is causing the exception. In this case, you should set breakpoints at the unaligned exception handler, to trap the dynamic occurrence of such an exception or register your own custom handler for unaligned exceptions.

**Note:** The lowest layer of exception handling, always provided by Standalone, stores volatile and temporary registers on the stack; consequently, your custom handlers for exceptions must take into consideration that the first level exception handler will have saved some state on the stack, before invoking your handler.

Nested exceptions are allowed by the MicroBlaze processor. The exception handler, in its prologue, re-enables exceptions. Thus, exceptions within exception handlers are allowed and handled. When the predecode\_fpu\_exceptions parameter is set to true, it causes the low-level exception handler to:

- Decode the faulting floating point instruction
- Determine the operand registers
- Store their values into two global variables

Send Feedback

3



You can register a handler for floating point exceptions and retrieve the values of the operands from the global variables. You can use the microblaze\_getfpex\_operand\_a() and microblaze\_getfpex\_operand\_b() macros.

**Note:** These macros return the operand values of the last floating point (FP) exception. If there are nested exceptions, you cannot retrieve the values of outer exceptions. An FP instruction might have one of the source registers being the same as the destination operand. In this case, the faulting instruction overwrites the input operand value and it is again irrecoverable.

## **MicroBlaze Processor Instruction Cache Handling**

The following functions help manage instruction caches on the MicroBlaze processor. You must include the  $xil\_cache.h$  header file in your source code to use these functions.

**Note:** These functions work correctly only when the parameters that determine the caching system are configured appropriately in the MicroBlaze Microprocessor Hardware Specification (MHS) hardware block. Refer to the *MicroBlaze Reference Guide (UG081)* for information on how to configure these cache parameters. "MicroBlaze Processor API," page 1 contains a link to this document.

### MicroBlaze Processor Instruction Cache Handling Function Descriptions

```
void Xil_ICacheEnable(void)
```

Enable the instruction cache on the MicroBlaze processor. When the MicroBlaze processor starts up, the instruction cache is disabled. The instruction cache must be explicitly turned on using this function.

### void Xil ICacheDisable(void)

Disable the instruction cache on the MicroBlaze processor.

### void Xil\_ICacheInvalidate()

Invalidate the instruction icache.

**Note:** For MicroBlaze processors prior to version v7.20.a, the cache and interrupts are disabled before invalidation starts and restored to their previous state after invalidation.

```
void Xil_ICacheInvalidateRange(unsigned int cache_addr,
    unsigned int cache_size)
```

Invalidate the specified range in the instruction icache. This function can be used for invalidating all or part of the instruction icache.

The parameter  $cache\_addr$  indicates the beginning of the cache location to be invalidated. The  $cache\_size$  represents the number of bytes from the  $cache\_addr$  to invalidate.

Note that *cache lines* are invalidated starting from the cache line to which cache\_addr belongs and ending at the cache line containing the address (cache\_addr + cache\_size - 1).

For example, Xil\_ICacheInvalidateRange (0x00000300, 0x100) invalidates the instruction cache region from 0x300 to 0x3ff (0x100 bytes of cache memory is cleared starting from 0x300).

If the L2 cache system (system cache) is present in the hardware system, this function invalidates relevant cache lines in the L2 cache as well. The invalidation starts with the L2 cache and moves to the L1 cache.

**Note:** For MicroBlaze processors prior to version v7.20.a: The cache and interrupts are disabled before invalidation starts and restored to their previous state after invalidation.





## MicroBlaze Processor Data Cache Handling

The following functions help manage data caches on the MicroBlaze processor. You must include the header file xil\_cache.h in your source code to use these functions.

**Note:** These functions work correctly only when the parameters that determine the caching system are configured appropriately in the MicroBlaze MHS hardware block. Refer to the *MicroBlaze Processor Reference Guide (UG081)* for information on how to configure these cache parameters. "MicroBlaze Processor API," page 1 contains a link to this document.

### **Data Cache Handling Functions**

### void Xil DCacheEnable(void)

Enable the data cache on the MicroBlaze processor. When the MicroBlaze processor starts up, the data cache is disabled. The data cache must be explicitly turned on using this function.

### void Xil DCache Disable(void)

Disable the data cache on the MicroBlaze processor. If writeback caches are enabled in the MicroBlaze processor hardware, this function also flushes the dirty data in the cache back to external memory and invalidates the cache. For write through caches, this function does not do any extra processing other than disabling the cache.

If the L2 cache system is present in the hardware, this function flushes the L2 cache before disabling the DCache.

### void Xil\_DCacheFlush()

Flush the entire data cache. This function can be used when write-back caches are turned on in the MicroBlaze processor hardware. Executing this function ensures that the dirty data in the cache is written back to external memory and the contents invalidated.

If the L2 cache system is present in the hardware, this function flushes the L2 cache first, before flushing the L1 cache.

```
void Xil_DCacheFlushRange(unsigned int cache_addr,
    unsigned int cache_len)
```

Flush the specified data cache range. This function can be used when write-back caches are enabled in the MicroBlaze processor hardware. Executing this function ensures that the dirty data in the cache range is written back to external memory and the contents of the cache range are invalidated. Note that *cache lines* will be flushed starting from the cache line to which cache\_addr belongs and ending at the cache line containing the address (cache\_addr + cache\_size - 1).

If the L2 cache system is present in the hardware, this function flushes the relevant L2 cache range first, before flushing the L1 cache range.

For example,  $Xil_DCacheFlushRange (0x00000300, 0x100)$  flushes the data cache region from 0x300 to 0x3ff (0x100 bytes of cache memory is flushed starting from 0x300).

5



### void Xil\_DCacheInvalidate()

Invalidate the data cache.

If the L2 cache system is present in the hardware, this function invalidates the L2 cache first, before invalidating the L1 cache.

**Note:** For MicroBlaze processors prior to version v7.20.a, the cache and interrupts are disabled before invalidation starts and restored to their previous state after invalidation.

```
void Xil_DCacheInvalidateRange(unsigned int cache_addr,
    unsigned int cache_size
```

Invalidate the data cache. This function can be used for invalidating all or part of the data cache. The parameter <code>cache\_addr</code> indicates the beginning of the cache location and <code>cache\_size</code> represents the size from <code>cache\_addr</code> to invalidate.

Note that cache lines will be invalidated starting from the cache line to which cache\_addr belongs and ending at the cache line containing the address ( $cache\_addr + cache\_size - 1$ ).

If the L2 cache system is present in the hardware, this function invalidates the relevant L2 cache range first, before invalidating the L1 cache range.

**Note:** For MicroBlaze processors prior to version v7.20.a, the cache and interrupts are disabled before invalidation starts and restored to their previous state after invalidation.

For example, Xil\_DCacheInvalidateRange (0x00000300, 0x100) invalidates the data cache region from 0x300 to 0x3ff (0x100 bytes of cache memory is cleared starting from 0x300).

## Software Sequence for Initializing Instruction and Data Caches

Typically, before using the cache, your program must perform a particular sequence of cache operations to ensure that invalid/dirty data in the cache is not being used by the processor. This would typically happen during repeated program downloads and executions.

The following example snippets show the necessary software sequence for initializing instruction and data caches in your program.

```
/* Initialize ICache *//
Xil_ICacheInvalidate ();
Xil_ICacheEnable ();
/* Initialize DCache */
Xil_DCacheInvalidate ();
Xil_DCacheEnable ();
```

At the end of your program, you should also put in a sequence similar to the example snippet below. This ensures that the cache and external memory are left in a valid and clean state.

```
/* Clean up DCache. For writeback caches, the disable_dcache routine
  internally does the flush and invalidate. For write through caches,
  an explicit invalidation must be performed on the entire cache. */

#if XPAR_MICROBLAZE_DCACHE_USE_WRITEBACK == 0

Xil_DCacheInvalidate ();
#endif

Xil_DCacheDisable ();

/* Clean up ICache */
Xil_ICacheInvalidate ();
Xil_ICacheDisable ();
```

www.xilinx.com



## MicroBlaze Processor Fast Simplex Link (FSL) Interface Macros

Standalone includes macros to provide convenient access to accelerators connected to the MicroBlaze Fast Simplex Link (FSL) Interfaces.

### MicroBlaze Processor Fast Simplex Link (FSL) Interface Macro Summary

The following is a list of the available macros. Click on a macro name to go to the description of the active macros.

| getfslx(val,id,flags)                   | putdfslx(val,id,flags)  |
|-----------------------------------------|-------------------------|
| putfslx(val,id,flags)                   | tgetdfslx(val,id,flags) |
| tgetfslx(val,id,flags)                  | tputdfslx(val,id,flags) |
| getd fslx(val,id,flags)                 | fsl_isinvalid(invalid)  |
| , , , , , , , , , , , , , , , , , , , , | fsl_iserror(error)      |

## **MicroBlaze Processor FSL Macro Descriptions**

The following macros provide access to all of the functionality of the MicroBlaze FSL feature in one simple and parameterized interface. Some capabilities are available on MicroBlaze v7.00.a and later only, as noted in the descriptions.

In the macro descriptions, val refers to a variable in your program that can be the source or sink of the FSL operation.

**Note:** id must be an integer *literal* in the basic versions of the macro (getfslx, putfslx, tgetfslx, tputfslx) and can be an integer literal or an integer variable in the dynamic versions of the macros (getdfslx, putdfslx, tgetdfslx, tputdfslx.)

You must include fsl.h in your source files to make these macros available.

### getfslx(val,id,flags)

Performs a get function on an input FSL of the MicroBlaze processor; id is the FSL identifier and is a literal in the range of 0 to 7 (0 to 15 for MicroBlaze v7.00.a and later). The semantics of the instruction is determined by the valid FSL macro flags, which are listed in Table 2, page 9.

### putfslx(val,id,flags)

Performs a put function on an input FSL of the MicroBlaze processor; id is the FSL identifier and is a literal in the range of 0 to 7 (0 to 15 for MicroBlaze processor v7.00.a and later).

The semantics of the instruction is determined by the valid FSL macro flags, which are listed in Table 2, page 9.

### tgetfslx(val,id,flags)

Performs a test get function on an input FSL of the MicroBlaze processor; id is the FSL identifier and is a literal in the ranging of 0 to 7 (0 to 15 for MicroBlaze v7.00.a and later). This macro can be used to test reading a single value from the FSL. The semantics of the instruction is determined by the valid FSL macro flags, which are listed in Table 2, page 9.

### tputfslx(val,id,flags)

Performs a put function on an input FSL of the MicroBlaze processor; id is the FSL identifier and is a literal in the range of 0 to 7 (0 to 15 for MicroBlaze processor v7.00.a and later). This macro can be used to test writing a single value to the FSL. The semantics of the put instruction is determined by the valid FSL macro flags, which are listed in Table 2, page 9.



### getd fslx(val,id,flags)

Performs a get function on an input FSL of the MicroBlaze processor; id is the FSL identifier and is an integer value or variable in the range of 0 to 15. The semantics of the instruction is determined by the valid FSL macro flags, which are listed in Table 2, page 9. This macro is available on MicroBlaze processor v7.00.a and later only.

### putdfslx(val,id,flags)

Performs a put function on an input FSL of the MicroBlaze processor; id is the FSL identifier and is an integer value or variable in the range of 0 to 15. The semantics of the instruction is determined by the valid FSL macro flags, which are listed in Table 2, page 9. This macro is available on MicroBlaze processor v7.00.a and later only.

### tgetdfslx(val,id,flags)

Performs a test get function on an input FSL of the MicroBlaze processor; id is the FSL identifier and is an integer or variable in the range of 0 to 15. This macro can be used to test reading a single value from the FSL. The semantics of the instruction is determined by the valid FSL macro flags, listed in Table 2, page 9. This macro is available on MicroBlaze processor v7.00.a and later only.

## tputdfslx(val,id,flags)

Performs a put function on an input FSL of the MicroBlaze processor; id is the FSL identifier and is an integer or variable in the range of 0 to 15. This macro can be used to test writing a single value to the FSL. The semantics of the instruction is determined by the valid FSL macro flags, listed in Table 2, page 9. This macro is available on MicroBlaze processor v7.00.a and later only.

### fsl\_isinvalid(invalid)

Checks if the last FSL operation returned valid data. This macro is applicable after invoking a non-blocking FSL put or get instruction. If there was no data on the FSL channel on a get, or if the FSL channel was full on a put, <code>invalid</code> is set to 1; otherwise, it is set to 0.

### fsl\_iserror(error)

This macro is used to check if the last FSL operation set an error flag. This macro is applicable after invoking a control FSL put or get instruction. If the control bit was set error is set to 1; otherwise, it is set to 0.

8



## **MicroBlaze Processor FSL Macro Flags**

Table 2 lists the available FSL Macro flags.

Table 2: FSL Macro Flags

| Flag                                         | Description                                                                                |
|----------------------------------------------|--------------------------------------------------------------------------------------------|
| FSL_DEFAULT                                  | Blocking semantics (on MicroBlaze processor v7.00.a and later this mode is interruptible). |
| FSL_NONBLOCKING                              | Non-blocking semantics. <sup>1</sup>                                                       |
| FSL_EXCEPTION                                | Generate exceptions on control bit mismatch. <sup>2</sup>                                  |
| FSL_CONTROL                                  | Control semantics.                                                                         |
| FSL_ATOMIC                                   | Atomic semantics. A sequence of FSL instructions cannot be interrupted.                    |
| FSL_NONBLOCKING_EXCEPTION                    | Combines non-blocking and exception semantics.                                             |
| FSL_NONBLOCKING_CONTROL                      | Combines non-blocking and control semantics.                                               |
| FSL_NONBLOCKING_ATOMIC                       | Combines non-blocking and atomic semantics.                                                |
| FSL_EXCEPTION_CONTROL                        | Combines exception and control semantics.                                                  |
| FSL_EXCEPTION_ATOMIC                         | Combines exception and atomic semantics.                                                   |
| FSL_CONTROL_ATOMIC                           | Combines control and atomic semantics.                                                     |
| FSL_NONBLOCKING_EXCEPTION_<br>CONTROL        | Combines non-blocking, exception, and control semantics. <sup>2</sup>                      |
| FSL_NONBLOCKING_EXCEPTION_<br>ATOMIC         | Combines non-blocking, exception, and atomic semantics.                                    |
| FSL_NONBLOCKING_CONTROL_<br>ATOMIC           | Combines non-blocking, atomic, and control semantics.                                      |
| FSL_EXCEPTION_CONTROL_<br>ATOMIC             | Combines exception, atomic, and control semantics.                                         |
| FSL_NONBLOCKING_EXCEPTION_<br>CONTROL_ATOMIC | Combines non-blocking, exception, control, and atomic semantics.                           |

<sup>1.</sup> When non-blocking semantics are not applied, blocking semantics are implied.

### Deprecated MicroBlaze Processor Fast Simplex Link (FSL) Macros

The following macros are deprecated:

getfsl(val,id)(deprecated)

Performs a blocking data get function on an input FSL of the MicroBlaze processor; id is the FSL identifier in the range of 0 to 7. This macro is uninterruptible.

putfsl(val,id)(deprecated)

Performs a blocking data put function on an output FSL of the MicroBlaze processor; id is the FSL identifier in the range of 0 to 7. This macro is uninterruptible.

<sup>2.</sup> This combination of flags is available only on MicroBlaze processor v7.00.a and later versions.



## ngetfsl(val,id)(deprecated)

Performs a non-blocking data get function on an input FSL of the MicroBlaze processor; id is the FSL identifier in the range of 0 to 7.

### nputfsl(val,id)(deprecated)

Performs a non-blocking data put function on an output FSL of the MicroBlaze processor; id is the FSL identifier in the range of 0 to 7.

### **cgetfsl**(val, id)(deprecated)

Performs a blocking control get function on an input FSL of the MicroBlaze processor; *id* is the FSL identifier in the range of 0 to 7. This macro is uninterruptible.

### cputfsl(val, id)(deprecated)

Performs a blocking control put function on an output FSL of the MicroBlaze processor; id is the FSL identifier in the range of 0 to 7. This macro is uninterruptible.

### ncgetfsl(val, id)(deprecated)

Performs a non-blocking control get function on an input FSL of the MicroBlaze processor; id is the FSL identifier in the range of 0 to 7.

### ncputfsl(val, id)(deprecated)

Performs a non-blocking control put function on an output FSL of the MicroBlaze processor; *id* is the FSL identifier in the range of 0 to 7.

### **getfsl\_interruptible**(*val*, *id*)(deprecated)

Performs repeated non-blocking data get operations on an input FSL of the MicroBlaze processor until valid data is actually fetched; id is the FSL identifier in the range of 0 to 7. Because the FSL access is non-blocking, interrupts will be serviced by the processor.

## putfsl\_interruptible(val, id)(deprecated)

Performs repeated non-blocking data put operations on an output FSL of the MicroBlaze processor until valid data is sent out; id is the FSL identifier in the range of 0 to 7. Because the FSL access is non-blocking, interrupts will be serviced by the processor.

UG647 October 5, 2016 www.xilinx.com Send Feedback



```
cgetfsl_interruptible(val, id)(deprecated)
```

Performs repeated non-blocking control get operations on an input FSL of the MicroBlaze processor until valid data is actually fetched; id is the FSL identifier in the range of 0 to 7. Because the FSL access is non-blocking, interrupts are serviced by the processor.

```
cputfsl_interruptible(val, id)(deprecated)
```

Performs repeated non-blocking control put operations on an output FSL of the MicroBlaze processor until valid data is sent out; id is the FSL identifier in the range of 0 to 7. Because the FSL access is non-blocking, interrupts are serviced by the processor.

### **MicroBlaze Processor Pseudo-asm Macros**

Standalone includes macros to provide convenient access to various registers in the MicroBlaze processor. Some of these macros are very useful within exception handlers for retrieving information about the exception. To use these macros, you must include the mb\_interface.h header file in your source code.

### MicroBlaze Processor Pseudo-asm Macro Summary

The following is a summary of the MicroBlaze processor pseudo-asm macros. Click on the macro name to go to the description.

```
mfgpr(rn)
mfmsr()
mfesr()
mfear()
mffsr()
mtmsr(v)
mtgpr(rn,v)
microblaze_getfpex_operand_a()
microblaze_getfpex_operand_b()
clz(v)
mbar(mask)
mb_swapb(v)
mb_swaph(v)
mb_sleep
```

mfear( )

### MicroBlaze Processor Pseudo-asm Macro Descriptions

```
mfgpr(rn)
Return value from the general purpose register (GPR) rn.

mfmsr()
Return the current value of the MSR.

mfesr()
Return the current value of the Exception Status Register (ESR).
```

Return the current value of the Exception Address Register (EAR).



### mffsr()

Return the current value of the Floating Point Status (FPS).

### $\mathtt{mtmsr}(v)$

Move the value v to MSR.

### mtgpr(rn, v)

Move the value v to GPR rn.

### microblaze\_getfpex\_operand\_a( )

Return the saved value of operand A of the last faulting floating point instruction.

### microblaze\_getfpex\_operand\_b( )

Return the saved value of operand B of the last faulting floating point instruction.

**Note:** Because of the way some of these macros have been written, they cannot be used as parameters to function calls and other such constructs.

### clz(v)

Counts the number of leading zeros in the data specified by v

### mbar (mask)

This instruction ensures that outstanding memory accesses on memory interfaces are completed before any subsequent instructions are executed. mask value of 1 specifies data side barrier, mask value of 2 specifies instruction side barrier and mask value of 16 specifies to put the processor in sleep.

### mb\_swapb(v)

Swaps the bytes in the data specified bv v. This converts the bytes in the data from little endian to big endian or vice versa. So v contains a value of 0x12345678, the macro will return a value of 0x78563412.

### $mb_swaph(v)$

Swaps the half words in the data specified by v. So if v has a value of 0x12345678, the macro will return a value of 0x56781234.

### mb\_sleep

Puts the processor in sleep.



# MicroBlaze Processor Version Register (PVR) Access Routine and Macros

MicroBlaze processor v5.00.a and later versions have configurable Processor Version Registers (PVRs). The contents of the PVR are captured using the  $pvr_t$  data structure, which is defined as an array of 32-bit words, with each word corresponding to a PVR register on hardware. The number of PVR words is determined by the number of PVRs configured in the hardware. You should not attempt to access PVR registers that are not present in hardware, as the  $pvr_t$  data structure is resized to hold only as many PVRs as are present in hardware.

To access information in the PVR:

- Use the microblaze\_get\_pvr() function to populate the PVR data into a pvr\_t data structure.
- 2. In subsequent steps, you can use any one of the PVR access macros list to get individual data stored in the PVR.

**Note:** The PVR access macros take a parameter, which must be of type  $pvr_t$ .

### **PVR Access Routine**

The following routine is used to access the PVR. You must include pvr.h file to make this routine available.

### int microblaze\_get\_pvr(pvr\_t \*pvr)

Populate the PVR data structure to which pvr points with the values of the hardware PVR registers. This routine populates only as many PVRs as are present in hardware and the rest are zeroed. This routine is not available if C PVR is set to NONE in hardware.

### **PVR Macros**

The following processor macros are used to access the PVR. You must include pvr.h file to make these macros available.

Table 3 lists the MicroBlaze processor PVR macros and descriptions.

Table 3: PVR Access Macros

| Macro                          | Description                                                                                             |
|--------------------------------|---------------------------------------------------------------------------------------------------------|
| MICROBLAZE_PVR_IS_FULL(pvr)    | Return non-zero integer if PVR is of type FULL, 0 if basic.                                             |
| MICROBLAZE_PVR_USE_BARREL(pvr) | Return non-zero integer if hardware barrel shifter present.                                             |
| MICROBLAZE_PVR_USE_DIV(pvr)    | Return non-zero integer if hardware divider present.                                                    |
| MICROBLAZE_PVR_USE_HW_MUL(pvr) | Return non-zero integer if hardware multiplier present.                                                 |
| MICROBLAZE_PVR_USE_FPU(pvr)    | Return non-zero integer if hardware floating point unit (FPU) present.                                  |
| MICROBLAZE_PVR_USE_FPU2(pvr)   | Return non-zero integer if hardware floating point conversion and square root instructions are present. |
| MICROBLAZE_PVR_USE_ICACHE(pvr) | Return non-zero integer if I-cache present.                                                             |
| MICROBLAZE_PVR_USE_DCACHE(pvr) | Return non-zero integer if D-cache present.                                                             |



Table 3: PVR Access Macros (Cont'd)

| Масго                                        | Description                                                                                                                                                                                                                                  |
|----------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| MICROBLAZE_PVR_MICROBLAZE_VERSION (pvr)      | Return MicroBlaze processor version encoding. Refer to the <i>MicroBlaze Processor Reference Guide (UG081)</i> for mappings from encodings to actual hardware versions. "MicroBlaze Processor API," page 1 contains a link to this document. |
| MICROBLAZE_PVR_USER1(pvr)                    | Return the USER1 field stored in the PVR.                                                                                                                                                                                                    |
| MICROBLAZE_PVR_USER2(pvr)                    | Return the USER2 field stored in the PVR.                                                                                                                                                                                                    |
| MICROBLAZE_PVR_INTERCONNECT(pvr)             | Return non-zero if MicroBlaze processor has PLB interconnect; otherwise return zero.                                                                                                                                                         |
| MICROBLAZE_PVR_D_PLB(pvr)                    | Return non-zero integer if Data Side PLB interface is present.                                                                                                                                                                               |
| MICROBLAZE_PVR_D_OPB(pvr)                    | Return non-zero integer if Data Side On-chip<br>Peripheral Bus (OPB) interface present.                                                                                                                                                      |
| MICROBLAZE_PVR_D_LMB(pvr)                    | Return non-zero integer if Data Side Local Memory Bus (LMB) interface present.                                                                                                                                                               |
| MICROBLAZE_PVR_I_PLB(pvr)                    | Return non-zero integer if Instruction Side PLB interface is present.                                                                                                                                                                        |
| MICROBLAZE_PVR_I_OPB(pvr)                    | Return non-zero integer if Instruction side OPB interface present.                                                                                                                                                                           |
| MICROBLAZE_PVR_I_LMB(pvr)                    | Return non-zero integer if Instruction side LMB interface present.                                                                                                                                                                           |
| MICROBLAZE_PVR_INTERRUPT_IS_EDGE (pvr)       | Return non-zero integer if interrupts are configured as edge-triggered.                                                                                                                                                                      |
| MICROBLAZE_PVR_EDGE_IS_POSITIVE (pvr)        | Return non-zero integer if interrupts are configured as positive edge triggered.                                                                                                                                                             |
| MICROBLAZE_PVR_USE_MUL64(pvr)                | Return non-zero integer if MicroBlaze processor supports 64-bit products for multiplies.                                                                                                                                                     |
| MICROBLAZE_PVR_OPCODE_OxO_ILLEGAL (pvr)      | Return non-zero integer if opcode 0x0 is treated as an illegal opcode.                                                                                                                                                                       |
| MICROBLAZE_PVR_UNALIGNED_EXCEPTION (pvr)     | Return non-zero integer if unaligned exceptions are supported.                                                                                                                                                                               |
| MICROBLAZE_PVR_ILL_OPCODE_<br>EXCEPTION(pvr) | Return non-zero integer if illegal opcode exceptions are supported.                                                                                                                                                                          |
| MICROBLAZE_PVR_IOPB_EXCEPTION(pvr)           | Return non-zero integer if I-OPB exceptions are supported.                                                                                                                                                                                   |
| MICROBLAZE_PVR_DOPB_EXCEPTION(pvr)           | Return non-zero integer if D-OPB exceptions are supported.                                                                                                                                                                                   |
| MICROBLAZE_PVR_IPLB_EXCEPTION(pvr)           | Return non-zero integer if I-PLB exceptions are supported.                                                                                                                                                                                   |
| MICROBLAZE_PVR_DPLB_EXCEPTION(pvr)           | Return non-zero integer if D-PLB exceptions are supported.                                                                                                                                                                                   |
| MICROBLAZE_PVR_DIV_ZERO_EXCEPTION (pvr)      | Return non-zero integer if divide by zero exceptions are supported.                                                                                                                                                                          |



Table 3: PVR Access Macros (Cont'd)

| Macro                                        | Description                                                        |
|----------------------------------------------|--------------------------------------------------------------------|
| MICROBLAZE_PVR_FPU_EXCEPTION(pvr)            | Return non-zero integer if FPU exceptions are supported.           |
| MICROBLAZE_PVR_FSL_EXCEPTION(pvr)            | Return non-zero integer if FSL exceptions are present.             |
| MICROBLAZE_PVR_DEBUG_ENABLED(pvr)            | Return non-zero integer if debug is enabled.                       |
| MICROBLAZE_PVR_NUM_PC_BRK(pvr)               | Return the number of hardware PC breakpoints available.            |
| MICROBLAZE_PVR_NUM_RD_ADDR_BRK (pvr)         | Return the number of read address hardware watchpoints supported.  |
| MICROBLAZE_PVR_NUM_WR_ADDR_BRK (pvr)         | Return the number of write address hardware watchpoints supported. |
| MICROBLAZE_PVR_FSL_LINKS(pvr)                | Return the number of FSL links present.                            |
| MICROBLAZE_PVR_ICACHE_BASEADDR (pvr)         | Return the base address of the I-cache.                            |
| MICROBLAZE_PVR_ICACHE_HIGHADDR (pvr)         | Return the high address of the I-cache.                            |
| MICROBLAZE_PVR_ICACHE_ADDR_TAG_<br>BITS(pvr) | Return the number of address tag bits for the I-cache.             |
| MICROBLAZE_PVR_ICACHE_USE_FSL(pvr)           | Return non-zero if I-cache uses FSL links.                         |
| MICROBLAZE_PVR_ICACHE_ALLOW_WR (pvr)         | Return non-zero if writes to I-caches are allowed.                 |
| MICROBLAZE_PVR_ICACHE_LINE_LEN (pvr)         | Return the length of each I-cache line in bytes.                   |
| MICROBLAZE_PVR_ICACHE_BYTE_SIZE (pvr)        | Return the size of the D-cache in bytes.                           |
| MICROBLAZE_PVR_DCACHE_BASEADDR (pvr)         | Return the base address of the D-cache.                            |
| MICROBLAZE_PVR_DCACHE_HIGHADDR (pvr)         | Return the high address of the D-cache.                            |
| MICROBLAZE_PVR_DCACHE_ADDR_TAG_<br>BITS(pvr) | Return the number of address tag bits for the D-cache.             |
| MICROBLAZE_PVR_DCACHE_USE_FSL(pvr)           | Return non-zero if the D-cache uses FSL links.                     |
| MICROBLAZE_PVR_DCACHE_ALLOW_WR (pvr)         | Return non-zero if writes to D-cache are allowed.                  |
| MICROBLAZE_PVR_DCACHE_LINE_LEN (pvr)         | Return the length of each line in the D-cache in bytes.            |
| MICROBLAZE_PVR_DCACHE_BYTE_SIZE (pvr)        | Return the size of the D-cache in bytes.                           |
| MICROBLAZE_PVR_TARGET_FAMILY(pvr)            | Return the encoded target family identifier.                       |



Table 3: PVR Access Macros (Cont'd)

| Macro                          | Description                                                                                                                                                                                                              |
|--------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| MICROBLAZE_PVR_MSR_RESET_VALUE | Refer to the <i>MicroBlaze Processor Reference Guide (UG081)</i> for mappings from encodings to target family name strings. "MicroBlaze Processor API," page 1 contains a link to this document.                         |
| MICROBLAZE_PVR_MMU_TYPE(pvr)   | Returns the value of C_USE_MMU. Refer to the <i>MicroBlaze Processor Reference Guide (UG081)</i> for mappings from MMU type values to MMU function. "MicroBlaze Processor API," page 1 contains a link to this document. |

## MicroBlaze Processor File Handling

The following routine is included for file handling:

```
int fcntl(int fd, int cmd, long arg);
```

A dummy implementation of fcntl(), which always returns 0, is provided. fcntl is intended to manipulate file descriptors according to the command specified by cmd. Because Standalone does not provide a file system, this function is included for completeness only.

### MicroBlaze Processor Errno

The following routine provides the error number value:

int errno( );

Return the global value of errno as set by the last C library call.

16



# Cortex A9 Processor API

Standalone BSP contains boot code, cache, exception handling, file and memory management, configuration, time and processor-specific include functions. It supports gcc compilers.

The following lists the Cortex A9 Processor API sections. You can click on a link to go directly to the function section.

- Cortex A9 Processor Boot Code
- Cortex A9 Processor Cache Functions
- Cortex A9 Processor Exception Handling
- Cortex A9 Processor File Support
- Cortex A9 gcc Errno Function
- Cortex A9 gcc Memory Management
- Cortex A9 gcc Process Functions
- Cortex A9 Processor-Specific Include Files
- Cortex A9 Time Functions

The following subsections describe the functions by type.

### **Cortex A9 Processor Boot Code**

The boot.S file contains a minimal set of code for transferring control from the processor's reset location to the start of the application. It performs the following tasks.

- Invalidate L1 caches, TLBs, Branch Predictor Array, etc.
- Invalidate L2 caches and initialize L2 Cache Controller
- Enable caches and MMU
- Load MMU translation table base address into the TTB registers
- Enable NEON coprocessor

The boot code also starts the Cycle Counter and initializes the Static Memory Controller.

### **Cortex A9 Processor Cache Functions**

The xil\_cache.c file and the corresponding xil\_cache.h header file provide access to the following cache and cache-related operations.

### Cache Function Summary

The following are links to the function descriptions. Click on the name to go to that function.

void Xil\_DCacheEnable(void)

void Xil\_DCacheInvalidate(void)

void Xil\_DCacheInvalidateLine(unsigned int adr)

void Xil\_DCacheInvalidateRange(unsigned int adr, unsigned len)

void Xil\_DCacheFlush(void)

void Xil\_DCacheFlushLine(unsigned int adr)

void Xil\_DCacheFlushRange(unsigned int adr, unsigned len)

void Xil\_DCacheStoreLine(unsigned int adr)

void Xil\_ICacheEnable(void)

void Xil\_ICacheDisable(void)



void Xil\_ICacheInvalidate(void)

void Xil\_ICacheInvalidateLine(unsigned int adr)

void Xil\_ICacheInvalidateRange(unsigned int adr, unsigned len)

void Xil\_L1DCacheEnable(void)

void Xil\_L1DCacheDisable(void)

void Xil\_L1DCacheInvalidate(void)

void Xil\_L1DCacheInvalidateLine(unsigned int adr)

void Xil\_L2CacheInvalidateRange(unsigned int adr, unsigned len)

void Xil\_L1DCacheFlush(void)

void Xil\_L1DCacheFlushLine(unsigned int adr)

void Xil\_L1DCacheFlushRange(unsigned int adr, unsigned len)

void Xil\_L1DCacheStoreLine(unsigned int adr)

void Xil\_L1ICacheEnable(void)

void Xil\_ICacheDisable(void)

void Xil\_ICacheInvalidate(void)

void Xil\_L1lCacheInvalidateLine(unsigned int adr)

void Xil\_L1ICacheInvalidateRange(unsigned int adr, unsigned len)

void Xil\_L2CacheEnable(void)

void Xil\_L2CacheDisable(void)

void Xil\_L2CacheInvalidate(void)

void Xil\_L2CacheInvalidateLine(unsigned int adr)

void Xil\_L2CacheInvalidateRange(unsigned int adr, unsigned len)

void Xil\_L2CacheFlush(void)

void Xil\_L2CacheFlushLine(unsigned int adr)

void Xil\_L2CacheFlushRange(unsigned int adr, unsigned len)

void Xil\_L2CacheStoreLine(unsigned int adr)

### **Cache Function Descriptions**

### void Xil\_DCacheEnable(void)

Enable the data caches.

### void Xil DCacheInvalidate(void)

Invalidate the entire data cache.

### void Xil\_DCacheInvalidateLine(unsigned int adr)

Invalidate a data cache line. If the byte specified by adr is cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated. A subsequent data access to this address results in a cache miss and a cache line refill.

UG647 October 5, 2016 www.xilinx.com Send Feedback



void Xil\_DCacheInvalidateRange(unsigned int adr, unsigned
len)

Invalidates the data cache lines that are described by the address range starting from adr and len bytes long. A subsequent data access to any address in this range results in a cache miss and a cache line refill.

### void Xil\_DCacheFlush(void)

Flush the entire Data cache.

## void Xil\_DCacheFlushLine(unsigned int adr)

Flush a Data cache line. If the byte specified by the address (adr) is cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the entire contents of the cacheline are written to system memory before the line is invalidated. A subsequent data access to this address results in a cache miss and a cache line refill.

### void Xil\_DCacheFlushRange(unsigned int adr, unsigned len)

Flushes the data cache lines that are described by the address range starting from adr and len bytes long. A subsequent data access to any address in this range results in a cache miss and a cache line refill.

### void Xil\_DCacheStoreLine(unsigned int adr)

Store a Data cache line. If the byte specified by the adr is cached by the data cache and the cacheline is modified (dirty), the entire contents of the cacheline are written to system memory. After the store completes, the cacheline is marked as unmodified (not dirty).

### void Xil ICacheEnable(void)

Enable the instruction caches.

### void Xil\_ICacheDisable(void)

Disable the instruction caches.

### void Xil\_ICacheInvalidate(void)

Invalidate the entire instruction cache.

### void Xil\_ICacheInvalidateLine(unsigned int adr)

Invalidate an instruction cache line. If the instruction specified by the parameter adr is cached by the instruction cache, the cacheline containing that instruction is invalidated.

UG647 October 5, 2016 <u>www.xilinx.com</u> Send Feedback



void Xil\_ICacheInvalidateRange(unsigned int adr, unsigned
len)

Invalidate the instruction cache for the given address range. If the bytes specified by the adr are cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated.

### void Xil\_L1DCacheEnable(void)

Enable the level 1 data cache.

### void Xil L1DCacheDisable(void)

Disable the level 1 data cache.

### void Xil\_L1DCacheInvalidate(void)

Invalidate the level 1 data cache.

### void Xil\_L1DCacheInvalidateLine(unsigned int adr)

Invalidate a level 1 data cache line. If the byte specified by the adr is cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated.

## void Xil\_L1DCacheInvalidateRange(unsigned int adr, unsigned len)

Invalidate the level 1 data cache for the given address range. If the bytes specified by the adr are cached by the sata cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated.

### void Xil\_L1DCacheFlush(void)

Flush the level 1 data cache.

### void Xil\_L1DCacheFlushLine(unsigned int adr)

Flush a level 1 data cache line. If the byte specified by the adr is cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the entire contents of the cacheline are written to system memory before the line is invalidated.

## void Xil\_L1DCacheFlushRange(unsigned int adr, unsigned len)

Flush the level 1 data cache for the given address range. If the bytes specified by the adr are cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the written to system memory first before the before the line is invalidated.

UG647 October 5, 2016 <a href="https://www.xilinx.com">www.xilinx.com</a> Send Feedback 20



### void Xil\_L1DCacheStoreLine(unsigned int adr)

Store a level 1 data cache line. If the byte specified by the adr is cached by the data cache and the cacheline is modified (dirty), the entire contents of the cacheline are written to system memory. After the store completes, the cacheline is marked as unmodified (not dirty).

### void Xil\_L1ICacheEnable(void)

Enable the level 1 instruction cache.

### void Xil\_L1ICacheDisable(void)

Disable level 1 the instruction cache.

### void Xil\_L1ICacheInvalidate(void)

Invalidate the entire level 1 instruction cache.

### void Xil\_L1ICacheInvalidateLine(unsigned int adr)

Invalidate a level 1 instruction cache line. If the instruction specified by the parameter adr is cached by the instruction cache, the cacheline containing that instruction is invalidated.

## void Xil\_L1ICacheInvalidateRange(unsigned int adr, unsigned len)

Invalidate the level 1 instruction cache for the given address range. If the bytes specified by the adr are cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated.

### void Xil L2CacheEnable(void)

Enable the L2 cache.

### void Xil\_L2CacheDisable(void)

Disable the L2 cache.

### void Xil L2CacheInvalidate(void)

Invalidate the L2 cache. If the byte specified by the adr is cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated.

### void Xil\_L2CacheInvalidateLine(unsigned int adr)

Invalidate a level 2 cache line. If the byte specified by the adr is cached by the Data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated.



## void Xil\_L2CacheInvalidateRange(unsigned int adr, unsigned len)

Invalidate the level 2 cache for the given address range. If the bytes specified by the adr are cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated.

## void Xil\_L2CacheFlush(void)

Flush the L2 cache. If the byte specified by the adr is cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the entire contents of the cacheline are written to system memory before the line is invalidated.

### void Xil\_L2CacheFlushLine(unsigned int adr)

Flush a level 1 cache line. If the byte specified by the adr is cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the entire contents of the cacheline are written to system memory before the line is invalidated.

## void Xil\_L2CacheFlushRange(unsigned int adr, unsigned len)

Flush the level 2 cache for the given address range. If the bytes specified by the adr are cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the written to system memory first before the before the line is invalidated.

### void Xil\_L2CacheStoreLine(unsigned int adr)

Store a level 2 cache line. If the byte specified by the adr is cached by the data cache and the cacheline is modified (dirty), the entire contents of the cacheline are written to system memory. After the store completes, the cacheline is marked as unmodified (not dirty).

### void XL2cc\_EventCtrInit(int Event0, int Event1)

This function initializes the event counters in L2 Cache controller with a set of event codes specified by the user. Use the event codes defined by XL2CC\_\* in xl2cc\_counter.h to specify the events Event0 and Event1.

### void XL2cc EventCtrStart(void)

This function starts the event counters in L2 Cache controller.

### void XL2cc\_EventCtrStop(u32 \*EveCtr0, u32 \*EveCtr1)

This function disables the event counters in L2 Cache controller, saves the counter values to address pointed to by EveCtr0 and EveCtr1 and resets the counters.

UG647 October 5, 2016 <u>www.xilinx.com</u> Send Feedback



## Cortex A9 Processor MMU Handling

The standalone BSP MMU handling API is implemented in file xil\_mmu.c and the corresponding header file xil\_mmu.h.

### **MMU Handling Function Summary**

The following function describes the available MMU handling API.

```
void Xil_SetTlbAttributes(u32 addr, u32 attrib)
```

This function changes the MMU attribute of the 1 MB address range in which the passed memory address "addr" falls.

The new MMU attribute is passed as an argument "attrib" to this API.

This API can be used to change attributes such as cache-ability and share-ability of a specified memory region.

## **Cortex A9 Processor Exception Handling**

The Standalone BSP provides an exception handling API. For details about the exceptions and interrupts on ARM Cortex-A9 processor, refer to "Exceptions" under the chapter "The System Level Programmers' Model" in the ARM Architecture Reference Manual ARMv7-A and ARMv-7R edition.

The exception handling API is implemented in a set of the files - asm\_vectors.S, vectors.c, xil\_exception.c, and the corresponding header files vectors.h and xil\_exception.h.

### **Exception Handling Function Summary**

The following are links to the function descriptions. Click on the name to go to that function.

void Xil ExceptionInit(void)

void Xil\_ExceptionRegisterHandler (u8 ExceptionId, XExceptionHandler Handler, void \*DataPtr)

void Xil\_ExceptionRemoveHandler (u8 ExceptionId)

void Xil ExceptionEnableMask(Mask)

void Xil ExceptionEnable(void)

void Xil\_ExceptionDisableMask(Mask)

void Xil\_ExceptionDisable(void)

## **Exception Handling Function Descriptions**

### void Xil\_ExceptionInit(void)

Sets up the interrupt vector table and registers a "do nothing" function for each exception. This function has no parameters and does not return a value. This function must be called before registering any exception handlers or enabling any interrupts.



Registers an exception handler for a specific exception; does not return a value. Refer to Table 1, for a list of exception types and their values.

The parameters are:

- **ExceptionId** is of parameter type u8, and is the exception to which this handler should be registered. The type and the values are defined in the xil\_exception.h header file.
- Handler is an Xil\_ExceptionHandler parameter that is the pointer to the exception handling function.

The function provided as the Handler parameter must have the following function prototype:

typedef void (\*Xil\_ExceptionHandler)(void \* DataPtr);

This prototype is declared in the xil\_exception.h header file.

 DataPtr is of parameter type void \* and is the user value to be passed when the Handler is called

When this Handler function is called, the parameter DataPtr contains the same value provided, when the Handler was registered.

Table 4: Registered Exception Types and Values

| Exception Type                      | Value |
|-------------------------------------|-------|
| XIL_EXCEPTION_ID_RESET              | 0     |
| XIL_EXCEPTION_ID_UNDEFINED_INT      | 1     |
| XIL_EXCEPTION_ID_SWI_INT            | 2     |
| XIL_EXCEPTION_ID_PREFETCH_ABORT_INT | 3     |
| XIL_EXCEPTION_ID_DATA_ABORT_INT     | 4     |
| XIL_EXCEPTION_ID_IRQ_INT            | 5     |
| XIL_EXCEPTION_ID_FIQ_INT            | 6     |

### void Xil\_ExceptionRemoveHandler (u8 ExceptionId)

De-register a handler function for a given exception. For possible values of parameter ExceptionId, refer to Table 1.

### void Xil\_ExceptionEnableMask(Mask)

Enable exceptions specified by Mask. The parameter Mask is a bitmask for exceptions to be enabled. The Mask parameter can have the values XIL\_EXCEPTION\_IRQ, XIL\_EXCEPTION\_FIQ, or XIL\_EXCEPTION\_ALL.

### void Xil\_ExceptionEnable(void)

Enable the IRQ exception.

These macros must be called after initializing the vector table with function Xil\_exceptionInit and registering exception handlers with function Xil\_ExceptionRegisterHandler.



### void Xil\_ExceptionDisableMask(Mask)

Disable exceptions specified by Mask. The parameter Mask is a bitmask for exceptions to be disabled. The Mask parameter can have the values XIL\_EXCEPTION\_IRQ, XIL\_EXCEPTION\_FIQ, or XIL\_EXCEPTION\_ALL.

### void Xil\_ExceptionDisable(void)

Disable the IRQ exception.

## Cortex A9 Processor and pl310 Errata Support

Various ARM errata are handled in the standalone BSP. The implementation for errata handling follows ARM guidelines and is based on the open source Linux support for these errata. The errata conditions handled in the standalone BSP are listed below.

- ARM erratum number 742230 (DMB operation may be faulty)
- ARM erratum number 743622 (Faulty hazard checking in the Store Buffer may lead to data corruption)
- ARM erratum number 775420 (A data cache maintenance operation which aborts, might lead to deadlock)
- ARM erratum number 794073 (Speculative instruction fetches with MMU disabled might not comply with architectural requirements)
- ARM erratum number 588369 (Clean & Invalidate maintenance operations do not invalidate clean lines)
- ARM PL310 erratum number 727915 (Background Clean and Invalidate by Way operation can cause data corruption)
- ARM PL310 erratum number 753970 (Cache sync operation may be faulty)

For further information on these errata items, please refer to the appropriate ARM documentation at ARM the information center.

The BSP file xil\_errata.h defines macros for these errata. The handling of the errata are enabled by default. To disable handling of all the errata globally, un-define the macro ENABLE\_ARM\_ERRATA in xil\_errata.h. To disable errata on a per-erratum basis, un-define relevant macros in xil\_errata.h.

## **Cortex A9 Processor File Support**

The following links take you directly to the gcc file support function.

```
int read(int fd, char *buf, int nbytes) int write(int fd, char *buf, int nbytes) int isatty(int fd) int fcntl (int fd, int cmd, long arg)
```

File support is limited to the stdin and stdout streams. Consequently, the following functions are *not* necessary:

### gcc

```
open() (in gcc/open.c)
close() (in gcc/close.c)
fstat() (in gcc/fstat.c)
unlink() (in gcc/unlink.c)
lseek() (in gcc/lseek.c)
```

These files are included for completeness and because they are referenced by the C library.

UG647 October 5, 2016 <a href="https://www.xilinx.com">www.xilinx.com</a> Send Feedback 25



### **Cortex A9 gcc File Support Function Descriptions**

```
int read(int fd, char *buf, int nbytes)
```

The read() function in gcc/read.c reads nbytes bytes from the standard input by calling inbyte(). It blocks until all characters are available, or the end of line character is read. The read() function returns the number of characters read. The fd parameter is ignored.

```
int write(int fd, char *buf, int nbytes)
```

Writes nbytes bytes to the standard output by calling outbyte(). It blocks until all characters have been written. The write() function returns the number of characters written. The fd parameter is ignored.

```
int isatty(int fd)
```

Reports if a file is connected to a tty. This function always returns 1, because only the stdin and stdout streams are supported.

```
int fcntl (int fd, int cmd, long arg)
```

A dummy implementation of fcntl, which always returns 0. fcntl is intended to manipulate file descriptors according to the command specified by cmd. Because Standalone does not provide a file system, this function is not used.

## **Cortex A9 gcc Errno Function**

```
int errno( )
```

Returns the global value of errno as set by the last C library call.

## **Cortex A9 gcc Memory Management**

```
char *sbrk(int nbytes)
```

Allocates nbytes of heap and returns a pointer to that piece of memory. This function is called from the memory allocation functions of the C library.

## **Cortex A9 gcc Process Functions**

The functions getpid() in getpid.c and kill() in kill.c are included for completeness and because they are referenced by the C library.

## Cortex A9 Processor-Specific Include Files

The xreg\_cortexa9.h include file contains the register numbers and the register bits for the ARM Cortex-A9 processor.

The xpseudo\_asm.h include file contains the definitions for the most often used inline assembler instructions, available as macros. These can be very useful for tasks such as setting or getting special purpose registers, synchronization, or cache manipulation. These inline assembler instructions can be used from drivers and user applications written in C.

UG647 October 5, 2016 <u>www.xilinx.com</u> Send Feedback



### **Cortex A9 Time Functions**

The xtime\_I.c file and corresponding xtime\_I.h include file provide access to the 64-bit Global Counter in the PMU. This counter increases by one at every 2 processor cycles. The sleep.c file and corresponding sleep.h include file implement sleep functions. Sleep functions are implemented as busy loops.

### **Cortex A9 Time Function Summary**

The time functions are summarized below. Click on the function name to go to the description.

typedef unsigned long long XTime void XTime\_SetTime(XTime xtime) void XTime\_GetTime(XTime \*xtime) unsigned int usleep(unsigned int useconds) unsigned int sleep(unsigned int \_seconds)

### **Cortex A9 Time Function Descriptions**

typedef unsigned long long XTime

The XTime type in xtime\_l.h is a 64-bit value, which represents the Global Counter.

void XTime\_SetTime(XTime xtime)

Sets the global timer to the value in xtime.

void XTime\_GetTime(XTime \*xtime)

Writes the current value of the Global Timer to variable xtime.

unsigned int usleep(unsigned int useconds)

Delays the execution of a program by useconds microseconds. It returns zero if the delay can be achieved or -1 if the delay can't be achieved. This function requires that the processor frequency (in Hz) is defined in xparameters.h.

unsigned int **sleep**(unsigned int \_seconds)

Delays the execution of a program by what is specified in seconds. It always returns zero. This function requires that the processor frequency (in Hz) is defined in xparameters.h.

Send Feedback 27



### **Cortex A9 Event Counters**

 $xpm\_counter.c$  and  $xpm\_counter.h$  provide APIs for configuring and controlling the Cortex-A9

Performance Monitor Events. Cortex-A9 Performance Monitor has 6 event counters which can be used to count a variety of events described in Cortex-A9 TRM.

 $xpm\_counter.h$  defines configurations (XPM\_CNTRCFGx) which specifies the event counters to count a set of events.

### **Cortex A9 Event Counters Function Summary**

The Event Counters functions are summarized below. Click on the function name to go to the description.

void Xpm\_SetEvents(int PmcrCfg)

void Xpm GetEventCounters(u32 \*PmCtrValue)

### **Cortex A9 Event Counters Function Description**

void Xpm\_SetEvents(int PmcrCfg)

This function configures the Cortex A9 event counters controller, with the event codes, in a configuration selected by the user and enables the counters.

PmcrCfg is configuration value based on which the event counters are configured.

Use XPM\_CNTRCFG\* values defined in xpm\_counter.h to define a configuration which specify the event counters to count a set of events.

## void Xpm\_GetEventCounters(u32 \*PmCtrValue)

This function disables the event counters and returns the counter values.

PmCtrValue returns the counter values.

# Cortex R5 Processor API

Standalone BSP contains boot code, cache, exception handling, file and memory management, configuration, time and processor-specific include functions. It supports gcc compiler. The following lists the Cortex R5 Processor API sections. You can click on a link to go directly to the function section.

- Cortex R5 Processor Boot Code
- Cortex R5 Processor Cache Functions
- Cortex R5 Processor MPU Handling
- Cortex R5 Processor Exception Handling
- Cortex R5 gcc File Support
- Cortex R5 gcc Errno Functions
- Cortex R5 gcc Memory Management
- Cortex R5 gcc Process Functions
- Cortex R5 Processor-Specific Include Files
- Cortex R5 Time Functions

The following subsections describe the functions by type.

Send Feedback 28



## **Cortex R5 Processor Boot Code**

The boot.S file contains a minimal set of code for transferring control from the processor's reset location to the start of the application. It performs the following tasks.

- Disable branch prediction, MPU, instruction and data caches,
- Initialize stack pointer for IRQ, FIQ, supervisor, abort, undefined and system mode
- Invalidate entire instruction and data caches
- Configure memory attributes for various region using MPU
- Enable branch prediction, data cache, instruction cache and MPU

### **Cortex R5 Processor Cache Functions**

The  $xil\_cache.c$  file and the corresponding  $xil\_cache.h$  header file provide access to the following cache and cache-related operations.

## **Cache Function Summary**

The following are links to the function descriptions. Click on the name to go to that function.

- void Xil\_DCacheEnable(void)
- void Xil\_DCacheDisable(void)
- void Xil\_DCacheInvalidate(void)
- void Xil\_DCacheInvalidateLine(INTPTR adr)
- void Xil DCacheInvalidateRange(INTPTR adr, u32 len)
- void Xil\_DCacheFlush(void)
- void Xil\_DCacheFlushLine(INTPTR adr)
- void Xil\_DCacheFlushRange(INTPTR adr,u32 len)
- void Xil\_DCacheStoreLine(INTPTR adr)
- void Xil\_ICacheEnable(void)
- void Xil\_ICacheDisable(void)
- void Xil\_ICacheInvalidate(void)
- void Xil\_ICacheInvalidateLine(INTPTR adr)
- void Xil\_ICacheInvalidateRange(INTPTR adr, u32 len)

### **Cache Function Descriptions**

void Xil\_DCacheEnable(void)

Enable the data caches.

void Xil\_DCacheDisable(void)

Disable the data caches.

void Xil\_DCacheInvalidate(void)

Invalidate the entire data cache.

UG647 October 5, 2016 www.xilinx.com Send Feedback 29



void Xil\_DCacheInvalidateLine(INTPTR adr)

Invalidate a data cache line. If the byte specified by adr is cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated. A subsequent data access to this address results in a cache miss and a cache line refill.

void Xil\_DCacheInvalidateRange(INTPTR adr, u32 len)

Invalidates the data cache lines that are described by the address range starting from adr and len bytes long. A subsequent data access to any address in this range results in a cache miss and a cache line refill.

void Xil\_DCacheFlush(void)

Flush the entire Data cache.

void Xil\_DCacheFlushLine(INTPTR adr)

Flush a Data cache line. If the byte specified by the address (adr) is cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the entire contents of the cacheline are written to system memory before the line is invalidated. A subsequent data access to this address results in a cache miss and a cache line refill.

void Xil\_DCacheFlushRange(INTPTR adr,u32 len)

Flush the data cache lines that are described by the address range starting from adr and len bytes long. A subsequent data access to any address in this range results in a cache miss and a cache line refill.

void Xil\_DCacheStoreLine(INTPTR adr)

Store a Data cache line. If the byte specified by the adr is cached by the data cache and the cacheline is modified (dirty), the entire contents of the cacheline are written to system memory. After the store completes, the cacheline is marked as unmodified (not dirty).

void Xil\_ICacheEnable(void)

Enable the instruction caches.

void Xil\_ICacheDisable(void)

Disable the instruction caches.

void Xil\_ICacheInvalidate(void)

Invalidate the entire instruction cache.



```
void Xil_ICacheInvalidateLine(INTPTR adr)
```

Invalidate an instruction cache line. If the instruction specified by the parameter adr is cached by the instruction cache, the cacheline containing that instruction is invalidated.

```
void Xil_ICacheInvalidateRange(INTPTR adr, u32 len)
```

Invalidate the instruction cache for the given address range. If the bytes specified by the adr are cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated.

## **Cortex R5 Processor MPU Handling**

The standalone BSP MPU handling API is implemented in file xil\_mpu.c and the corresponding header file xil\_mpu.h.

### **MMU Handling Function Summary**

The following functions describe the available MMU handling API.

- void Xil\_SetMPURegion(INTPTR addr, u64 size, u32 attrib)
- void Xil\_SetTlbAttributes(INTPTR addr, u32 attrib)
- void Xil EnableMPU(void)
- void Xil DisableMPU(void)

### MMU Handling Function Descriptions

```
void Xil_SetMPURegion(INTPTR addr, u64 size, u32 attrib)
```

This function changes the memory attributes for a section of memory with starting address given by variable addr of the region size provided by variable size and having attributes specified by variable attrib.

This API can be used to change attributes such as cache-ability and share-ability of a specified memory region, access to a particular region etc.

```
void Xil SetTlbAttributes(INTPTR addr, u32 attrib)
```

It sets the memory attributes for a section of memory with starting address specified by variable addr of the region size 1MB with attributes given by variable attrib. This API is provided for the usage of similar type of API across other platforms/processors.

void Xil\_EnableMPU(void)

Enable the MPU

void Xil\_DisableMPU(void)

Disable the MPU

## **Cortex R5 Processor Exception Handling**

The Standalone BSP provides an exception handling API. For details about the exceptions and interrupts on ARM Cortex-R5 processor, refer to "Exceptions" under the chapter "The System

UG647 October 5, 2016 <a href="https://www.xilinx.com">www.xilinx.com</a> Send Feedback 31



Level Programmers' Model" in the ARM Architecture Reference Manual ARMv7-A and ARMv-7R edition.

The exception handling API is implemented in a set of the files - asm\_vectors.S, vectors.c, xil\_exception.c, and the corresponding header files vectors.h and xil\_exception.h.

## **Exception Handling Function Summary**

The following are links to the function descriptions. Click on the name to go to that function.

- void Xil\_ExceptionInit(void)
- void Xil\_ExceptionRegisterHandler (u32 ExceptionId, Xil\_ExceptionHandler Handler, void \*DataPtr)
- void Xil\_ExceptionRemoveHandler (u32 ExceptionId)
- void Xil\_ExceptionEnableMask(Mask)
- void Xil ExceptionEnable(void)
- void Xil ExceptionDisableMask(Mask)
- void Xil\_ExceptionDisable(void)

### **Exception Handling Function Descriptions**

```
void Xil_ExceptionInit(void)
```

Sets up the interrupt vector table and registers a "do nothing" function for each exception. This function has no parameters and does not return a value. This function must be called before registering any exception handlers or enabling any interrupts.

```
void Xil_ExceptionRegisterHandler (u32 ExceptionId,
    Xil ExceptionHandler Handler, void *DataPtr)
```

Registers an exception handler for a specific exception; does not return a value. Refer to Table the for a list of exception types and their values.

The parameters are:

- **ExceptionId** is of parameter type u8, and is the exception to which this handler should be registered. The type and the values are defined in the xil\_exception.h header file. Refer Table 5 for more details.
- Handler is an Xil\_ExceptionHandler parameter that is the pointer to the exception handling function. The function provided as the Handler parameter must have the following function prototype:

```
typedef void (*Xil_ExceptionHandler)(void * DataPtr);
```

This prototype is declared in the xil\_exception.h header file.

• DataPtr is of parameter type void \* and is the user value to be passed when the Handler is called. When this Handler function is called, the parameter DataPtr contains the same value provided, when the Handler was registered.

Table 5: Registered Exception Types and Values

| Exception Type                      | Value |
|-------------------------------------|-------|
| XIL_EXCEPTION_ID_RESET              | 0     |
| XIL_EXCEPTION_ID_UNDEFINED_INT      | 1     |
| XIL_EXCEPTION_ID_SWI_INT            | 2     |
| XIL_EXCEPTION_ID_PREFETCH_ABORT_INT | 3     |

UG647 October 5, 2016 www.xilinx.com Send Feedback 32



Table 5: Registered Exception Types and Values

| Exception Type                  | Value |
|---------------------------------|-------|
| XIL_EXCEPTION_ID_DATA_ABORT_INT | 4     |
| XIL_EXCEPTION_ID_IRQ_INT        | 5     |
| XIL_EXCEPTION_ID_FIQ_INT        | 6     |

void Xil\_ExceptionRemoveHandler (u32 ExceptionId)

De-register a handler function for a given exception. For possible values of parameter ExceptionId, refer above the table

void Xil\_ExceptionEnableMask(Mask)

Enable exceptions specified by Mask. The parameter Mask is a bitmask for exceptions to be enabled. The Mask parameter can have the values XIL\_EXCEPTION\_IRQ, XIL\_EXCEPTION\_FIQ, or XIL\_EXCEPTION\_ALL.

void Xil ExceptionEnable(void)

Enable the IRQ exception.

These macros must be called after initializing the vector table with function Xil\_exceptionInit and registering exception handlers with function Xil\_ExceptionRegisterHandler

void Xil\_ExceptionDisableMask(Mask)

Disable exceptions specified by Mask. The parameter Mask is a bitmask for exceptions to be disabled. The Mask parameter can have the values XIL\_EXCEPTION\_IRQ, XIL\_EXCEPTION\_FIQ, or XIL\_EXCEPTION\_ALL.

void Xil\_ExceptionDisable(void)

Disable the IRQ exception.

## **Cortex R5 gcc File Support**

The following links take you directly to the gcc file support function.

- int read(int fd, char \*buf, int nbytes)
- int write(int fd, char \*buf, int nbytes)
- int isatty(int fd)
- int fcntl (int fd, int cmd, long arg)

File support is limited to the stdin and stdout streams. Consequently, the following functions are *not* necessary:

#### gcc

- open( ) (in gcc/open.c)
- close( ) (in gcc/close.c)
- fstat() (in gcc/fstat.c)
- unlink( ) (in gcc/unlink.c)



lseek( ) (in gcc/lseek.c)

These files are included for completeness and because they are referenced by the C library.

### **Cortex R5 gcc File Support Function Descriptions**

```
int read(int fd, char *buf, int nbytes)
```

The read() function in gcc/read.c reads nbytes bytes from the standard input by calling inbyte(). It blocks until all characters are available, or the end of line character is read. The read() function returns the number of characters read. The fd parameter is ignored.

```
int write(int fd, char *buf, int nbytes)
```

Writes nbytes bytes to the standard output by calling outbyte(). It blocks until all characters have been written. The write() function returns the number of characters written. The fd parameter is ignored.

```
int isatty(int fd)
```

Reports if a file is connected to a tty. This function always returns 1, because only the stdin and stdout streams are supported.

```
int fcntl (int fd, int cmd, long arg)
```

A dummy implementation of fcntl, which always returns 0. fcntl is intended to manipulate file descriptors according to the command specified by cmd. Because Standalone does not provide a file system, this function is not used.

## Cortex R5 gcc Errno Functions

```
int errno( )
```

Returns the global value of errno as set by the last C library call.

### Cortex R5 gcc Memory Management

```
char *sbrk(int nbytes)
```

Allocates nbytes of heap and returns a pointer to that piece of memory. This function is called from the memory allocation functions of the C library.

### **Cortex R5 gcc Process Functions**

The functions getpid() in getpid.c and kill() in kill.c are included for completeness and because they are referenced by the C library.

## Cortex R5 Processor-Specific Include Files

The xreg\_cortexr5.h include file contains the register numbers and the register bits for the ARM Cortex-R5 processor.

The xpseudo\_asm.h include file contains the definitions for the most often used inline assembler instructions, available as macros. These can be very useful for tasks such as setting

UG647 October 5, 2016 <a href="https://www.xilinx.com">www.xilinx.com</a> Send Feedback 34



or getting special purpose registers, synchronization, or cache manipulation. These inline assembler instructions can be used from drivers and user applications written in C.

### **Cortex R5 Time Functions**

The xtime\_I.c file and corresponding xtime\_I.h include file provide access to the 32-bit counter in TTC. The sleep.c & usleep.c file and corresponding sleep.h include file implement sleep functions. Sleep functions are implemented as busy loops.

### **Cortex R5 Time Function Summary**

The time functions are summarized below. Click on the function name to go to the description.

- Typedef u32 XTime
- void XTime\_StartTimer(void)
- void XTime\_SetTime(XTime xtime)
- void XTime\_GetTime(XTime \*xtime)
- unsigned int usleep(unsigned int useconds)
- unsigned int sleep(unsigned int \_seconds)

### **Cortex R5 Time Function Descriptions**

Typedef u32 XTime

The XTime type in xtime\_I.h is a 32-bit value, which represents the TTC counter value.

void XTime\_StartTimer(void)

Starts the TTC timer 3 counter 0 if present and if it is not already running with desired parameters for sleep functionalities.

void XTime SetTime(XTime xtime)

Sets the ttc counter to the value in xtime.

void XTime GetTime(XTime \*xtime)

Writes the current value of the TTC counter to variable xtime.

unsigned int usleep(unsigned int useconds)

Delays the execution of a program by useconds microseconds. The counts per microseconds are defined in the xtime\_1.h file.

The usleep API is implemented using TTC3 counter 0 if present. When TTC3 is absent, usleep is implemented using set of assembly instructions which is tested with instruction and data caches enabled and is known to provide proper delay. It may give more delay than expected when caches are disabled. If interrupt comes when usleep using assembly instruction is being executed, the delay may be greater than what is expected since once the interrupt is served count resumes from where it was interrupted.



unsigned int sleep(unsigned int \_seconds)

Delays the execution of a program by what is specified in seconds. The counts per seconds are defined in the xtime 1.h file.

The sleep API is implemented using TTC3 counter 0 if present. When TTC3 is absent, sleep is implemented using set of assembly instructions which is tested with instruction and data caches enabled and is known to provide proper delay. It may give more delay than expected when caches are disabled. If interrupt comes when sleep using assembly instruction is being executed, the delay may be greater than what is expected since once the interrupt is served count resumes from where it was interrupted.

# Cortex A53 Processor API

Cortex-A53 standalone BSP contains two separate BSPs for 32bit mode and 64bit mode. The 32bit mode of cortex-A53 is compatible with ARMv7-A architecture whereas 64bit mode of cortex-A53 contains ARMv8-A architecture.

Standalone BSP contains boot code, cache, exception handling, file and memory management, configuration, time and processor-specific include functions. It supports gcc compiler. The following Cortex A53 Processor API subsections describe the functions by type.

- Cortex A53 Processor Boot Code
- Cortex A53 Processor Cache Functions
- Cortex A53 Processor MMU Handling
- Cortex A53 Processor Exception Handling
- Cortex A53 gcc File Support
- Cortex A53 gcc Errno Function
- Cortex A53 gcc Memory Management
- Cortex A53 gcc Process Functions
- Cortex A53 Processor-Specific Include Files
- Cortex A53 Time Functions

### **Cortex A53 Processor Boot Code**

The boot.S file contains a minimal set of code for transferring control from the processor's reset location to the start of the application. It performs the following tasks.

- Disable branch prediction, MMU, instruction and data caches
- Initialize stack pointer (for current exception level for 64bit mode and for IRQ, FIQ, supervisor, abort, undefined and system mode for 32bit mode)
- Invalidate TLBs, entire instruction and data caches
- Program MMU
- Enable branch prediction, data cache, instruction cache and MPU

### **Cortex A53 Processor Cache Functions**

The  $xil\_cache.c$  file and the corresponding  $xil\_cache.h$  header file provide access to the following cache and cache-related operations.

## **Cache Function Summary**

The following are links to the function descriptions. Click on the name to go to that function.

- void Xil\_DCacheEnable(void)
- void Xil\_ DCacheDisable(void)

UG647 October 5, 2016 www.xilinx.com Send Feedback 36



- void Xil\_DCacheInvalidate(void)
- void Xil\_DCacheInvalidateLine(INTPTR adr)
- void Xil\_DCacheInvalidateRange(INTPTR adr, INTPTR len)
- void Xil\_DCacheFlush(void)
- void Xil DCacheFlushLine(INTPTR adr)
- void Xil\_DCacheFlushRange(INTPTR adr, INTPTR len)
- void Xil ICacheEnable(void)
- void Xil ICacheDisable(void)
- void Xil\_ICacheInvalidate(void)
- void Xil\_ICacheInvalidateLine(INTPTR adr)
- void Xil\_ICacheInvalidateRange(INTPTR adr, INTPTR len)

# **Cache Function Descriptions**

void Xil\_DCacheEnable(void)

Enable the data caches.

void Xil\_ DCacheDisable(void)

Disable the data caches.

void Xil\_DCacheInvalidate(void)

Clean and invalidate the entire data cache. ARMv8 architectures do not support simply invalidating the cachelines which are present in caches. In case of an environment which supports visualization like Cortex-A53, if simple invalidation functionalities are present, it may lead to loss of essential data in some scenarios. For example, If one OS invalidates a line belonging to another OS, it may crash the other OS due to the loss of essential data. Hence, such operations are prompted to clean and invalidate which avoids data corruption.

void Xil\_DCacheInvalidateLine(INTPTR adr)

Clean and invalidate a data cache line. ARMv8 architectures does not support simply invalidating the cachelines which are present in caches. In case of an environment which supports visualization like Cortex-A53, if simple invalidation functionalities are present, it may lead to loss of essential data in some scenarios. For example, If one OS invalidates a line belonging to another OS, it may crash the other OS due to the loss of essential data. Hence, such operations are prompted to clean and invalidate which avoids such corruption.

void Xil\_DCacheInvalidateRange(INTPTR adr, INTPTR len)

Clean and invalidate the data cache lines that are described by the address range starting from adr and len bytes long. ARMv8 architectures does not support simply invalidating the cachelines which are present in caches. In case of an environment which supports visualization like Cortex-A53, if simple invalidation functionalities are present, it may lead to loss of essential data in some scenarios. For example, If one OS invalidates a line belonging to another OS, it may crash the other OS due to the loss of essential data. Hence, such operations are prompted to clean and invalidate which avoids such corruption.



38

void Xil\_DCacheFlush(void)

Flush the entire Data cache.

void Xil\_DCacheFlushLine(INTPTR adr)

Flush a Data cache line. If the byte specified by the address (adr) is cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the entire contents of the cacheline are written to system memory before the line is invalidated. A subsequent data access to this address results in a cache miss and a cache line refill.

void Xil\_DCacheFlushRange(INTPTR adr, INTPTR len)

Flush the data cache lines that are described by the address range starting from adr and len bytes long. A subsequent data access to any address in this range results in a cache miss and a cache line refill.

void Xil\_ICacheEnable(void)

Enable the instruction caches.

void Xil\_ICacheDisable(void)

Disable the instruction caches.

void Xil\_ICacheInvalidate(void)

Invalidate the entire instruction cache.

void Xil\_ICacheInvalidateLine(INTPTR adr)

Invalidate an instruction cache line. If the instruction specified by the parameter adr is cached by the instruction cache, the cacheline containing that instruction is invalidated.

void Xil\_ICacheInvalidateRange(INTPTR adr, INTPTR len)

Invalidate the instruction cache for the given address range. If the bytes specified by the adr are cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost and are not written to system memory before the line is invalidated.

# Cortex A53 Processor MMU Handling

The standalone BSP MMU handling API is implemented in file  $xil\_mmu.c$  and the corresponding header file  $xil\_mmu.h$ .

## MMU Handling Function Summary

The following functions describe the available MMU handling API.



```
void Xil_SetTlbAttributes(INTPTR addr, INTPTR attrib)
```

For 64 bit mode it sets the memory attributes for a section of memory with starting address specified by variable addr of the region size 2MB (if the address is less than 4GB) or 1 GB (if the address is beyond 4GB) with attributes given by variable attrib whereas for 32bit mode it sets the memory attributes for a section of memory with starting address specified by variable addr of the region size 1MB with attributes given by variable attrib. This API can be used to change attributes such as cache-ability and share-ability of a specified memory region, access to a particular region.

# **Cortex A53 Processor Exception Handling**

The Standalone BSP provides an exception handling API. The exception handling API is implemented in a set of the files - asm\_vectors.S, vectors.c, xil\_exception.c, and the corresponding header files vectors.h and xil\_exception.h.

# **Exception Handling Function Summary**

The following are links to the function descriptions. Click on the name to go to that function.

- void Xil\_ExceptionInit(void)
- void Xil\_ExceptionRegisterHandler (u32 ExceptionId, Xil\_ExceptionHandler Handler, void \*DataPtr)
- void Xil\_ExceptionRemoveHandler (u32 ExceptionId)
- void Xil\_ExceptionEnableMask(Mask)
- void Xil\_ExceptionEnable(void)
- void Xil\_ExceptionDisableMask(Mask)
- void Xil\_ExceptionDisable(void)

#### **Exception Handling Function Descriptions**

```
void Xil_ExceptionInit(void)
```

Sets up the interrupt vector table and registers a "do nothing" function for each exception. This function has no parameters and does not return a value. This function must be called before registering any exception handlers or enabling any interrupts.

```
void Xil_ExceptionRegisterHandler (u32 ExceptionId,
    Xil_ExceptionHandler Handler, void *DataPtr)
```

Registers an exception handler for a specific exception; does not return a value. Refer to Table for a list of exception types and their values (for 64bit and 32bit mode).

The parameters are:

- **ExceptionId** is of parameter type u8, and is the exception to which this handler should be registered. The type and the values are defined in the xil\_exception.h header file. Refer Table 6 and Table 7 for more details.
- Handler is an Xil\_ExceptionHandler parameter that is the pointer to the exception handling function. The function provided as the Handler parameter must have the following function prototype:

```
typedef void (*Xil_ExceptionHandler)(void * DataPtr);
```

This prototype is declared in the xil\_exception.h header file.



• **DataPtr** is of parameter type **void** \* and is the user value to be passed when the Handler is called. When this Handler function is called, the parameter **DataPtr** contains the same value provided, when the Handler was registered.

Table 6: Registered Exception Types and Values for 64bit Mode

| Exception Type                            | Value |
|-------------------------------------------|-------|
| #define XIL_EXCEPTION_ID_SYNC_INT         | 1     |
| #define XIL_EXCEPTION_ID_IRQ_INT          | 2     |
| #define XIL_EXCEPTION_ID_FIQ_INT          | 3     |
| #define XIL_EXCEPTION_ID_SERROR_ABORT_INT | 4     |

Table 7: Registered Exception Types and Values for 32bit Mode

| Exception Type                      | Value |
|-------------------------------------|-------|
| XIL_EXCEPTION_ID_RESET              | 0     |
| XIL_EXCEPTION_ID_UNDEFINED_INT      | 1     |
| XIL_EXCEPTION_ID_SWI_INT            | 2     |
| XIL_EXCEPTION_ID_PREFETCH_ABORT_INT | 3     |
| XIL_EXCEPTION_ID_DATA_ABORT_INT     | 4     |
| XIL_EXCEPTION_ID_IRQ_INT            | 5     |
| XIL_EXCEPTION_ID_FIQ_INT            | 6     |

void Xil\_ExceptionRemoveHandler (u32 ExceptionId)

De-register a handler function for a given exception. For possible values of parameter ExceptionId, refer above the table

void Xil\_ExceptionEnableMask(Mask)

Enable exceptions specified by Mask. The parameter Mask is a bitmask for exceptions to be enabled. The Mask parameter can have the values XIL\_EXCEPTION\_IRQ, XIL\_EXCEPTION\_FIQ, or XIL\_EXCEPTION\_ALL.

void Xil\_ExceptionEnable(void)

Enable the IRQ exception.

These macros must be called after initializing the vector table with function Xil\_exceptionInit and registering exception handlers with function Xil\_ExceptionRegisterHandler

void Xil\_ExceptionDisableMask(Mask)

Disable exceptions specified by Mask. The parameter Mask is a bitmask for exceptions to be disabled. The Mask parameter can have the values XIL\_EXCEPTION\_IRQ, XIL\_EXCEPTION\_FIQ, or XIL\_EXCEPTION\_ALL.

void Xil\_ExceptionDisable(void)

Disable the IRQ exception.



# **Cortex A53 gcc File Support**

The following links take you directly to the gcc file support function.

- int read(int fd, char \*buf, int nbytes)
- int write(int fd, char \*buf, int nbytes)
- int isatty(int fd)
- int fcntl (int fd, int cmd, long arg)

File support is limited to the stdin and stdout streams. Consequently, the following functions are *not* necessary:

# gcc

- open() (in gcc/open.c)
- close( ) (in gcc/close.c)
- fstat() (in gcc/fstat.c)
- unlink() (in gcc/unlink.c)
- lseek( ) (in gcc/lseek.c)

These files are included for completeness and because they are referenced by the C library.

# **Cortex A53 gcc File Support Function Descriptions**

```
int read(int fd, char *buf, int nbytes)
```

The read() function in gcc/read.c reads nbytes bytes from the standard input by calling inbyte(). It blocks until all characters are available, or the end of line character is read. The read() function returns the number of characters read. The fd parameter is ignored.

```
int write(int fd, char *buf, int nbytes)
```

Writes nbytes bytes to the standard output by calling outbyte(). It blocks until all characters have been written. The write() function returns the number of characters written. The fd parameter is ignored.

```
int isatty(int fd)
```

Reports if a file is connected to a tty. This function always returns 1, because only the stdin and stdout streams are supported.

```
int fcntl (int fd, int cmd, long arg)
```

A dummy implementation of fcntl, which always returns 0. fcntl is intended to manipulate file descriptors according to the command specified by cmd. Because Standalone does not provide a file system, this function is not used.

# Cortex A53 gcc Errno Function

```
int errno( )
```

Returns the global value of errno as set by the last C library call.

UG647 October 5, 2016 <a href="https://www.xilinx.com">www.xilinx.com</a> Send Feedback 41



# **Cortex A53 gcc Memory Management**

char \*sbrk(int nbytes)

Allocates nbytes of heap and returns a pointer to that piece of memory. This function is called from the memory allocation functions of the C library.

# **Cortex A53 gcc Process Functions**

The functions <code>getpid()</code> in <code>getpid.c</code> and <code>kill()</code> in <code>kill.c</code> are included for completeness and because they are referenced by the <code>C</code> library. The function <code>initialise\_monitor\_handles()</code> in <code>initialise\_monitor\_handles.c</code> is included for completeness and because they are referenced by the <code>C</code> library for 64bit mode.

# **Cortex A53 Processor-Specific Include Files**

The xreg\_cortexa53.h include file contains the register numbers and the register bits for the ARM Cortex-A53 processor.

The xpseudo\_asm.h include file contains the definitions for the most often used inline assembler instructions, available as macros. These can be very useful for tasks such as setting or getting special purpose registers, synchronization, or cache manipulation. These inline assembler instructions can be used from drivers and user applications written in C.

# **Cortex A53 Time Functions**

The xtime\_1.c file and corresponding xtime\_1.h include file provide access to the 64-bit generic counter in Cortex-A53. The sleep.c & usleep.c file and corresponding sleep.h include file implement sleep functions. Sleep functions are implemented as busy loops.

#### **Cortex A53 Time Function Summary**

The time functions are summarized below.

- typedef u64 XTime
- void XTime\_StartTimer(void)
- void XTime SetTime(XTime xtime)
- void XTime GetTime(XTime \*xtime)
- unsigned int usleep(unsigned int useconds)
- unsigned int sleep(unsigned int \_seconds)

## **Cortex A53 Time Function Descriptions**

typedef u64 XTime

The XTime type in xtime\_1.h is a 64-bit value, which represents the Generic counter value.

void XTime\_StartTimer(void)

Starts the global timer counter.

void XTime\_SetTime(XTime xtime)

The function does not contain anything, as the generic counter in A53 runs constantly and it does not display actual time. This function is kept for the uniformity across platform.



void XTime\_GetTime(XTime \*xtime)

Writes the current value of the generic counter to variable xtime.

unsigned int usleep(unsigned int useconds)

Delays the execution of a program by useconds microseconds. The counts per microseconds are defined in the usleep.c file.

unsigned int sleep(unsigned int \_seconds)

Delays the execution of a program by what is specified in seconds. The counts per seconds are defined in the xtime\_I.h file.



# Xilinx Hardware Abstraction Layer

The following sections describe the Xilinx® Hardware Abstraction Layer API. It contains the following sections:

- Types (xil\_types)
- Register IO (xil\_io)
- Exception (xil\_exception)
- Cache (xil\_cache)
- Assert (xil\_assert)
- Extra Header File
- Test Memory (xil\_testmem)
- Test Register IO (xil\_testio)
- Test Cache (xil\_testcache)
- Hardware Abstraction Layer Migration Tips

# Types (xil\_types)

# **Header File**

#include "xil\_types.h"

# **Typedef**

```
typedef unsigned char u8
typedef unsigned short u16
typedef unsigned long u32
typedef unsigned long long u64
typedef char s8
typedef short s16
typedef long s32
typedef long long s64
```

#### **Macros**

| Macro                            | Value      |
|----------------------------------|------------|
| #define TRUE                     | 1          |
| #define FALSE                    | 0          |
| #define NULL                     | 0          |
| #define XIL_COMPONENT_IS_READY   | 0x11111111 |
| #define XIL_COMPONENT_IS_STARTED | 0x2222222  |



# Register IO (xil\_io)

# **Header File**

#include "xil\_io.h"

#### Common API

The following is a linked summary of register IO functions. They can run on MicroBlaze and Cortex A9 processors.

```
u8 Xil_ln8(u32 Addr)
u16 Xil_EndianSwap16 (u16 Data)
u16 Xil_Htons(u16 Data)
u16 Xil_In16(u32 Addr)
u16 Xil In16BE(u32 Addr)
u16 Xil_In16LE(u32 Addr)
u16 Xil_Ntohs(u16 Data)
u32 Xil_EndianSwap32 u32 Data)
u32 Xil_Htonl(u32 Data)
u32 Xil In32(u32 Addr)
u32 Xil_In32BE(u32 Addr)
u32 Xil_In32LE(u32 Addr)
u32 Xil_Ntohs(u32 Data)
void Xil_Out8(u32 Addr, u8 Value)
void Xil_Out16(u32 Addr, u16 Value)
void Xil Out16BE(u32 Addr, u16 Value)
void Xil_Out16LE(u32 Addr, u16 Value)
void Xil_Out32(u32 Addr, u32 Value)
void Xil_Out32BE(u32 Addr, u32 Value)
void Xil_Out32LE(u32 Addr, u32 Value)
```

# u8 **Xil\_In8**(u32 Addr)

Perform an input operation for an 8-bit memory location by reading from the specified address and returning the value read from that address.

#### Parameters:

Addr contains the address at which to perform the input operation.

#### Returns:

The value read from the specified input address.

#### ul6 Xil\_EndianSwap16 (ul6 Data)

Perform a 16-bit endian swapping.

#### Parameters:

Data contains the value to be swapped.

## Returns:

Endian swapped value.



# ul6 Xil\_Htons(ul6 Data)

Convert a 16-bit number from host byte order to network byte order.

#### Parameters:

Data the 16-bit number to be converted.

#### Returns:

The converted 16-bit number in network byte order.

## u16 **Xil\_In16**(u32 Addr)

Perform an input operation for a 16-bit memory location by reading from the specified address and returning the value read from that address.

#### Parameters:

Addr contains the address at which to perform the input operation.

#### Returns:

The value read from the specified input address.

## u16 Xil\_In16BE(u32 Addr)

Perform an big-endian input operation for a 16-bit memory location by reading from the specified address and returning the value read from that address.

#### Parameters:

Addr contains the address at which to perform the input operation.

#### Returns:

The value read from the specified input address with the proper endianness. The return value has the same endianness as that of the processor. For example, if the processor is little-endian, the return value is the byte-swapped value read from the address.

### u16 **Xil\_In16LE**(u32 Addr)

Perform a little-endian input operation for a 16-bit memory location by reading from the specified address and returning the value read from that address.

## Parameters:

Addr contains the address at which to perform the input operation.

#### Returns:

The value read from the specified input address with the proper endianness. The return value has the same endianness as that of the processor. For example, if the processor is big-endian, the return value is the byte-swapped value read from the address.



# ul6 Xil\_Ntohs(ul6 Data)

Convert a 16-bit number from network byte order to host byte order.

#### Parameters:

Data the 16-bit number to be converted.

#### Returns:

The converted 16-bit number in host byte order.

#### u32 Xil\_EndianSwap32 (u32 Data)

Perform a 32-bit endian swapping.

#### Parameters:

Data contains the value to be swapped.

#### Returns:

Endian swapped value.

# u32 Xil\_Htonl(u32 Data)

Convert a 32-bit number from host byte order to network byte order.

#### Parameters:

Data the 32-bit number to be converted.

#### Returns:

The converted 32-bit number in network byte order.

## u32 **Xil\_In32**(u32 Addr)

Perform an input operation for a 32-bit memory location by reading from the specified address and returning the value read from that address.

#### Parameters:

Addr contains the address at which to perform the input operation.

#### Returns:

The value read from the specified input address.

# u32 Xil\_In32BE(u32 Addr)

Perform a big-endian input operation for a 32-bit memory location by reading from the specified address and returning the value read from that address.

#### Parameters:

Addr contains the address at which to perform the input operation.

# Returns:

The value read from the specified input address with the proper endianness. The return value has the same endianness as that of the processor. For example, if the processor is little-endian, the return value is the byte-swapped value read from the address.



## u32 Xil\_In32LE(u32 Addr)

Perform a little-endian input operation for a 32-bit memory location by reading from the specified address and returning the value read from that address.

#### Parameters:

Addr contains the address at which to perform the input operation.

#### Returns:

The value read from the specified input address with the proper endianness. The return value has the same endianness as that of the processor. For example, if the processor is big-endian, the return value is the byte-swapped value read from the address.

## u32 Xil\_Ntohs(u32 Data)

Convert a 32-bit number from network byte order to host byte order.

#### Parameters:

Data the 32-bit number to be converted.

#### Returns:

The converted 32-bit number in host byte order.

# void Xil\_Out8(u32 Addr, u8 Value)

Perform an output operation for an 8-bit memory location by writing the specified value to the specified address.

#### Parameters:

Addr contains the address at which to perform the output operation.

Value contains the value to be output at the specified address.

```
void Xil_Out16(u32 Addr, u16 Value)
```

Perform an output operation for a 16-bit memory location by writing the specified value to the specified address.

# Parameters:

Addr contains the address at which to perform the output operation.

Value contains the value to be output at the specified address.

```
void Xil Out16BE(u32 Addr, u16 Value)
```

Perform a big-endian output operation for a 16-bit memory location by writing the specified value to the specified address.

## Parameters:

Addr contains the address at which to perform the output operation.

Value contains the value to be output at the specified address. The value has the same endianness as that of the processor. For example, if the processor is little-endian, the byte-swapped value is written to the address.



void Xil\_Out16LE(u32 Addr, u16 Value)

Perform a little-endian output operation for a 16-bit memory location by writing the specified value to the specified address.

#### Parameters:

Addr contains the address at which to perform the output operation.

Value contains the value to be output at the specified address. The value has the same endianness as that of the processor. For example, if the processor is big-endian, the byte-swapped value is written to the address.

```
void Xil_Out32(u32 Addr, u32 Value)
```

Perform an output operation for a 32-bit memory location by writing the specified value to the specified address.

#### Parameters:

Addr contains the address at which to perform the output operation.

Value contains the value to be output at the specified address.

```
void Xil_Out32BE(u32 Addr, u32 Value)
```

Perform a big-endian output operation for a 32-bit memory location by writing the specified value to the specified address.

#### Parameters:

Addr contains the address at which to perform the output operation.

Value contains the value to be output at the specified address. The value has the same endianness as that of the processor. For example, if the processor is little-endian, the byte-swapped value is written to the address.

```
void Xil_Out32LE(u32 Addr, u32 Value)
```

Perform a little-endian output operation for a 32-bit memory location by writing the specified value to the specified address.

#### Parameters:

Addr contains the address at which to perform the output operation.

Value contains the value to be output at the specified address. The value has the same endianness as that of the processor. For example, if the processor is big-endian, the byte-swapped value is written to the address.



# **Exception (xil\_exception)**

#### **Header File**

#include "xil\_exception.h"

## Typedef

```
typedef void(* Xil_ExceptionHandler)(void *Data)
```

This typedef is the exception handler function pointer.

#### Common API

The following are exception functions. They can run on MicroBlaze and Cortex A9 processors.

```
void Xil_ExceptionDisable()
```

Disable Exceptions.

# void Xil\_ExceptionEnable()

Enable Exceptions.

# void Xil\_ExceptionInit()

Initialize exception handling for the processor. The exception vector table is set up with the stub handler for all exceptions.

```
void Xil_ExceptionRegisterHandler(u32 Id,
    Xil ExceptionHandler Handler, void *Data)
```

Make the connection between the ID of the exception source and the associated handler that runs when the exception is recognized. Data is used as the argument when the handler is called.

#### Parameters:

Handler is the handler for that exception.

Data is a reference to data that will be passed to the handler when it is called.

#### void Xil\_ExceptionRemoveHandler(u32 Id)

Remove the handler for a specific exception ID. The stub handler is then registered for this exception ID.

## Parameters:

Id contains the ID of the exception source. It should be <code>XIL\_EXCEPTION\_INT</code> or in the range of <code>0</code> to <code>XIL\_EXCEPTION\_LAST</code>. Refer to the <code>xil\_exception.h</code> file for further information.



#### **Common Macro**

The common macro is:

```
#define XIL_EXCEPTION_ID_INT
```

This macro is defined for all processors and used to set the exception handler that corresponds to the interrupt controller handler. The value is processor-dependent. For example:

```
Xil_ExceptionRegisterHandler(XIL_EXCEPTION_ID_INT,
(XilExceptionHandler)IntcHandler, IntcData)
```

# MicroBlaze Processor-Specific Macros

| Масто                                     | Value                |
|-------------------------------------------|----------------------|
| #define XIL_EXCEPTION_ID_FIRST            | 0                    |
| #define XIL_EXCEPTION_ID_FSL              | 0                    |
| #define XIL_EXCEPTION_ID_UNALIGNED_ACCESS | 1                    |
| #define XIL_EXCEPTION_ID_ILLEGAL_OPCODE   | 2                    |
| #define XIL_EXCEPTION_ID_IOPB_EXCEPTION   | 3                    |
| #define XIL_EXCEPTION_ID_IPLB_EXCEPTION   | 3                    |
| #define XIL_EXCEPTION_ID_DOPB_EXCEPTION   | 4                    |
| #define XIL_EXCEPTION_ID_DPLB_EXCEPTION   | 4                    |
| #define XIL_EXCEPTION_ID_DIV_BY_ZERO      | 5                    |
| #define XIL_EXCEPTION_ID_FPU              | 6                    |
| #define XIL_EXCEPTION_ID_MMU              | 7                    |
| #define XIL_EXCEPTION_ID_LAST             | XIL_EXCEPTION_ID_MMU |

# Cache (xil\_cache)

## **Header File**

#include "xil\_cache.h"

#### Common API

The functions listed in this sub-section can be executed on all processors.

# void Xil\_DCacheDisable()

Disable the data cache.

# void Xil\_DCacheEnable()

Enable the data cache.

# void Xil\_DCacheFlush()

Flush the entire data cache. If any cacheline is dirty (has been modified), it is written to system memory. The entire data cache will be invalidated.



## void Xil\_DCacheFlushRange(u32 Addr, u32 Len)

Flush the data cache for the given address range. If any memory in the address range (identified as Addr) has been modified (and are dirty), the modified cache memory will be written back to system memory. The cacheline will also be invalidated.

#### Parameters:

Addr is the starting address of the range to be flushed.

Len is the length, in bytes, to be flushed.

# void Xil\_DCacheInvalidate()

Invalidate the entire data cache. If any cacheline is dirty (has been modified), the modified contents are lost.

## void Xil\_DCacheInvalidateRange(u32 Addr, u32 Len)

Invalidate the data cache for the given address range. If the bytes specified by the address (Addr) are cached by the data cache, the cacheline containing that byte is invalidated. If the cacheline is modified (dirty), the modified contents are lost.

#### Parameters:

Addr is address of range to be invalidated.

Len is the length in bytes to be invalidated.

## void Xil\_ICacheDisable()

Disable the instruction cache.

# void Xil\_ICacheEnable()

On MicroBlaze processors, enable the instruction cache.

# void Xil\_ICacheInvalidate()

Invalidate the entire instruction cache.

#### void Xil\_ICacheInvalidateRange(u32 Addr, u32 Len)

Invalidate the instruction cache for the given address range.

#### Parameters:

Addr is address of range to be invalidated.

Len is the length in bytes to be invalidated.



# Assert (xil\_assert)

#### **Header File**

#include "xil\_assert.h"

# **Typedef**

typedef void(\* Xil\_AssertCallback)(char \*Filename, int Line)

#### Common API

The functions listed in this sub-section can be executed on all processors.

```
void Xil_Assert(char *File, int Line)
```

Implement assert. Currently, it calls a user-defined callback function if one has been set. Then, potentially enters an infinite loop depending on the value of the Xil\_AssertWait variable.

#### Parameters:

File is the name of the filename of the source.

Line is the line number within File.

# void Xil\_AssertSetCallback(xil\_AssertCallback Routine)

Set up a callback function to be invoked when an assert occurs. If there was already a callback installed, then it is replaced.

#### Parameters:

Routine is the callback to be invoked when an assert is taken.

# #define Xil\_AssertVoid(Expression)

This assert macro is to be used for functions that do not return anything (void). This can be used in conjunction with the Xil\_AssertWait boolean to accommodate tests so that asserts that fail still allow execution to continue.

#### Parameters:

Expression is the expression to evaluate. If it evaluates to false, the assert occurs.

# #define Xil\_AssertNonvoid(Expression)

This assert macro is to be used for functions that return a value. This can be used in conjunction with the Xil\_AssertWait boolean to accommodate tests so that asserts that fail still allow execution to continue.

#### Parameters:

Expression is the expression to evaluate. If it evaluates to false, the assert occurs.

#### Returns:

Returns 0 unless the  $Xil\_AssertWait$  variable is true, in which case no return is made and an infinite loop is entered.



```
#define Xil_AssertVoidAlways()
```

Always assert. This assert macro is to be used for functions that do not return anything (void). Use for instances where an assert should always occur.

#### Returns:

Returns void unless the Xil\_AssertWait variable is true, in which case no return is made and an infinite loop is entered.

```
#define Xil_AssertNonvoidAlways()
```

Always assert. This assert macro is to be used for functions that return a value. Use for instances where an assert should always occur.

#### Returns:

Returns void unless the xil\_AssertWait variable is true, in which case no return is made and an infinite loop is entered.

#### Extra Header File

The xil hal.h header file is provided as a convenience. It includes all the header files related to the Hardware Abstraction Layer. The contents of this header file are as follows:

```
#ifndef XIL_HAL_H
#define XIL_HAL_H
#include "xil assert.h"
#include "xil exception.h"
#include "xil_cache.h"
#include "xil_io.h"
#include "xil_types.h"
#endif
```

# Test Memory (xil\_testmem)

# **Description**

A subset of the memory tests can be selected or all of the tests can be run in order. If there is an error detected by a subtest, the test stops and the failure code is returned. Further tests are not run even if all of the tests are selected.

# **Subtest Descriptions**

# XIL\_TESTMEM\_ALLMEMTESTS

Runs all of the subtests.

#### XIL\_TESTMEM\_INCREMENT

Incrementing Value test.

This test starts at XIL\_TESTMEM\_MEMTEST\_INIT\_VALUE and uses the incrementing value as the test value for memory.

UG647 October 5, 2016 www.xilinx.com 54





## XIL\_TESTMEM\_WALKONES

Walking Ones test.

This test uses a walking "1" as the test value for memory.

```
location 1 = 0 \times 000000001
location 2 = 0 \times 000000002
```

#### XIL TESTMEM WALKZEROS

Walking Zeros test.

This test uses the inverse value of the walking ones test as the test value for memory.

```
location 1 = 0xfffffffff
location 2 = 0xffffffff
...
```

#### XIL TESTMEM INVERSEADDR

Inverse Address test.

This test uses the inverse of the address of the location under test as the test value for memory.

#### XIL\_TESTMEM\_FIXEDPATTERN

Fixed Pattern test.

This test uses the provided patterns as the test value for memory.

If zero is provided as the pattern, the test uses <code>OxDEADBEEF</code>.

**Caution!** The tests are DESTRUCTIVE. Run them before any initialized memory spaces have been set up. The address provided to the memory tests is not checked for validity, except for the case where the value is NULL. It is possible to provide a code-space pointer for this test to start with and ultimately destroy executable code causing random failures.

**Note:** This test is used for spaces where the address range of the region is smaller than the data width. If the memory range is greater than 2 to the power of width, the patterns used in XIL\_TESTMEM\_WALKONES and XIL\_TESTMEM\_WALKZEROS will repeat on a boundary of a power of two, making it more difficult to detect addressing errors. The XIL\_TESTMEM\_INCREMENT and XIL\_TESTMEM\_INVERSEADDR tests suffer the same problem. If you need to test large blocks of memory, it is recommended that you break them up into smaller regions of memory to allow the test patterns used not to repeat over the region tested.

#### **Header File**

```
#include "xil_testmem.h"
```

## **Common API**





int Xil\_Testmem8(u8 \*Addr, u32 Words, u8 Pattern, u8
 Subtest)

Perform a destructive 8-bit wide memory test.

#### Parameters:

Addr is a pointer to the region of memory to be tested.

Words is the length of the block.

Pattern is the constant used for the constant pattern test, if 0, 0xDEADBEEF is used.

Subtest is the test selected. See xil testmem.h for possible values.

#### Returns:

-1 is returned for a failure

0 is returned for a pass

**Note:** Used for spaces where the address range of the region is smaller than the data width. If the memory range is greater than 2 to the power of width, the patterns used in XIL\_TESTMEM\_WALKONES and XIL\_TESTMEM\_WALKZEROS repeat on a boundary of a power of two, which makes detecting addressing errors more difficult. This is true of XIL\_TESTMEM\_INCREMENT and XIL\_TESTMEM\_INVERSEADDR tests also. If you need to test large blocks of memory, it is recommended that you break them up into smaller regions of memory to allow the test patterns used not to repeat over the region tested.

int Xil\_Testmem16(u16 \*Addr, u32 Words, u16 Pattern, u8
 Subtest)

Perform a destructive 16-bit wide memory test.

# Parameters:

Addr is a pointer to the region of memory to be tested.

Words is the length of the block.

Pattern is the constant used for the constant pattern test, if 0, 0xDEADBEEF is used.

Subtest is the test selected. See xil testmem.h for possible values.

#### Returns:

-1 is returned for a failure.

0 is returned for a pass.

**Note:** Used for spaces where the address range of the region is smaller than the data width. If the memory range is greater than 2 to the power of width, the patterns used in XIL\_TESTMEM\_WALKONES and XIL\_TESTMEM\_WALKZEROS repeat on a boundary of a power of two, making detecting addressing errors more difficult.

This is true of XIL\_TESTMEM\_INCREMENT and XIL\_TESTMEM\_INVERSEADDR tests also. If you need to test large blocks of memory, it is recommended that you break them up into smaller regions of memory to allow the test patterns used not to repeat over the region tested.



int Xil\_Testmem32 (u32 \*Addr, u32 Words, u32 pattern, u8
 Subtest)

Perform a destructive 32-bit wide memory test.

#### Parameters:

Addr is a pointer to the region of memory to be tested.

Words is the length of the block.

Pattern is the constant used for the constant pattern test, if 0, 0xDEADBEEF is used. Subtest is the test selected. See xil testmem.h for possible values.

#### Returns:

0 is returned for a pass.

-1 is returned for a failure.

**Note:** This test is used for spaces where the address range of the region is smaller than the data width. If the memory range is greater than 2 to the power of width, the patterns used in XIL\_TESTMEM\_WALKONES and XIL\_TESTMEM\_WALKZEROS repeat on a boundary of a power of two, making detecting addressing errors more difficult. This is true of the XIL\_TESTMEM\_INCREMENT and XIL\_TESTMEM\_INVERSEADDR tests also. If you need to test large blocks of memory, it is recommended that you break them up into smaller regions of memory to allow the test patterns used not to repeat over the region tested.

# Test Register IO (xil\_testio)

This file contains utility functions to teach endian-related memory I/O functions.

#### **Header File**

#include "xil\_testio.h"

#### Common API

```
int Xil_TestIO8(u8 *Addr, int Len, u8 Value)
```

Perform a destructive 8-bit wide register IO test where the register is accessed using Xil\_Out8 and Xil\_In8. the Xil\_TestIO8 function compares the read and write values.

#### Parameters:

Addr is a pointer to the region of memory to be tested.

Len is the length of the block.

Value is the constant used for writing the memory.

#### Returns:

0 is returned for a pass.

-1 is returned for a failure.



int Xil\_TestIO16(u8 \*Addr, int Len, u16 Value, int Kind,
 int Swap)

Perform a destructive 16-bit wide register IO test. Each location is tested by sequentially writing a 16-bit wide register, reading the register, and comparing value. This function tests three kinds of register IO functions, normal register IO, little-endian register IO, and bigendian register IO. When testing little/big-endian IO, the function performs the following sequence:

Xil\_Out16LE/Xil\_Out16BE, Xil\_In16, Compare In-Out values, Xil\_Out16, Xil\_In16LE/Xil\_In16BE, Compare In-Out values. Whether to swap the read-in value before comparing is controlled by the 5th argument.

#### Parameters:

Addr is a pointer to the region of memory to be tested.

Len is the length of the block.

Value is the constant used for writing the memory.

Kind is the test kind. Acceptable values are: XIL\_TESTIO\_DEFAULT, XIL\_TESTIO\_LE, XIL\_TESTIO\_BE.

Swap indicates whether to byte swap the read-in value.

#### Returns:

0 is returned for a pass.

-1 is returned for a failure.

```
int Xil_TestIO32(u8 *Addr, int Len, u32 Value, int Kind,
   int Swap)
```

Perform a destructive 32-bit wide register IO test. Each location is tested by sequentially writing a 32-bit wide register, reading the register, and comparing value. This function tests three kinds of register IO functions, normal register IO, little-endian register IO, and big-endian register IO. When testing little/big-endian IO, the function performs the following sequence:

Xil\_Out32LE/Xil\_Out32BE, Xil\_In32, Compare In-Out values, Xil\_Out32,

Xil\_In32LE/Xil\_In32BE, Compare In-Out values. Whether to swap the read-in value before comparing is controlled by the 5th argument.

#### Parameters:

Addr is a pointer to the region of memory to be tested.

Len is the length of the block.

Value is the constant used for writing the memory.

Kind is the test kind. Acceptable values are: XIL\_TESTIO\_DEFAULT, XIL\_TESTIO\_LE, XIL\_TESTIO\_BE.

 ${\tt Swap}$  indicates whether to byte swap the read-in value.

www.xilinx.com

#### Returns:

0 is returned for a pass.

-1 is returned for a failure.



# Test Cache (xil\_testcache)

This file contains utility functions to test the cache.

#### **Header File**

#include "xil\_testcache.h"

#### Common API

# int Xil\_TestDCacheAll()

Tests the DCache related functions on all related API tests such as Xil\_DCacheFlush and Xil\_DCacheInvalidate. This test function writes a constant value to the data array, flushes the DCache, writes a new value, and then invalidates the DCache.

#### Returns:

0 is returned for a pass.

-1 is returned for a failure.

# int Xil\_TestDCacheRange()

Tests the DCache range-related functions on a range of related API tests such as Xil\_DCacheFlushRange and Xil\_DCacheInvalidate\_range. This test function writes a constant value to the data array, flushes the range, writes a new value, and then invalidates the corresponding range.

#### Returns:

0 is returned for a pass.

-1 is returned for a failure.

# int Xil\_TestICacheAll()

Perform xil icache invalidate().

## Returns:

0 is returned for a pass. The function will hang if it fails.

# int Xil\_TestICacheRange()

Perform Xil\_ICacheInvalidateRange() on a few function pointers.

## Returns:

0 is returned for a pass. The function will hang if it fails.



# **Hardware Abstraction Layer Migration Tips**

# **Mapping Header Files to HAL Header Files**

You should map the old header files to the new HAL header files as listed in Table 8.

Table 8: HAL Header File Mapping

| Area                             | Old Header File                   | New Header File                                          |
|----------------------------------|-----------------------------------|----------------------------------------------------------|
| Register IO                      | "xio.h"                           | "xil_io.h"                                               |
| Exception                        | "xexception_l.h" "mb_interface.h" | "xil_exception.h"                                        |
| Cache                            | "xcache.h" "mb_interface.h"       | "xil_cache.h"                                            |
| Interrupt                        | "xexception_l.h" "mb_interface.h" | "xil_exception.h"                                        |
| Typedef u8 u16 u32               | "xbasic_types.h"                  | "xil_types.h"                                            |
| Typedef of Xuint32 Xfloat32      | "xbasic_types.h"                  | Deprecated. Do not use.                                  |
| Assert                           | "xbasic_types.h"                  | "xil_assert.h"                                           |
| Test Memory                      | "xutil.h"                         | "xil_testmem.h"                                          |
| Test Register IO                 | None                              | "xil_testio.h"                                           |
| Test Cache                       | None                              | "xil_testcache.h"                                        |
| XTRUE, XFALSE, XNULL definitions | "xbasic_types.h"                  | Deprecated. Use TRUE, FALSE, NULL defined in xil_types.h |



# **Mapping Functions to HAL Functions**

You should map old functions to the new HAL functions as follows.

# Table 9: I/O Function Mapping

| Old xio          | New xil io          |
|------------------|---------------------|
|                  |                     |
| #include "xio.h" | #include "xil_io.h" |
| XIo_In8          | Xil_In8             |
| XIo_Out8         | Xil_Out8            |
| XIo_In16         | Xil_In16            |
| XIo_Out16        | Xil_Out16           |
| XIo_In32         | Xil_In32            |
| XIo_Out32        | Xil_Out32           |

# Table 10: Exception Function Mapping

| Old xexception                                             | New Xil_exception                             |
|------------------------------------------------------------|-----------------------------------------------|
| #include "xexception_I.h"<br>#include "mb_interface.h"     | #include "xil_exception.h"                    |
| XExc_Init                                                  | Xil_ExceptionInit                             |
| XExc_mEnableException microblaze_enable_exceptions         | For all processors: Xil_ExceptionEnable(void) |
| XExc_registerHandler microblaze_register_exception_handler | Xil_ExceptionRegisterHandler                  |
| XExc_RemoveHandler                                         | Xil_ExceptionRemoveHandler                    |
| XExc_mDisableExceptions microblaze_disable_exceptions      | Xil_ExceptionDisable                          |

# Table 11: Interrupt Function Mapping

| Old Interrupt                                                                             | New Xil_interrupt                                            |
|-------------------------------------------------------------------------------------------|--------------------------------------------------------------|
| #include "xexception_l.h"                                                                 | #include "xil_exception.h"                                   |
| XExc_RegisterHandler(XEXC_ID_NON_CRITI CAL, handler) microblaze_register_handler(handler) | Xil_ExceptionRegisterHandler(XIL_EXCEPTIO N_ID_INT, handler) |



Table 12: Cache Function Mapping

| Old xcache                                                      | New xil_cache                              |
|-----------------------------------------------------------------|--------------------------------------------|
| #include "Xcache_l.h"<br>#include "mb_interface.h"              | #include "xil_cache.h"                     |
| XCache_EnableDCache microblaze_enable_dcache                    | For all processors: Xil_DCacheEnable(void) |
| XCache_DisableDCache microblaze_disable_dcache                  | Xil_DCacheDisable                          |
| XCache_InvalidateDCacheRange microblaze_invalidate_dcache_range | Xil_DCacheInvalidateRange                  |
| microblaze_invalidate_dcache                                    | Xil_DCacheInvalidate                       |
| XCache_FlushDCacheRange microblaze_flush_dcache_range           | Xil_DCacheFlushRange                       |
| microblaze_flush_dcache                                         | Xil_DCacheFlush                            |
| XCache_EnablelCache microblaze_enable_icache                    | For all processors: Xil_ICacheEnable(void) |
| XCache_DisableICache microblaze_disable_icache                  | Xil_ICacheDisable                          |
| XCache_InvalidateICacheRange microblaze_invalidate_icache_Range | Xil_ICacheInvalidateRange                  |
| microblaze_invalidate_icache                                    | Xil_ICacheInvalidate                       |

# Table 13: Assert Function Mapping

| Old ASSERT                | New xil_assert          |
|---------------------------|-------------------------|
| #include "xbasic_types.h" | #include "xil_assert.h" |
| XAssert                   | Xil_Assert              |
| XASSERT_VOID              | Xil_AssertVoid          |
| XASSERT_NONVOID           | Xil_AssertNonvoid       |
| XASSERT_VOID_ALWAYS       | Xil_AssertVoidAlways    |
| XASSERT_NONVOID_ALWAYS    | Xil_AssertNonvoidAlways |
| XAssertSetCallback        | Xil_AssertSetCallback   |

# Table 14: Memory Test Function Mapping

| Old XUtil_Memtest  | New xil_testmem          |  |
|--------------------|--------------------------|--|
| #include "xutil.h" | #include "xil_testmem.h" |  |
| XUtil_MemoryTest32 | Xil_Testmem32            |  |
| XUtil_MemoryTest16 | Xil_Testmem16            |  |
| XUtil_MemoryTest8  | Xil_Testmem8             |  |

UG647 October 5, 2016 <a href="https://www.xilinx.com">www.xilinx.com</a> Send Feedback 62



# Program Profiling

The Standalone OS supports program profiling in conjunction with the GNU compiler tools and the Xilinx Microprocessor Debugger (XMD).

**Note:** XMD has been deprecated and will be removed in future versions of the Xilinx Software Development Kit (SDK). XSDB replaces XMD and provides additional functionality. We recommend you switch to XSDB for command-line debugging.

Profiling a program running on a hardware (board) provides insight into program execution and identifies where execution time is spent. The interaction of the program with memory and other peripherals can be more accurately captured.

Program running on hardware target is profiled using *software intrusive* method. In this method, the profiling software code is instrumented in the user program. The profiling software code is a part of the libxil. a library and is generated when software intrusive profiling is enabled in Standalone. For more details on about profiling, refer to the "Profile Overview" section of the *SDK Help*.

When the **-pg** profile option is specified to the compiler (mb-gcc), the profiling functions are linked with the application to profile automatically. The generated executable file contains code to generate profile information.

Upon program execution, this instrumented profiling function stores information on the hardware. The profile information can be read by the GNU gprof tool. The program functionality remains unchanged but it slows down the execution.

**Note:** The profiling functions do not have any explicit application API. The library is linked due to profile calls (\_mcount) introduced by GCC for profiling.

# **Profiling Requirements**

- Software intrusive profiling requires memory for storing profile information. You can use any memory in the system for profiling.
- A timer is required for sampling instruction address. The axi\_timer is the supported profile timer for MicroBlaze processors. The Global Timer is used as the profile timer for Cortex A9 processors.

# **Profiling Functions**

## \_profile\_init

Called before the application main() function. Initializes the profile timer routine and registers timer handler accordingly, based on the timer used, connects to the processor, and starts the timer. The Tcl routine of Standalone library determines the timer type and the connection to processor, then generates the #defines in the profile\_config.h file.

Refer to the "Microprocessor Library Definition (MLD)" chapter in the *Embedded System Tools Reference Manual (UG111)*, which is available in the installation directory. A link to this document is also provided in "MicroBlaze Processor API," page 1.

#### mcount

Called by the \_mcount function, which is inserted at every function start by gcc. Records the *caller* and *callee* information (Instruction address), which is used to generate call graph information.

63



## \_profile\_intr\_handler

The interrupt handler for the profiling timer. The timer is set to sample the executing application for PC values at fixed intervals and increment the Bin count. This function is used to generate the histogram information.

# Configuring the Standalone OS

You can configure the Standalone OS using the Board Support Package Settings dialog box in SDK.

Table 15 lists the configurable parameters for the Standalone OS.

**Table 15: Configuration Parameters** 

| Parameter                         | Туре                   | Default<br>Value | Description                                                                                                                                                                                                                           |
|-----------------------------------|------------------------|------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| enable_sw_intrusive_<br>profiling | Bool                   | false            | Enable software intrusive profiling functionality. Select true to enable.                                                                                                                                                             |
| profile_timer                     | Peripheral<br>Instance | none             | Specify the timer to use for profiling. Select an axi_timer from the list of displayed instances. For Cortex A9, the ARM Cortex-A9 Private timer is used.                                                                             |
| stdin                             | Peripheral<br>Instance | none             | Specify the <b>STDIN</b> peripheral from the drop down list                                                                                                                                                                           |
| stdout                            | Peripheral<br>Instance | none             | Specify the <b>STDOUT</b> peripheral from the drop down list.                                                                                                                                                                         |
| predecode_fpu_exception           | Bool                   | false            | This parameter is valid only for MicroBlaze processor when FPU exceptions are enabled in the hardware. Setting this to true will include extra code that decodes and stores the faulting FP instruction operands in global variables. |

# MicroBlaze MMU Example

The tlb\_add function adds a single TLB entry. The parameters are:

| tlbindex | The index of the TLB entry to be updated. |
|----------|-------------------------------------------|
| tlbhi    | The value of the TLBHI field.             |
| tlblo    | The value of the TLBLO field.             |



Given a base and high address, the tlb\_add\_entries function figures the minimum number page mappings/TLB entries required to cover the area. This function uses recursion to figure the entire range of mappings.

#### Parameters:

| base      | The base address of the region of memory                                                                   |
|-----------|------------------------------------------------------------------------------------------------------------|
| high      | The high address of the region of memory                                                                   |
| tlbaccess | The type of access required for this region of memory. It can be a logical or -ing of the following flags: |
|           | 0 indicates read-only access  TLB_ACCESS_EXECUTABLE means the region is executable                         |
|           | TLB_ACCESS_WRITABLE means the region is writable                                                           |

#### Returns: 1 on success and 0 on failure

```
static int tlb_add_entries (unsigned int base, unsigned int high, unsigned
int tlbaccess)
    int sizeindex, tlbsizemask;
    unsigned int tlbhi, tlblo;
    unsigned int area_base, area_high, area_size;
    static int tlbindex = 0;
    // Align base and high to 1KB boundaries
   base = base & 0xfffffc00;
   high = (high >= 0xfffffc00) ? 0xfffffffff : ((high + 0x400) & 0xfffffc00)
- 1;
    // Start trying to allocate pages from 16 MB granularity down to 1 KB
    area_size = 0x1000000;
                                         // 16 MB
    tlbsizemask = 0x380;
                                          // \text{TLBHI[SIZE]} = 7 (16 \text{ MB})
    for (sizeindex = 7; sizeindex >= 0; sizeindex--) {
        area_base = base & sizemask[sizeindex];
        area_high = area_base + (area_size - 1);
        // if (area_base <= (0xffffffff - (area_size - 1))) {</pre>
        if ((area_base >= base) && (area_high <= high)) {</pre>
            if (tlbindex < TLBSIZE) {</pre>
           tlbhi = (base & sizemask[sizeindex]) | tlbsizemask | 0x40;
// TLBHI: TAG, SIZE, V
            tlblo = (base & sizemask[sizeindex]) | tlbaccess | 0x8;
// TLBLO: RPN, EX, WR, W
                tlb_add (tlbindex, tlbhi, tlblo);
                tlbindex++;
            } else {
                // We only handle the 64 entry UTLB management for now
                return 0;
            // Recursively add entries for lower area
            if (area_base > base)
                if (!tlb_add_entries (base, area_base - 1, tlbaccess))
                     return 0;
```



```
// Recursively add entries for higher area
if (area_high < high)
    if (!tlb_add_entries(area_high + 1, high, tlbaccess))
        return 0;

break;

break;

// else, we try the next lower page size
    area_size = area_size >> 2;
    tlbsizemask = tlbsizemask - 0x80;
}
return 1;
}
```

# For a complete example, refer to:

 $$XILINX\_SDK/data/embeddedsw/lib/bsp/xilkernel\_v6\_1/src/src/arch/microblaze/mpu.c.$ 

