# Application Note Migrating Software From 8-Bit (Byte) Addressable CPU's to C28x CPU



Veena Kamath

#### ABSTRACT

While developing applications, you may assume that the device runs on a 8-bit addressable device and can face issues while porting the code to a 16-bit addressable device. This application note discusses such common scenarios and provides a guide on how to develop application irrespective of the addressability.

# **Table of Contents**

| 1 Introduction                                                      | 2 |
|---------------------------------------------------------------------|---|
| 2 Byte vs Word Terminology                                          | 2 |
| 3 Key Points to Consider                                            |   |
| 3.1 8-Bit Data Types are not Supported                              |   |
| 3.2 Memory Size is Expressed in 16 Bits                             |   |
| 3.3 Arrays and Structures: Individual Element Offsets are Different |   |
| 3.4 Difference in Standard Data Type Widths                         |   |
| 3.5 Dealing With 8-Bit Communication Protocols                      | 6 |
| 4 References                                                        | 6 |
| 5 Revision History                                                  |   |
| -                                                                   |   |

# List of Figures

| Figure 3-1. Memory Allocation in C28x Core | 5 |
|--------------------------------------------|---|
| Figure 3-2. Memory Allocation in Arm Core  | 5 |

# **List of Tables**

Table 2-1. Memory Map Table...... 2

#### Trademarks

TMS320<sup>™</sup> and Code Composer Studio<sup>™</sup> are trademarks of Texas Instruments. Arm<sup>®</sup> is a registered trademark of Arm Limited (or its subsidiaries) in the US and/or elsewhere. All trademarks are the property of their respective owners.

1



# **1** Introduction

The TMS320C28x is one of several fixed-point CPUs in the TMS320<sup>™</sup> family. It has a 16-bit addressable architecture compared to 8-bit addressable architecture like Arm<sup>®</sup> CPUs.

In a 8-bit addressable architecture, every 8 bits of data has a unique address, where as in a 16-bit addressable device, every 16 bits of data has a unique address. Hence, in a 16-bit architecture, the smallest addressable unit of memory and the smallest data type supported is 16 bits.

# 2 Byte vs Word Terminology

Historically, byte is defined as the smallest addressable unit of memory. Hence, technically, the size of a byte is hardware dependent: 16 bits for C28x devices and 8 bits for Arm devices. But nowadays, byte is used as a synonym of 8 bits, since majority of the device architectures are 8 bit addressable. To avoid the confusion, let us use the terminology 8-bit bytes and 16-bit words

Byte and Word Usage in the device feature set Documentation:



From the C28x compiler's perspective, the memory size is always expressed in smallest addressable units, which is 16 bits. This includes the memory length, code/data size provided in the linker cmd files, .map file, and so forth. The standard sizeof() function also returns the size in 16 bits.

• Memory Map Table in device-specific data sheet:

| Table 2-1. Memory Map Table |
|-----------------------------|
|-----------------------------|

| Memory  | Size    | Start Address | End Address |
|---------|---------|---------------|-------------|
| LS0 RAM | 2K × 16 | 0x0000 8000   | 0x0000 87FF |
| LS1 RAM | 2K × 16 | 0x0000 8800   | 0x0000 8FFF |

• Linker command file:

```
RAMLS0 : origin = 0x00008000, length = 0x0800
RAMLS1 : origin = 0x00008800, length = 0x0800
```

#### • .map file generated by Code Composer Studio<sup>™</sup> (CCS):

| MEMORY CONFIGURATIONS |          |          |          |          |      |
|-----------------------|----------|----------|----------|----------|------|
| name                  | origin   | length   | used     | unused   | attr |
| RAMLSO                | 00008000 | 00000800 | 00000112 | 000006ee | RWIX |
| RAMLS1                | 00088000 | 00000800 | 00000194 | 0000066c | RWIX |

• sizeof(uint32\_t) =  $2 \rightarrow 2 * 16$  bits



# **3 Key Points to Consider**

#### 3.1 8-Bit Data Types are not Supported

In C28x CPU-based projects, 8-bit data types are not supported. char is 16 bits wide and the types uint8\_t and int8\_t are not defined by the C28x compiler. C2000Ware remaps these to uint16\_t and int16\_t data types. For more details on the data types, see the *TMS320C28x Optimizing C/C++ Compiler v22.6.0.LTS User's Guide*.

