Motivation
In this blog, we’ll explore how to add a custom section in the linker script to store image information, such as the firmware version, checksum, and build details. We’ll also discuss how to access this information from both the bootloader and the main application, ensuring effective validation and diagnostics.
Why Image Information Matters in Embedded Systems
When working with embedded systems, particularly in scenarios involving firmware updates or bootloader applications, having metadata about the image is crucial. This metadata can include:
- Version Information: To track updates and ensure compatibility.
- Checksum: To verify the integrity of the image and detect corruption.
- Build Date/Time: Useful for identifying when the firmware was built, aiding in debugging or compliance tracking.
- Image Size: To check if the correct size of the firmware has been loaded and manage memory effectively.
Storing this information in a dedicated section of memory makes it easily accessible for both the bootloader and the main application.
Understanding ELF Sections: .text, .data, and .bss
In the context of ELF files, different sections serve specific purposes:
-
.text
Section: This section contains the executable code of the program. It is typically loaded into memory as read-only, preventing any modifications during execution. The.text
section is where the actual instructions that the CPU executes reside. -
.data
Section: This section holds initialized global and static variables. The data in this section is loaded into RAM when the program starts. Unlike the.text
section, the.data
section is writable, allowing the program to modify its variables. -
.bss
Section: The BSS (Block Started by Symbol) section contains uninitialized global and static variables. This section is not loaded with any data but reserves space in memory for these variables. When the program starts, the OS initializes this section to zero.
we will try to add a custom section to store image information to make it accessible to both the bootloader and the main application.
Step 1: Defining the Image Information Struct
Using a structured data type (struct) in C is the best way to represent this metadata:
|
|
This struct encapsulates all necessary metadata. Here’s how you can initialize and place this data in a custom section in C:
|
|
Explanation:
__attribute__((section(".image_info")))
: Directs the linker to place theimage_info
variable in the.image_info
section.__DATE__
and__TIME__
: These macros insert the build date and time.
Step 2: Modifying the Linker Script to Include a Custom Section
The linker script defines how the binary is laid out in memory. Here’s how to modify it to include the .image_info
section:
This is our memory layout after adding the custom section:
Linker Script After Modification:
|
|
Explanation:
KEEP(*(.image_info))
: Ensures the section is retained in the final binary.
Step 3: Accessing the Image Information in Another Program
To access the metadata from the main application, declare an external reference to the image_info
struct:
|
|
Linker Script for the Application:
Ensure the application’s linker script includes the .image_info
section:
|
|
Explanation:
NOLOAD
: Indicates that this section will not be loaded by the linker because it already exists in FLASH.
Step 4: Verifying the Section Placement
To verify the placement of the .image_info
section, use tools like objdump
:
|
|
Real-World Use Cases
Firmware Updates:
The bootloader can compare the version and checksum of a new firmware image before allowing it to overwrite the existing one.
Diagnostics:
The bootloader can transmit metadata for diagnostics, providing details about the current firmware version and build information.
Conclusion
By adding a custom section in the linker script for image information, you provide a reliable and accessible way to store important metadata about the firmware image. This approach enhances validation, version control, and diagnostics, making embedded systems more robust and maintainable.