mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 04:52:54 +01:00
64ba553d1f
Co-authored-by: Marian Buschsieweke <marian.buschsieweke@posteo.net>
251 lines
8.3 KiB
C
251 lines
8.3 KiB
C
/*
|
|
* Copyright (C) 2019 Inria
|
|
* 2019 Freie Universität Berlin
|
|
* 2019 Kaspar Schleiser <kaspar@schleiser.de>
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup sys_riotboot_flashwrite
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Firmware update helper functions
|
|
*
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <string.h>
|
|
|
|
#include "architecture.h"
|
|
#include "riotboot/flashwrite.h"
|
|
#include "riotboot/slot.h"
|
|
#include "od.h"
|
|
|
|
#define LOG_PREFIX "riotboot_flashwrite: "
|
|
#include "log.h"
|
|
|
|
static inline size_t min(size_t a, size_t b)
|
|
{
|
|
return a <= b ? a : b;
|
|
}
|
|
|
|
size_t riotboot_flashwrite_slotsize(
|
|
const riotboot_flashwrite_t *state)
|
|
{
|
|
return riotboot_slot_size(state->target_slot);
|
|
}
|
|
|
|
int riotboot_flashwrite_init_raw(riotboot_flashwrite_t *state, int target_slot,
|
|
size_t offset)
|
|
{
|
|
#ifdef FLASHPAGE_SIZE
|
|
assert(offset <= FLASHPAGE_SIZE);
|
|
/* the flashpage size must be a multiple of the riotboot flashpage buffer */
|
|
static_assert(!(FLASHPAGE_SIZE % RIOTBOOT_FLASHPAGE_BUFFER_SIZE),
|
|
"Flashpage size must be a multiple of riotboot flashpage buffer.");
|
|
#else
|
|
/* The flashpage buffer must be a multiple of the write block size */
|
|
static_assert(!(RIOTBOOT_FLASHPAGE_BUFFER_SIZE %
|
|
FLASHPAGE_WRITE_BLOCK_SIZE),
|
|
"Flashpage buffer must be a multiple of write block size.");
|
|
#endif
|
|
|
|
LOG_INFO(LOG_PREFIX "initializing update to target slot %i\n",
|
|
target_slot);
|
|
|
|
memset(state, 0, sizeof(riotboot_flashwrite_t));
|
|
|
|
state->offset = offset;
|
|
state->target_slot = target_slot;
|
|
state->flashpage =
|
|
flashpage_page((void *)riotboot_slot_get_hdr(target_slot));
|
|
|
|
if (CONFIG_RIOTBOOT_FLASHWRITE_RAW && offset) {
|
|
/* Erase the first page only if the offset (!=0) specifies that there is
|
|
* a checksum or other mechanism at the start of the page. */
|
|
flashpage_erase(state->flashpage);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int riotboot_flashwrite_flush(riotboot_flashwrite_t *state)
|
|
{
|
|
if (CONFIG_RIOTBOOT_FLASHWRITE_RAW) {
|
|
/* Check if there is leftover data in the buffer */
|
|
size_t flashwrite_buffer_pos = state->offset %
|
|
RIOTBOOT_FLASHPAGE_BUFFER_SIZE;
|
|
if (flashwrite_buffer_pos == 0) {
|
|
return 0;
|
|
}
|
|
uint8_t *slot_start =
|
|
(uint8_t *)riotboot_slot_get_hdr(state->target_slot);
|
|
/* Get the offset of the remaining chunk */
|
|
size_t flashpage_pos = state->offset - flashwrite_buffer_pos;
|
|
/* Write remaining chunk */
|
|
flashpage_write(slot_start + flashpage_pos,
|
|
state->flashpage_buf,
|
|
RIOTBOOT_FLASHPAGE_BUFFER_SIZE);
|
|
}
|
|
else {
|
|
if (flashpage_write_and_verify(state->flashpage,
|
|
state->flashpage_buf) != FLASHPAGE_OK) {
|
|
LOG_WARNING(LOG_PREFIX "error writing flashpage %u!\n",
|
|
state->flashpage);
|
|
return -1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int riotboot_flashwrite_putbytes(riotboot_flashwrite_t *state,
|
|
const uint8_t *bytes, size_t len, bool more)
|
|
{
|
|
LOG_DEBUG(LOG_PREFIX "processing bytes %" PRIuSIZE "-%" PRIuSIZE "\n", state->offset,
|
|
state->offset + len - 1);
|
|
|
|
while (len) {
|
|
/* Position within the page, calculated from state->offset by
|
|
* subtracting the start offset of the current page */
|
|
size_t flashpage_pos = state->offset -
|
|
(flashpage_addr(state->flashpage) -
|
|
(void *)riotboot_slot_get_hdr(
|
|
state->target_slot));
|
|
size_t flashwrite_buffer_pos = state->offset %
|
|
RIOTBOOT_FLASHPAGE_BUFFER_SIZE;
|
|
size_t flashpage_avail = RIOTBOOT_FLASHPAGE_BUFFER_SIZE -
|
|
flashwrite_buffer_pos;
|
|
|
|
size_t to_copy = min(flashpage_avail, len);
|
|
|
|
if (CONFIG_RIOTBOOT_FLASHWRITE_RAW &&
|
|
flashpage_pos == flashpage_size(state->flashpage)) {
|
|
/* Erase the next page */
|
|
state->flashpage++;
|
|
flashpage_pos = 0;
|
|
flashpage_erase(state->flashpage);
|
|
}
|
|
if (CONFIG_RIOTBOOT_FLASHWRITE_RAW &&
|
|
flashwrite_buffer_pos == 0) {
|
|
memset(state->flashpage_buf, 0, RIOTBOOT_FLASHPAGE_BUFFER_SIZE);
|
|
}
|
|
|
|
memcpy(state->flashpage_buf + flashwrite_buffer_pos, bytes, to_copy);
|
|
flashpage_avail -= to_copy;
|
|
|
|
state->offset += to_copy;
|
|
bytes += to_copy;
|
|
len -= to_copy;
|
|
if ((!flashpage_avail) || (!more)) {
|
|
#if CONFIG_RIOTBOOT_FLASHWRITE_RAW /* Guards access to state::firstblock_buf */
|
|
void *addr = flashpage_addr(state->flashpage);
|
|
if (addr == riotboot_slot_get_hdr(state->target_slot) &&
|
|
state->offset == RIOTBOOT_FLASHPAGE_BUFFER_SIZE) {
|
|
/* Skip flashing the first block, store it for later to flash it
|
|
* during the flashwrite_finish function */
|
|
memcpy(state->firstblock_buf,
|
|
state->flashpage_buf, RIOTBOOT_FLASHPAGE_BUFFER_SIZE);
|
|
}
|
|
else {
|
|
flashpage_write((uint8_t *)addr + flashpage_pos,
|
|
state->flashpage_buf,
|
|
RIOTBOOT_FLASHPAGE_BUFFER_SIZE);
|
|
}
|
|
#else
|
|
int res = flashpage_write_and_verify(state->flashpage,
|
|
state->flashpage_buf);
|
|
if (res != FLASHPAGE_OK) {
|
|
LOG_WARNING(LOG_PREFIX "error writing flashpage %u!\n",
|
|
state->flashpage);
|
|
return -1;
|
|
}
|
|
state->flashpage++;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int riotboot_flashwrite_invalidate(int slot)
|
|
{
|
|
if (riotboot_slot_numof == 1) {
|
|
LOG_WARNING(LOG_PREFIX "abort, only one slot configured\n");
|
|
return -1;
|
|
}
|
|
if (riotboot_slot_validate(1 - slot) != 0) {
|
|
LOG_WARNING(
|
|
LOG_PREFIX "abort, can not erase slot[%d], other slot[%d] is invalid\n", slot,
|
|
1 - slot);
|
|
return -2;
|
|
}
|
|
|
|
/* invalidate header (checksum and magic number must be invalidated),
|
|
write the whole header to avoid running in memory alignment issues
|
|
with FLASHPAGE_WRITE_BLOCK_SIZE */
|
|
riotboot_hdr_t tmp_hdr;
|
|
|
|
memset(&tmp_hdr, (~FLASHPAGE_ERASE_STATE), sizeof(riotboot_hdr_t));
|
|
|
|
flashpage_write((void *)riotboot_slot_get_hdr(slot), &tmp_hdr,
|
|
sizeof(riotboot_hdr_t));
|
|
|
|
return 0;
|
|
}
|
|
|
|
int riotboot_flashwrite_invalidate_latest(void)
|
|
{
|
|
int _slot_to_revert;
|
|
|
|
_slot_to_revert = (riotboot_slot_get_hdr(riotboot_slot_other())->version
|
|
> riotboot_slot_get_hdr(riotboot_slot_current())->version)
|
|
? riotboot_slot_other() : riotboot_slot_current();
|
|
return riotboot_flashwrite_invalidate(_slot_to_revert);
|
|
}
|
|
|
|
int riotboot_flashwrite_finish_raw(riotboot_flashwrite_t *state,
|
|
const uint8_t *bytes, size_t len)
|
|
{
|
|
#ifndef PERIPH_FLASHPAGE_CUSTOM_PAGESIZES
|
|
assert(len <= FLASHPAGE_SIZE);
|
|
#endif
|
|
|
|
uint8_t *slot_start = (uint8_t *)riotboot_slot_get_hdr(state->target_slot);
|
|
|
|
#if IS_ACTIVE(CONFIG_RIOTBOOT_FLASHWRITE_RAW)
|
|
memcpy(state->firstblock_buf, bytes, len);
|
|
flashpage_write(slot_start, state->firstblock_buf,
|
|
RIOTBOOT_FLASHPAGE_BUFFER_SIZE);
|
|
#else
|
|
uint8_t *firstpage;
|
|
|
|
if (len < FLASHPAGE_SIZE) {
|
|
firstpage = state->flashpage_buf;
|
|
memcpy(firstpage, bytes, len);
|
|
memcpy(firstpage + len,
|
|
slot_start + len,
|
|
FLASHPAGE_SIZE - len);
|
|
}
|
|
else {
|
|
firstpage = (void *)bytes;
|
|
}
|
|
|
|
int flashpage = flashpage_page((void *)slot_start);
|
|
if (flashpage_write_and_verify(flashpage, firstpage) == FLASHPAGE_OK) {
|
|
LOG_INFO(LOG_PREFIX "riotboot flashing completed successfully\n");
|
|
}
|
|
else {
|
|
LOG_ERROR(LOG_PREFIX "re-flashing first block failed!\n");
|
|
return -1;
|
|
}
|
|
#endif /* !CONFIG_RIOTBOOT_FLASHWRITE_RAW */
|
|
return 0;
|
|
}
|