However, the C28x compiler provides intrinsic \_\_byte() for byte accesses. For more details, see the https:// software-dl.ti.com/ccs/esd/documents/c2000 byte-accesses-with-the-c28x-cpu.html.

• While porting an application from an 8-bit addressable architecture, such as Arm to C28x, you may find an increase in memory requirement due to this difference.

Example:

```
struct
{
    uint8_t a;
    uint8_t b;
    uint16_t c;
} myStruct;
```

In an Arm device, the size of myStruct would be 8 + 8 + 16 = 32 bits. Whereas, in C28x, the size would be 16 + 16 + 16 = 48 bits.

Since the int8 or char data types in C28x is 16 bit wide, the compiler will not do a wraparound at 0xFF while
performing arithmetic or shift operations.

Example:

```
uint8_t a = 0xFF;
a += 1;
if (a == 0)
{
    //Condition is true for Arm and false for C28x
}
```

While converting larger data types to smaller ones or vise versa, care should be taken to properly typecast, use masks, and be aware of the device endianness. The C28x-based devices are little endian devices.
 uint8\_t a[4] = {0x1, 0x2, 0x3, 0x4};

```
The result varies with endianness as well. C28x is little endian device and
uint32_t b1 = *(uint32_t *)a; ------ 🗶
                                                b1 = 0x00020001. In ARM little endian device, b1 = 0x04030201
                                                Shifting beyond the data type size. The behavior is compiler specific. In the C28x
uint32_t b2 = (a[0] << 0)
                                ----- 🗴
                                                compiler, the value is truncated if shifted beyond its size. It also throws a warning
                (a[1] << 8)
                (a[2] << 16)
                (a[3] << 24);
uint32_t b2 = ((uint32_t)a[0] << 0) | \checkmark Recommended usage of typecasts and shift operators.
                                                b3 = 0x04030201 in both C28x and ARM-LE
                ((uint32_t)a[1] << 8)
                ((uint32_t)a[2] << 16)
                ((uint32_t)a[3] << 24);
uint32 t a = 0x12345678;
uint8 t b1 = a; ----- 🗴 b1 = 0x5678 in C28x, 0x78 in ARM
uint8_t b2 = (uint8_t)a; ----- 🗴 b2 = 0x5678 in C28x, 0x78 in ARM
uint8_t b3 = a & 0xFF; ------ 
    Recommended usage of masks. b3 = 0x78 in both C28x and ARM
```

```
Usage of unions needs to be revisited.
            union
            {
                struct
                {
                                                  The total size of the union will be 64 bits in C28x and 32 bits in ARM
                     uint8_t byte1;
                                              x
                                                 b1=1, b2=1, b3 =1, b4 =1 → word = 0x00010001 in C28x
                     uint8_t byte2;
                     uint8_t byte3;
                                                                        \rightarrow word = 0x01010101 in ARM
                     uint8_t byte4;
                };
                uint32_t word;
            }test;
            union
            {
                struct
                {
                                              ✓ Recommended usage of bitfields
                     uint8_t byte1 : 8;
                     uint8_t byte2 : 8;
                                                 b1=1, b2=1, b3 =1, b4 =1 \rightarrow word = 0x01010101 in C28x and ARM
                     uint8_t byte3 : 8;
                     uint8_t byte4 : 8;
                };
                uint32_t word;
            }test;
```

#### 3.2 Memory Size is Expressed in 16 Bits

The memory sizes mentioned in the linker cmd files, .map files are all expressed in 16 bits. The sizeof() function always returns the size with respect to smallest addressable memory units (8 bits for Arm and 16 bits for C28x).

Be careful to not hardcode the size information in the application. One common scenario is while using memset/cpy/cmp functions, the size parameter to these functions are in terms of smallest addressable units.

| <pre>struct {     uint16_t a;     uint16_t b;     uint32_t c; } myStruct;</pre> | The total size of the struct is 64 bits in C28x and in ARM.<br>But sizeof(myStruct) = 4 in C28x, and 8 in ARM |
|---------------------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------|
| memset(&myStruct, 0xA, 8); 🗴                                                    | Hardcoded memory size – In the case of C28x, corrupts the adjacent memory                                     |
| <pre>memset(&amp;myStruct, 0xA, sizeof(myStruct)); *</pre>                      | Recommended usage of sizeof() instead of assuming it as 8                                                     |

Also, note the difference in data packing after memset usage:

| C28x              | Arm               |
|-------------------|-------------------|
| myStruct          | myStruct          |
| 000A000A 000A000A | 0A0A0A0A 0A0A0A0A |

