-
Notifications
You must be signed in to change notification settings - Fork 114
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
some tiny improvements and way more comments #162
base: develop
Are you sure you want to change the base?
Changes from all commits
4723ae7
6435cd2
1083773
8481962
dea5ad7
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -11,49 +11,88 @@ | |
#include <cstring> | ||
#include <cstdarg> | ||
#include <algorithm> | ||
#include <cstring> | ||
#include <memory> | ||
|
||
#include "elf2uf2.h" | ||
#include "errors.h" | ||
|
||
// Size of a flash sector in bytes | ||
#define FLASH_SECTOR_ERASE_SIZE 4096u | ||
|
||
// Global variable to control verbose mode | ||
static bool verbose; | ||
|
||
/** | ||
* @brief Terminates the program with a read error. | ||
*/ | ||
static void fail_read_error() { | ||
fail(ERROR_READ_FAILED, "Failed to read input file"); | ||
} | ||
|
||
/** | ||
* @brief Terminates the program with a write error. | ||
*/ | ||
static void fail_write_error() { | ||
fail(ERROR_WRITE_FAILED, "Failed to write output file"); | ||
} | ||
|
||
/** | ||
* @struct page_fragment | ||
* @brief Represents a fragment of a page with file offset and size. | ||
*/ | ||
struct page_fragment { | ||
page_fragment(uint32_t file_offset, uint32_t page_offset, uint32_t bytes) : file_offset(file_offset), page_offset(page_offset), bytes(bytes) {} | ||
uint32_t file_offset; | ||
uint32_t page_offset; | ||
uint32_t bytes; | ||
/** | ||
* @brief Constructor for page_fragment. | ||
* @param file_offset Offset in the input file. | ||
* @param page_offset Offset within the page. | ||
* @param bytes Number of bytes in the fragment. | ||
*/ | ||
page_fragment(uint32_t file_offset, uint32_t page_offset, uint32_t bytes) | ||
: file_offset(file_offset), page_offset(page_offset), bytes(bytes) {} | ||
|
||
uint32_t file_offset; ///< Offset in the input file. | ||
uint32_t page_offset; ///< Offset within the page. | ||
uint32_t bytes; ///< Number of bytes in the fragment. | ||
}; | ||
|
||
/** | ||
* @brief Checks if an address and the size is within the valid range. | ||
* | ||
* @param valid_ranges Valid address ranges. | ||
* @param addr Physical address to check. | ||
* @param vaddr Virtual address. Only used in verbose mode. | ||
* @param size Size of the area to check. | ||
* @param uninitialized Flag indicating if the area is uninitialized. | ||
* @param ar Reference to an address_range that will be set on success. | ||
* @return 0 on success, error code otherwise. | ||
*/ | ||
int check_address_range(const address_ranges& valid_ranges, uint32_t addr, uint32_t vaddr, uint32_t size, bool uninitialized, address_range &ar) { | ||
for(const auto& range : valid_ranges) { | ||
if (range.from <= addr && range.to >= addr + size) { | ||
if (range.type == address_range::type::NO_CONTENTS && !uninitialized) { | ||
fail(ERROR_INCOMPATIBLE, "ELF contains memory contents for uninitialized memory at %p", addr); | ||
fail(ERROR_INCOMPATIBLE, "ELF contains memory contents for uninitialized memory at 0x%08x", addr); | ||
} | ||
ar = range; | ||
if (verbose) { | ||
printf("%s segment %08x->%08x (%08x->%08x)\n", uninitialized ? "Uninitialized" : "Mapped", addr, | ||
addr + size, vaddr, vaddr+size); | ||
printf("%s segment 0x%08x->0x%08x (0x%08x->0x%08x)\n", | ||
uninitialized ? "Uninitialized" : "Mapped", | ||
addr, addr + size, vaddr, vaddr + size); | ||
} | ||
return 0; | ||
} | ||
} | ||
fail(ERROR_INCOMPATIBLE, "Memory segment %08x->%08x is outside of valid address range for device", addr, addr+size); | ||
fail(ERROR_INCOMPATIBLE, "Memory segment 0x%08x->0x%08x is outside of valid address range for device", addr, addr + size); | ||
return ERROR_INCOMPATIBLE; | ||
} | ||
|
||
/** | ||
* @brief Checks ELF32 program header entries and populates the page fragments map. | ||
* | ||
* @param entries Vector of ELF32 program header entries. | ||
* @param valid_ranges Valid address ranges. | ||
* @param pages Map of pages with their fragments. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. "Reference to the page fragments map", to make it more obvious that this is "the page fragments map" to which the |
||
* @return 0 on success, error code otherwise. | ||
*/ | ||
int check_elf32_ph_entries(const std::vector<elf32_ph_entry>& entries, const address_ranges& valid_ranges, std::map<uint32_t, std::vector<page_fragment>>& pages) { | ||
for(const auto & entry : entries) { | ||
if (entry.type == PT_LOAD && entry.memsz) { | ||
|
@@ -68,9 +107,11 @@ int check_elf32_ph_entries(const std::vector<elf32_ph_entry>& entries, const add | |
if (verbose) printf(" ignored\n"); | ||
continue; | ||
} | ||
|
||
unsigned int addr = entry.paddr; | ||
unsigned int remaining = mapped_size; | ||
unsigned int file_offset = entry.offset; | ||
|
||
while (remaining) { | ||
unsigned int off = addr & (UF2_PAGE_SIZE - 1); | ||
unsigned int len = std::min(remaining, UF2_PAGE_SIZE - off); | ||
|
@@ -102,8 +143,18 @@ int check_elf32_ph_entries(const std::vector<elf32_ph_entry>& entries, const add | |
return 0; | ||
} | ||
|
||
/** | ||
* @brief Realizes a page by reading the corresponding fragments from the input stream. | ||
* | ||
* @param in Input stream (e.g., ELF file). | ||
* @param fragments Vector of page fragments. | ||
* @param buf Buffer to store the page data. | ||
* @param buf_len Length of the buffer. | ||
* @return 0 on success, error code otherwise. | ||
*/ | ||
int realize_page(std::shared_ptr<std::iostream> in, const std::vector<page_fragment> &fragments, uint8_t *buf, unsigned int buf_len) { | ||
assert(buf_len >= UF2_PAGE_SIZE); | ||
|
||
for(auto& frag : fragments) { | ||
assert(frag.page_offset < UF2_PAGE_SIZE && frag.page_offset + frag.bytes <= UF2_PAGE_SIZE); | ||
in->seekg(frag.file_offset, in->beg); | ||
|
@@ -118,13 +169,26 @@ int realize_page(std::shared_ptr<std::iostream> in, const std::vector<page_fragm | |
return 0; | ||
} | ||
|
||
/** | ||
* @brief Checks if a specific address is mapped. | ||
* | ||
* @param pages Map of pages with their fragments. | ||
* @param addr Address to check. | ||
* @return true if the address is mapped, false otherwise. | ||
*/ | ||
static bool is_address_mapped(const std::map<uint32_t, std::vector<page_fragment>>& pages, uint32_t addr) { | ||
uint32_t page = addr & ~(UF2_PAGE_SIZE - 1); | ||
if (!pages.count(page)) return false; | ||
// todo check actual address within page | ||
// TODO: Check actual address within page | ||
return true; | ||
} | ||
|
||
/** | ||
* @brief Generates an absolute UF2 block. The absolute block is required to work around Errata E10 in the RP2350 datasheet. | ||
* | ||
* @param abs_block_loc Target address for the absolute block. | ||
* @return Generated UF2 block. | ||
*/ | ||
will-v-pi marked this conversation as resolved.
Show resolved
Hide resolved
|
||
uf2_block gen_abs_block(uint32_t abs_block_loc) { | ||
uf2_block block; | ||
block.magic_start0 = UF2_MAGIC_START0; | ||
|
@@ -141,6 +205,12 @@ uf2_block gen_abs_block(uint32_t abs_block_loc) { | |
return block; | ||
} | ||
|
||
/** | ||
* @brief Checks if a UF2 block is an absolute block. | ||
* | ||
* @param block UF2 block to check. | ||
* @return true if the block is an absolute block, false otherwise. | ||
*/ | ||
bool check_abs_block(uf2_block block) { | ||
return std::all_of(block.data, block.data + UF2_PAGE_SIZE, [](uint8_t i) { return i == 0xef; }) && | ||
block.magic_start0 == UF2_MAGIC_START0 && | ||
|
@@ -153,6 +223,16 @@ bool check_abs_block(uf2_block block) { | |
block.block_no == 0; | ||
} | ||
|
||
/** | ||
* @brief Converts page information into UF2 blocks and writes them to the output stream. | ||
* | ||
* @param pages Map of pages with their fragments. | ||
* @param in Input stream (e.g., ELF or binary file). | ||
* @param out Output stream for UF2 data. | ||
* @param family_id Family ID for UF2. | ||
* @param abs_block_loc Optional absolute block location. | ||
* @return 0 on success, error code otherwise. | ||
*/ | ||
int pages2uf2(std::map<uint32_t, std::vector<page_fragment>>& pages, std::shared_ptr<std::iostream> in, std::shared_ptr<std::iostream> out, uint32_t family_id, uint32_t abs_block_loc=0) { | ||
// RP2350-E10: add absolute block to start of flash UF2s, targeting end of flash by default | ||
if (family_id != ABSOLUTE_FAMILY_ID && family_id != RP2040_FAMILY_ID && abs_block_loc) { | ||
|
@@ -166,22 +246,26 @@ int pages2uf2(std::map<uint32_t, std::vector<page_fragment>>& pages, std::shared | |
} | ||
} | ||
} | ||
|
||
uf2_block block; | ||
unsigned int page_num = 0; | ||
block.magic_start0 = UF2_MAGIC_START0; | ||
block.magic_start1 = UF2_MAGIC_START1; | ||
block.flags = UF2_FLAG_FAMILY_ID_PRESENT; | ||
block.payload_size = UF2_PAGE_SIZE; | ||
block.num_blocks = (uint32_t)pages.size(); | ||
block.num_blocks = static_cast<uint32_t>(pages.size()); | ||
block.file_size = family_id; | ||
block.magic_end = UF2_MAGIC_END; | ||
|
||
for(auto& page_entry : pages) { | ||
block.target_addr = page_entry.first; | ||
block.block_no = page_num++; | ||
|
||
if (verbose) { | ||
printf("Page %d / %d %08x%s\n", block.block_no, block.num_blocks, block.target_addr, | ||
page_entry.second.empty() ? " (padding)": ""); | ||
printf("Page %u / %u 0x%08x%s\n", block.block_no, block.num_blocks, block.target_addr, | ||
page_entry.second.empty() ? " (padding)" : ""); | ||
} | ||
|
||
memset(block.data, 0, sizeof(block.data)); | ||
int rc = realize_page(in, page_entry.second, block.data, sizeof(block.data)); | ||
if (rc) return rc; | ||
|
@@ -193,9 +277,21 @@ int pages2uf2(std::map<uint32_t, std::vector<page_fragment>>& pages, std::shared | |
return 0; | ||
} | ||
|
||
/** | ||
* @brief Converts a binary file to UF2 format. | ||
* | ||
* @param in Input stream for the binary file. | ||
* @param out Output stream for UF2 data. | ||
* @param address Start address in the target memory. | ||
* @param family_id Family ID for UF2. | ||
* @param abs_block_loc Optional absolute block location. If set to 0, this is ignored. Any non-zero value will | ||
* be used to add an absolute UF2 block, targeting the specified location. | ||
* @return 0 on success, error code otherwise. | ||
*/ | ||
int bin2uf2(std::shared_ptr<std::iostream> in, std::shared_ptr<std::iostream> out, uint32_t address, uint32_t family_id, uint32_t abs_block_loc) { | ||
std::map<uint32_t, std::vector<page_fragment>> pages; | ||
|
||
// Determine the size of the input file | ||
in->seekg(0, in->end); | ||
if (in->fail()) { | ||
fail_read_error(); | ||
|
@@ -208,6 +304,7 @@ int bin2uf2(std::shared_ptr<std::iostream> in, std::shared_ptr<std::iostream> ou | |
unsigned int addr = address; | ||
unsigned int remaining = size; | ||
unsigned int file_offset = 0; | ||
|
||
while (remaining) { | ||
unsigned int off = addr & (UF2_PAGE_SIZE - 1); | ||
unsigned int len = std::min(remaining, UF2_PAGE_SIZE - off); | ||
|
@@ -231,80 +328,100 @@ int bin2uf2(std::shared_ptr<std::iostream> in, std::shared_ptr<std::iostream> ou | |
return pages2uf2(pages, in, out, family_id, abs_block_loc); | ||
} | ||
|
||
/** | ||
* @brief Converts an ELF file to UF2 format. | ||
* | ||
* @param in Input stream for the ELF file. | ||
* @param out Output stream for UF2 data. | ||
* @param family_id Family ID for UF2. | ||
* @param package_addr Optional packaging address. | ||
* @param abs_block_loc Optional absolute block location. | ||
Comment on lines
+337
to
+338
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Same questions about "Optional" |
||
* @return 0 on success, error code otherwise. | ||
*/ | ||
int elf2uf2(std::shared_ptr<std::iostream> in, std::shared_ptr<std::iostream> out, uint32_t family_id, uint32_t package_addr, uint32_t abs_block_loc) { | ||
elf_file elf; | ||
std::map<uint32_t, std::vector<page_fragment>> pages; | ||
|
||
// Read the ELF file | ||
int rc = elf.read_file(in); | ||
bool ram_style = false; | ||
address_ranges valid_ranges = {}; | ||
address_ranges flash_range; address_ranges ram_range; | ||
|
||
// Determine the valid address ranges based on the family ID | ||
if (family_id == RP2040_FAMILY_ID) { | ||
flash_range = rp2040_address_ranges_flash; | ||
ram_range = rp2040_address_ranges_ram; | ||
} else { | ||
flash_range = rp2350_address_ranges_flash; | ||
ram_range = rp2350_address_ranges_ram; | ||
} | ||
|
||
if (!rc) { | ||
rc = rp_determine_binary_type(elf.header(), elf.segments(), flash_range, ram_range, &ram_style); | ||
if (!rc) { | ||
if (verbose) { | ||
if (ram_style) { | ||
printf("Detected RAM binary\n"); | ||
} else { | ||
printf("Detected FLASH binary\n"); | ||
} | ||
printf("Detected %s binary\n", ram_style ? "RAM" : "FLASH"); | ||
} | ||
valid_ranges = ram_style ? ram_range : flash_range; | ||
rc = check_elf32_ph_entries(elf.segments(), valid_ranges, pages); | ||
} | ||
} | ||
|
||
if (rc) return rc; | ||
|
||
if (pages.empty()) { | ||
fail(ERROR_INCOMPATIBLE, "The input file has no memory pages"); | ||
} | ||
// No Thumb bit on RISC-V | ||
|
||
// Determine the Thumb bit (for ARM architecture) | ||
elf32_header eh = elf.header(); | ||
uint32_t thumb_bit = eh.common.machine == EM_ARM ? 0x1u : 0x0u; | ||
uint32_t thumb_bit = (eh.common.machine == EM_ARM) ? 0x1u : 0x0u; | ||
|
||
if (ram_style) { | ||
uint32_t expected_ep_main_ram = UINT32_MAX; | ||
uint32_t expected_ep_xip_sram = UINT32_MAX; | ||
|
||
// Determine expected entry points | ||
for(auto& page_entry : pages) { | ||
if ( ((page_entry.first >= SRAM_START) && (page_entry.first < ram_range[0].to)) && (page_entry.first < expected_ep_main_ram) ) { | ||
if ((page_entry.first >= SRAM_START) && (page_entry.first < ram_range[0].to) && (page_entry.first < expected_ep_main_ram)) { | ||
expected_ep_main_ram = page_entry.first | thumb_bit; | ||
} else if ( ((page_entry.first >= ram_range[1].from) && (page_entry.first < ram_range[1].to)) && (page_entry.first < expected_ep_xip_sram) ) { | ||
} else if (((page_entry.first >= ram_range[1].from) && (page_entry.first < ram_range[1].to)) && (page_entry.first < expected_ep_xip_sram)) { | ||
expected_ep_xip_sram = page_entry.first | thumb_bit; | ||
} | ||
} | ||
uint32_t expected_ep = (UINT32_MAX != expected_ep_main_ram) ? expected_ep_main_ram : expected_ep_xip_sram; | ||
|
||
// Check entry points for RP2040 | ||
if (eh.entry == expected_ep_xip_sram && family_id == RP2040_FAMILY_ID) { | ||
fail(ERROR_INCOMPATIBLE, "RP2040 B0/B1/B2 Boot ROM does not support direct entry into XIP_SRAM\n"); | ||
} else if (eh.entry != expected_ep && family_id == RP2040_FAMILY_ID) { | ||
fail(ERROR_INCOMPATIBLE, "A RP2040 RAM binary should have an entry point at the beginning: %08x (not %08x)\n", expected_ep, eh.entry); | ||
fail(ERROR_INCOMPATIBLE, "A RP2040 RAM binary should have an entry point at the beginning: 0x%08x (not 0x%08x)\n", expected_ep, eh.entry); | ||
} | ||
|
||
// Ensure SRAM_START is aligned to a page boundary | ||
static_assert(0 == (SRAM_START & (UF2_PAGE_SIZE - 1)), ""); | ||
// currently don't require this as entry point is now at the start, we don't know where reset vector is | ||
// todo can be re-enabled for RP2350 | ||
|
||
// Currently don't require this as entry point is now at the start, we don't know where reset vector is | ||
// TODO: Can be re-enabled for RP2350 | ||
#if 0 | ||
uint8_t buf[UF2_PAGE_SIZE]; | ||
rc = realize_page(in, pages[SRAM_START], buf, sizeof(buf)); | ||
if (rc) return rc; | ||
uint32_t sp = ((uint32_t *)buf)[0]; | ||
uint32_t ip = ((uint32_t *)buf)[1]; | ||
if (!is_address_mapped(pages, ip)) { | ||
fail(ERROR_INCOMPATIBLE, "Vector table at %08x is invalid: reset vector %08x is not in mapped memory", | ||
SRAM_START, ip); | ||
fail(ERROR_INCOMPATIBLE, "Vector table at 0x%08x is invalid: reset vector 0x%08x is not in mapped memory", | ||
SRAM_START, ip); | ||
} | ||
if (!is_address_valid(valid_ranges, sp - 4)) { | ||
fail(ERROR_INCOMPATIBLE, "Vector table at %08x is invalid: stack pointer %08x is not in RAM", | ||
fail(ERROR_INCOMPATIBLE, "Vector table at 0x%08x is invalid: stack pointer 0x%08x is not in RAM", | ||
SRAM_START, sp); | ||
} | ||
#endif | ||
} else { | ||
// Fill in empty dummy uf2 pages to align the binary to flash sectors (except for the last sector which we don't | ||
// need to pad, and choose not to to avoid making all SDK UF2s bigger) | ||
// That workaround is required because the bootrom uses the block number for erase sector calculations: | ||
// Fill in empty dummy UF2 pages to align the binary to flash sectors (except for the last sector) | ||
// This workaround is required because the boot ROM uses the block number for erase sector calculations: | ||
// https://github.com/raspberrypi/pico-bootrom/blob/c09c7f08550e8a36fc38dc74f8873b9576de99eb/bootrom/virtual_disk.c#L205 | ||
|
||
std::set<uint32_t> touched_sectors; | ||
|
@@ -317,16 +434,16 @@ int elf2uf2(std::shared_ptr<std::iostream> in, std::shared_ptr<std::iostream> ou | |
for (uint32_t sector : touched_sectors) { | ||
for (uint32_t page = sector * FLASH_SECTOR_ERASE_SIZE; page < (sector + 1) * FLASH_SECTOR_ERASE_SIZE; page += UF2_PAGE_SIZE) { | ||
if (page < last_page) { | ||
// Create a dummy page, if it does not exist yet. note that all present pages are first | ||
// Create a dummy page if it does not exist yet. Note that all present pages are first | ||
// zeroed before they are filled with any contents, so a dummy page will be all zeros. | ||
auto &dummy = pages[page]; | ||
} | ||
} | ||
} | ||
} | ||
|
||
// Optional: Package binary at a specific address | ||
if (package_addr) { | ||
// Package binary at address | ||
uint32_t base_addr = pages.begin()->first; | ||
int32_t package_delta = package_addr - base_addr; | ||
if (verbose) printf("Base %x\n", base_addr); | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"Reference to the valid_range that the provided (addr + size) fits inside" ?