300 lines
7.0 KiB
C
300 lines
7.0 KiB
C
|
// SPDX-License-Identifier: GPL-2.0+
|
||
|
/*
|
||
|
* (C) Copyright 2015
|
||
|
* Linus Walleij, Linaro
|
||
|
*
|
||
|
* Support for ARM Flash Partitions
|
||
|
*/
|
||
|
#include <common.h>
|
||
|
#include <command.h>
|
||
|
#include <console.h>
|
||
|
#include <asm/io.h>
|
||
|
|
||
|
#define MAX_REGIONS 4
|
||
|
#define MAX_IMAGES 32
|
||
|
|
||
|
struct afs_region {
|
||
|
u32 load_address;
|
||
|
u32 size;
|
||
|
u32 offset;
|
||
|
};
|
||
|
|
||
|
struct afs_image {
|
||
|
flash_info_t *flinfo;
|
||
|
const char *name;
|
||
|
u32 version;
|
||
|
u32 entrypoint;
|
||
|
u32 attributes;
|
||
|
u32 region_count;
|
||
|
struct afs_region regions[MAX_REGIONS];
|
||
|
ulong flash_mem_start;
|
||
|
ulong flash_mem_end;
|
||
|
};
|
||
|
|
||
|
static struct afs_image afs_images[MAX_IMAGES];
|
||
|
static int num_afs_images;
|
||
|
|
||
|
static u32 compute_crc(ulong start, u32 len)
|
||
|
{
|
||
|
u32 sum = 0;
|
||
|
int i;
|
||
|
|
||
|
if (len % 4 != 0) {
|
||
|
printf("bad checksumming\n");
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < len; i += 4) {
|
||
|
u32 val;
|
||
|
|
||
|
val = readl((void *)start + i);
|
||
|
if (val > ~sum)
|
||
|
sum++;
|
||
|
sum += val;
|
||
|
}
|
||
|
return ~sum;
|
||
|
}
|
||
|
|
||
|
static void parse_bank(ulong bank)
|
||
|
{
|
||
|
int i;
|
||
|
ulong flstart, flend;
|
||
|
flash_info_t *info;
|
||
|
|
||
|
info = &flash_info[bank];
|
||
|
if (info->flash_id != FLASH_MAN_CFI) {
|
||
|
printf("Bank %lu: missing or unknown FLASH type\n", bank);
|
||
|
return;
|
||
|
}
|
||
|
if (!info->sector_count) {
|
||
|
printf("Bank %lu: no FLASH sectors\n", bank);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
flstart = info->start[0];
|
||
|
flend = flstart + info->size;
|
||
|
|
||
|
for (i = 0; i < info->sector_count; ++i) {
|
||
|
ulong secend;
|
||
|
u32 foot1, foot2;
|
||
|
|
||
|
if (ctrlc())
|
||
|
break;
|
||
|
|
||
|
if (i == info->sector_count-1)
|
||
|
secend = flend;
|
||
|
else
|
||
|
secend = info->start[i+1];
|
||
|
|
||
|
/* Check for v1 header */
|
||
|
foot1 = readl((void *)secend - 0x0c);
|
||
|
if (foot1 == 0xA0FFFF9FU) {
|
||
|
struct afs_image *afi = &afs_images[num_afs_images];
|
||
|
ulong imginfo;
|
||
|
|
||
|
afi->flinfo = info;
|
||
|
afi->version = 1;
|
||
|
afi->flash_mem_start = readl((void *)secend - 0x10);
|
||
|
afi->flash_mem_end = readl((void *)secend - 0x14);
|
||
|
afi->attributes = readl((void *)secend - 0x08);
|
||
|
/* Adjust to even address */
|
||
|
imginfo = afi->flash_mem_end + afi->flash_mem_end % 4;
|
||
|
/* Record as a single region */
|
||
|
afi->region_count = 1;
|
||
|
afi->regions[0].offset = readl((void *)imginfo + 0x04);
|
||
|
afi->regions[0].load_address =
|
||
|
readl((void *)imginfo + 0x08);
|
||
|
afi->regions[0].size = readl((void *)imginfo + 0x0C);
|
||
|
afi->entrypoint = readl((void *)imginfo + 0x10);
|
||
|
afi->name = (const char *)imginfo + 0x14;
|
||
|
num_afs_images++;
|
||
|
}
|
||
|
|
||
|
/* Check for v2 header */
|
||
|
foot1 = readl((void *)secend - 0x04);
|
||
|
foot2 = readl((void *)secend - 0x08);
|
||
|
/* This makes up the string "HSLFTOOF" flash footer */
|
||
|
if (foot1 == 0x464F4F54U && foot2 == 0x464C5348U) {
|
||
|
struct afs_image *afi = &afs_images[num_afs_images];
|
||
|
ulong imginfo;
|
||
|
u32 block_start, block_end;
|
||
|
int j;
|
||
|
|
||
|
afi->flinfo = info;
|
||
|
afi->version = readl((void *)secend - 0x0c);
|
||
|
imginfo = secend - 0x30 - readl((void *)secend - 0x10);
|
||
|
afi->name = (const char *)secend - 0x30;
|
||
|
|
||
|
afi->entrypoint = readl((void *)imginfo+0x08);
|
||
|
afi->attributes = readl((void *)imginfo+0x0c);
|
||
|
afi->region_count = readl((void *)imginfo+0x10);
|
||
|
block_start = readl((void *)imginfo+0x54);
|
||
|
block_end = readl((void *)imginfo+0x58);
|
||
|
afi->flash_mem_start = afi->flinfo->start[block_start];
|
||
|
afi->flash_mem_end = afi->flinfo->start[block_end];
|
||
|
|
||
|
/*
|
||
|
* Check footer CRC, the algorithm saves the inverse
|
||
|
* checksum as part of the summed words, and thus
|
||
|
* the result should be zero.
|
||
|
*/
|
||
|
if (compute_crc(imginfo + 8, 0x88) != 0) {
|
||
|
printf("BAD CRC on ARM image info\n");
|
||
|
printf("(continuing anyway)\n");
|
||
|
}
|
||
|
|
||
|
/* Parse regions */
|
||
|
for (j = 0; j < afi->region_count; j++) {
|
||
|
afi->regions[j].load_address =
|
||
|
readl((void *)imginfo+0x14 + j*0x10);
|
||
|
afi->regions[j].size =
|
||
|
readl((void *)imginfo+0x18 + j*0x10);
|
||
|
afi->regions[j].offset =
|
||
|
readl((void *)imginfo+0x1c + j*0x10);
|
||
|
/*
|
||
|
* At offset 0x20 + j*0x10 there is a region
|
||
|
* checksum which seems to be the running
|
||
|
* sum + 3, however since we anyway checksum
|
||
|
* the entire footer this is skipped over for
|
||
|
* checking here.
|
||
|
*/
|
||
|
}
|
||
|
num_afs_images++;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void parse_flash(void)
|
||
|
{
|
||
|
ulong bank;
|
||
|
|
||
|
/* We have already parsed the images in flash */
|
||
|
if (num_afs_images > 0)
|
||
|
return;
|
||
|
for (bank = 0; bank < CONFIG_SYS_MAX_FLASH_BANKS; ++bank)
|
||
|
parse_bank(bank);
|
||
|
}
|
||
|
|
||
|
static int load_image(const char * const name, const ulong address)
|
||
|
{
|
||
|
struct afs_image *afi = NULL;
|
||
|
int i;
|
||
|
|
||
|
parse_flash();
|
||
|
for (i = 0; i < num_afs_images; i++) {
|
||
|
struct afs_image *tmp = &afs_images[i];
|
||
|
|
||
|
if (!strcmp(tmp->name, name)) {
|
||
|
afi = tmp;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!afi) {
|
||
|
printf("image \"%s\" not found in flash\n", name);
|
||
|
return CMD_RET_FAILURE;
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < afi->region_count; i++) {
|
||
|
ulong from, to;
|
||
|
|
||
|
from = afi->flash_mem_start + afi->regions[i].offset;
|
||
|
if (address) {
|
||
|
to = address;
|
||
|
} else if (afi->regions[i].load_address) {
|
||
|
to = afi->regions[i].load_address;
|
||
|
} else {
|
||
|
printf("no valid load address\n");
|
||
|
return CMD_RET_FAILURE;
|
||
|
}
|
||
|
|
||
|
memcpy((void *)to, (void *)from, afi->regions[i].size);
|
||
|
|
||
|
printf("loaded region %d from %08lX to %08lX, %08X bytes\n",
|
||
|
i,
|
||
|
from,
|
||
|
to,
|
||
|
afi->regions[i].size);
|
||
|
}
|
||
|
return CMD_RET_SUCCESS;
|
||
|
}
|
||
|
|
||
|
static void print_images(void)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
parse_flash();
|
||
|
for (i = 0; i < num_afs_images; i++) {
|
||
|
struct afs_image *afi = &afs_images[i];
|
||
|
int j;
|
||
|
|
||
|
printf("Image: \"%s\" (v%d):\n", afi->name, afi->version);
|
||
|
printf(" Entry point: 0x%08X\n", afi->entrypoint);
|
||
|
printf(" Attributes: 0x%08X: ", afi->attributes);
|
||
|
if (afi->attributes == 0x01)
|
||
|
printf("ARM executable");
|
||
|
if (afi->attributes == 0x08)
|
||
|
printf("ARM backup");
|
||
|
printf("\n");
|
||
|
printf(" Flash mem start: 0x%08lX\n",
|
||
|
afi->flash_mem_start);
|
||
|
printf(" Flash mem end: 0x%08lX\n",
|
||
|
afi->flash_mem_end);
|
||
|
for (j = 0; j < afi->region_count; j++) {
|
||
|
printf(" region %d\n"
|
||
|
" load address: %08X\n"
|
||
|
" size: %08X\n"
|
||
|
" offset: %08X\n",
|
||
|
j,
|
||
|
afi->regions[j].load_address,
|
||
|
afi->regions[j].size,
|
||
|
afi->regions[j].offset);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static int exists(const char * const name)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
parse_flash();
|
||
|
for (i = 0; i < num_afs_images; i++) {
|
||
|
struct afs_image *afi = &afs_images[i];
|
||
|
|
||
|
if (strcmp(afi->name, name) == 0)
|
||
|
return CMD_RET_SUCCESS;
|
||
|
}
|
||
|
return CMD_RET_FAILURE;
|
||
|
}
|
||
|
|
||
|
static int do_afs(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
|
||
|
{
|
||
|
int ret = CMD_RET_SUCCESS;
|
||
|
|
||
|
if (argc == 1) {
|
||
|
print_images();
|
||
|
} else if (argc == 3 && !strcmp(argv[1], "exists")) {
|
||
|
ret = exists(argv[2]);
|
||
|
} else if (argc == 3 && !strcmp(argv[1], "load")) {
|
||
|
ret = load_image(argv[2], 0x0);
|
||
|
} else if (argc == 4 && !strcmp(argv[1], "load")) {
|
||
|
ulong load_addr;
|
||
|
|
||
|
load_addr = simple_strtoul(argv[3], NULL, 16);
|
||
|
ret = load_image(argv[2], load_addr);
|
||
|
} else {
|
||
|
return CMD_RET_USAGE;
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
U_BOOT_CMD(afs, 4, 0, do_afs, "show AFS partitions",
|
||
|
"no arguments\n"
|
||
|
" - list images in flash\n"
|
||
|
"exists <image>\n"
|
||
|
" - returns 1 if an image exists, else 0\n"
|
||
|
"load <image>\n"
|
||
|
" - load an image to the location indicated in the header\n"
|
||
|
"load <image> 0x<address>\n"
|
||
|
" - load an image to the location specified\n");
|