#### 3.3 Arrays and Structures: Individual Element Offsets are Different

• In C28x, there is a unique address for every 16 bits, as opposed to 8 bits in an Arm device.

Example:

| uint32 t | Array32[4] | = {1,2,3,4}; |
|----------|------------|--------------|
| uint16 t | Array16[4] | = {1,2,3,4}; |
| uint8 t  | Array8[4]  | = {1,2,3,4}; |

| Expression  | Туре             | Value     | Address         |                         |
|-------------|------------------|-----------|-----------------|-------------------------|
| 🗸 🥭 Array32 | unsigned long[4] | [1,2,3,4] | 0x0000A800@Data |                         |
| (×)= [0]    | unsigned long    | 1         | 0x0000A800@Data | )                       |
| (×)= [1]    | unsigned long    | 2         | 0x0000A802@Data | Address increments      |
| (×)= [2]    | unsigned long    | 3         | 0x0000A804@Data | by 2 for a 32-bit array |
| (×)= [3]    | unsigned long    | 4         | 0x0000A806@Data | J                       |
| 🗸 🍃 Array16 | unsigned int[4]  | [1,2,3,4] | 0x0000A80E@Data |                         |
| (×)= [0]    | unsigned int     | 1         | 0x0000A80E@Data | )                       |
| (×)= [1]    | unsigned int     | 2         | 0x0000A80F@Data | Address increments      |
| (×)= [2]    | unsigned int     | 3         | 0x0000A810@Data | by 1 for a 16-bit array |
| (×)= [3]    | unsigned int     | 4         | 0x0000A811@Data | J                       |
| 🗸 🥭 Array8  | unsigned int[4]  | [1,2,3,4] | 0x0000A812@Data |                         |
| (×)= [0]    | unsigned int     | 1         | 0x0000A812@Data | )                       |
| (×)= [1]    | unsigned int     | 2         | 0x0000A813@Data | uint8_t = uint16_t      |
| (×)= [2]    | unsigned int     | 3         | 0x0000A814@Data | Address increments by 1 |
| (×)= [3]    | unsigned int     | 4         | 0x0000A815@Data | J                       |

#### Figure 3-1. Memory Allocation in C28x Core

| Expression    | Туре              | Value         | Address    |                         |
|---------------|-------------------|---------------|------------|-------------------------|
| ✓ (=) Array32 | unsigned int[4]   | [1,2,3,4]     | 0x2000C000 |                         |
| (×)= [0]      | unsigned int      | 1             | 0x2000C000 | 1                       |
| (×)= [1]      | unsigned int      | 2             | 0x2000C004 | Address increments      |
| (×)= [2]      | unsigned int      | 3             | 0x2000C008 | by 4 for a 32-bit array |
| (×)= [3]      | unsigned int      | 4             | 0x2000C00C | J                       |
| 🗸 😑 Array16   | unsigned short[4] | [1,2,3,4]     | 0x2000C010 |                         |
| (×)= [0]      | unsigned short    | 1             | 0x2000C010 | ]                       |
| (×)= [1]      | unsigned short    | 2             | 0x2000C012 | Address increments      |
| (×)= [2]      | unsigned short    | 3             | 0x2000C014 | by 2 for a 16-bit array |
| (×)= [3]      | unsigned short    | 4             | 0x2000C016 | J                       |
| 🗸 🥭 Array8    | unsigned char[4]  | [1 '\x01',2 ' | 0x2000C018 |                         |
| (×)= [0]      | unsigned char     | 1 '\x01'      | 0x2000C018 | )                       |
| (×)= [1]      | unsigned char     | 2 '\x02'      | 0x2000C019 | Address increments      |
| (×)= [2]      | unsigned char     | 3 '\x03'      | 0x2000C01A | by 1 for a 8-bit array  |
| (×)= [3]      | unsigned char     | 4 '\x04'      | 0x2000C01B | J                       |

## Figure 3-2. Memory Allocation in Arm Core

• While using arrays, the index-based accesses and pointer increment-based access would yield the same results in both the devices. But if you try to use hardcoded offsets, it would produce different results.

| uint32_t read1 = Array32[2];                                                                                                             |  |
|------------------------------------------------------------------------------------------------------------------------------------------|--|
| uint32_t read2 = *(Array32 + 2);                                                                                                         |  |
| uint32_t read3 = *((uint32_t *)(baseAddr + 8)); × Hardcoded offset value – Gives different results in C28x and ARM                       |  |
| uint32_t read4 = *((uint32_t *)(baseAddr + 2*sizeof(uint32_t))); Recommended usage of sizeof() instead of assuming it as 4</td <td></td> |  |

 Same is applicable for structs. While struct.element provides the same results in both devices, the usage of (baseaddr + offset) to access struct.



# 3.4 Difference in Standard Data Type Widths

As opposed to any 8-bit addressable architecture, the sizes of int and char are different in C28x devices. For better portability, it is highly recommended to use the width-based data types such as uint16 t, int16 t, uint32 t, int32 t, uint64 t, int64 t, float32 t, float64 t, and so forth. These data types are defined in the C28 compiler header file stdint.h.

Note that the data types uint8 t and int8 t are not defined by the C28x compiler. C2000Ware remaps these to uint16\_t and int16\_t data types, respectively.

| Туре         | Size    |
|--------------|---------|
| char         | 16 bits |
| Bool         | 16 bits |
| short        | 16 bits |
| int          | 16 bits |
| long         | 32 bits |
| long long    | 64 bits |
| float        | 32 bits |
| double(COFF) | 32 bits |
| double(EABI) | 64 bits |
| long double  | 64 bits |
| Pointers     | 32 bits |

For more details on the data types, see the TMS320C28x Optimizing C/C++ Compiler v22.6.0.LTS User's Guide .

#### 3.5 Dealing With 8-Bit Communication Protocols

Communication peripherals like the Controller Area Network (CAN), Serial Communications Interface (SCI) and Universal Asynchronous Receiver/Transmitter (UART) in C28x devices support 8-bit data transfers. But note that the data would be stored as 16 bits in the memory. The C28x compiler does not support pragma pack(1).

Example:



In this example, the function takes in uint16 t\* as the input data parameter. But it only expects 8 bits of data per uint16 t type. The upper half of the uint16 t is ignored and only the lower 8 bits are transferred.

## 4 References

- TMS320C28x Optimizing C/C++ Compiler v22.6.0.LTS User's Guide
- https://software-dl.ti.com/ccs/esd/documents/c2000 byte-accesses-with-the-c28x-cpu.html



# **5 Revision History**

NOTE: Page numbers for previous revisions may differ from page numbers in the current version.

| Changes from Revision * (March 2023) to Revision A (April 2023) |                                                                                               | Page |
|-----------------------------------------------------------------|-----------------------------------------------------------------------------------------------|------|
| •                                                               | Updated the numbering format for tables, figures and cross-references throughout the document | 2    |
| •                                                               | Updated Section 2                                                                             | 2    |
|                                                                 | Updated Section 3.1                                                                           |      |
|                                                                 | Updated Section 3.2.                                                                          |      |
|                                                                 |                                                                                               |      |

7

# IMPORTANT NOTICE AND DISCLAIMER

TI PROVIDES TECHNICAL AND RELIABILITY DATA (INCLUDING DATA SHEETS), DESIGN RESOURCES (INCLUDING REFERENCE DESIGNS), APPLICATION OR OTHER DESIGN ADVICE, WEB TOOLS, SAFETY INFORMATION, AND OTHER RESOURCES "AS IS" AND WITH ALL FAULTS, AND DISCLAIMS ALL WARRANTIES, EXPRESS AND IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT OF THIRD PARTY INTELLECTUAL PROPERTY RIGHTS.

These resources are intended for skilled developers designing with TI products. You are solely responsible for (1) selecting the appropriate TI products for your application, (2) designing, validating and testing your application, and (3) ensuring your application meets applicable standards, and any other safety, security, regulatory or other requirements.

These resources are subject to change without notice. TI grants you permission to use these resources only for development of an application that uses the TI products described in the resource. Other reproduction and display of these resources is prohibited. No license is granted to any other TI intellectual property right or to any third party intellectual property right. TI disclaims responsibility for, and you will fully indemnify TI and its representatives against, any claims, damages, costs, losses, and liabilities arising out of your use of these resources.

TI's products are provided subject to TI's Terms of Sale or other applicable terms available either on ti.com or provided in conjunction with such TI products. TI's provision of these resources does not expand or otherwise alter TI's applicable warranties or warranty disclaimers for TI products.

TI objects to and rejects any additional or different terms you may have proposed.

Mailing Address: Texas Instruments, Post Office Box 655303, Dallas, Texas 75265 Copyright © 2023, Texas Instruments Incorporated