mirror of
https://github.com/u-boot/u-boot.git
synced 2025-05-09 03:21:51 +00:00
stm32mp: stm32prog: add flash layout parsing
Build the list of device and of partition with a tab separated value file with a stm32 header: the FlashLayout.tsv (https://wiki.st.com/stm32mpu/wiki/STM32CubeProgrammer_flashlayout) Signed-off-by: Patrick Delaunay <patrick.delaunay@st.com> Reviewed-by: Patrice Chotard <patrice.chotard@st.com>
This commit is contained in:
parent
954bd1a923
commit
6ee6839183
1 changed files with 371 additions and 1 deletions
|
@ -24,6 +24,17 @@
|
||||||
|
|
||||||
DECLARE_GLOBAL_DATA_PTR;
|
DECLARE_GLOBAL_DATA_PTR;
|
||||||
|
|
||||||
|
/* order of column in flash layout file */
|
||||||
|
enum stm32prog_col_t {
|
||||||
|
COL_OPTION,
|
||||||
|
COL_ID,
|
||||||
|
COL_NAME,
|
||||||
|
COL_TYPE,
|
||||||
|
COL_IP,
|
||||||
|
COL_OFFSET,
|
||||||
|
COL_NB_STM32
|
||||||
|
};
|
||||||
|
|
||||||
char *stm32prog_get_error(struct stm32prog_data *data)
|
char *stm32prog_get_error(struct stm32prog_data *data)
|
||||||
{
|
{
|
||||||
static const char error_msg[] = "Unspecified";
|
static const char error_msg[] = "Unspecified";
|
||||||
|
@ -34,11 +45,370 @@ char *stm32prog_get_error(struct stm32prog_data *data)
|
||||||
return data->error;
|
return data->error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
u8 stm32prog_header_check(struct raw_header_s *raw_header,
|
||||||
|
struct image_header_s *header)
|
||||||
|
{
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
|
header->present = 0;
|
||||||
|
header->image_checksum = 0x0;
|
||||||
|
header->image_length = 0x0;
|
||||||
|
|
||||||
|
if (!raw_header || !header) {
|
||||||
|
pr_debug("%s:no header data\n", __func__);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (raw_header->magic_number !=
|
||||||
|
(('S' << 0) | ('T' << 8) | ('M' << 16) | (0x32 << 24))) {
|
||||||
|
pr_debug("%s:invalid magic number : 0x%x\n",
|
||||||
|
__func__, raw_header->magic_number);
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
/* only header v1.0 supported */
|
||||||
|
if (raw_header->header_version != 0x00010000) {
|
||||||
|
pr_debug("%s:invalid header version : 0x%x\n",
|
||||||
|
__func__, raw_header->header_version);
|
||||||
|
return -3;
|
||||||
|
}
|
||||||
|
if (raw_header->reserved1 != 0x0 || raw_header->reserved2) {
|
||||||
|
pr_debug("%s:invalid reserved field\n", __func__);
|
||||||
|
return -4;
|
||||||
|
}
|
||||||
|
for (i = 0; i < (sizeof(raw_header->padding) / 4); i++) {
|
||||||
|
if (raw_header->padding[i] != 0) {
|
||||||
|
pr_debug("%s:invalid padding field\n", __func__);
|
||||||
|
return -5;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
header->present = 1;
|
||||||
|
header->image_checksum = le32_to_cpu(raw_header->image_checksum);
|
||||||
|
header->image_length = le32_to_cpu(raw_header->image_length);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static u32 stm32prog_header_checksum(u32 addr, struct image_header_s *header)
|
||||||
|
{
|
||||||
|
u32 i, checksum;
|
||||||
|
u8 *payload;
|
||||||
|
|
||||||
|
/* compute checksum on payload */
|
||||||
|
payload = (u8 *)addr;
|
||||||
|
checksum = 0;
|
||||||
|
for (i = header->image_length; i > 0; i--)
|
||||||
|
checksum += *(payload++);
|
||||||
|
|
||||||
|
return checksum;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* FLASHLAYOUT PARSING *****************************************/
|
||||||
|
static int parse_option(struct stm32prog_data *data,
|
||||||
|
int i, char *p, struct stm32prog_part_t *part)
|
||||||
|
{
|
||||||
|
int result = 0;
|
||||||
|
char *c = p;
|
||||||
|
|
||||||
|
part->option = 0;
|
||||||
|
if (!strcmp(p, "-"))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
while (*c) {
|
||||||
|
switch (*c) {
|
||||||
|
case 'P':
|
||||||
|
part->option |= OPT_SELECT;
|
||||||
|
break;
|
||||||
|
case 'E':
|
||||||
|
part->option |= OPT_EMPTY;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
result = -EINVAL;
|
||||||
|
stm32prog_err("Layout line %d: invalid option '%c' in %s)",
|
||||||
|
i, *c, p);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
c++;
|
||||||
|
}
|
||||||
|
if (!(part->option & OPT_SELECT)) {
|
||||||
|
stm32prog_err("Layout line %d: missing 'P' in option %s", i, p);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int parse_id(struct stm32prog_data *data,
|
||||||
|
int i, char *p, struct stm32prog_part_t *part)
|
||||||
|
{
|
||||||
|
int result = 0;
|
||||||
|
unsigned long value;
|
||||||
|
|
||||||
|
result = strict_strtoul(p, 0, &value);
|
||||||
|
part->id = value;
|
||||||
|
if (result || value > PHASE_LAST_USER) {
|
||||||
|
stm32prog_err("Layout line %d: invalid phase value = %s", i, p);
|
||||||
|
result = -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int parse_name(struct stm32prog_data *data,
|
||||||
|
int i, char *p, struct stm32prog_part_t *part)
|
||||||
|
{
|
||||||
|
int result = 0;
|
||||||
|
|
||||||
|
if (strlen(p) < sizeof(part->name)) {
|
||||||
|
strcpy(part->name, p);
|
||||||
|
} else {
|
||||||
|
stm32prog_err("Layout line %d: partition name too long [%d]: %s",
|
||||||
|
i, strlen(p), p);
|
||||||
|
result = -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int parse_type(struct stm32prog_data *data,
|
||||||
|
int i, char *p, struct stm32prog_part_t *part)
|
||||||
|
{
|
||||||
|
int result = 0;
|
||||||
|
|
||||||
|
if (!strcmp(p, "Binary")) {
|
||||||
|
part->part_type = PART_BINARY;
|
||||||
|
} else if (!strcmp(p, "System")) {
|
||||||
|
part->part_type = PART_SYSTEM;
|
||||||
|
} else if (!strcmp(p, "FileSystem")) {
|
||||||
|
part->part_type = PART_FILESYSTEM;
|
||||||
|
} else if (!strcmp(p, "RawImage")) {
|
||||||
|
part->part_type = RAW_IMAGE;
|
||||||
|
} else {
|
||||||
|
result = -EINVAL;
|
||||||
|
}
|
||||||
|
if (result)
|
||||||
|
stm32prog_err("Layout line %d: type parsing error : '%s'",
|
||||||
|
i, p);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int parse_ip(struct stm32prog_data *data,
|
||||||
|
int i, char *p, struct stm32prog_part_t *part)
|
||||||
|
{
|
||||||
|
int result = 0;
|
||||||
|
unsigned int len = 0;
|
||||||
|
|
||||||
|
part->dev_id = 0;
|
||||||
|
if (!strcmp(p, "none")) {
|
||||||
|
part->target = STM32PROG_NONE;
|
||||||
|
} else {
|
||||||
|
result = -EINVAL;
|
||||||
|
}
|
||||||
|
if (len) {
|
||||||
|
/* only one digit allowed for device id */
|
||||||
|
if (strlen(p) != len + 1) {
|
||||||
|
result = -EINVAL;
|
||||||
|
} else {
|
||||||
|
part->dev_id = p[len] - '0';
|
||||||
|
if (part->dev_id > 9)
|
||||||
|
result = -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (result)
|
||||||
|
stm32prog_err("Layout line %d: ip parsing error: '%s'", i, p);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int parse_offset(struct stm32prog_data *data,
|
||||||
|
int i, char *p, struct stm32prog_part_t *part)
|
||||||
|
{
|
||||||
|
int result = 0;
|
||||||
|
char *tail;
|
||||||
|
|
||||||
|
part->part_id = 0;
|
||||||
|
part->size = 0;
|
||||||
|
part->addr = simple_strtoull(p, &tail, 0);
|
||||||
|
if (tail == p || *tail != '\0') {
|
||||||
|
stm32prog_err("Layout line %d: invalid offset '%s'",
|
||||||
|
i, p);
|
||||||
|
result = -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int (* const parse[COL_NB_STM32])(struct stm32prog_data *data, int i, char *p,
|
||||||
|
struct stm32prog_part_t *part) = {
|
||||||
|
[COL_OPTION] = parse_option,
|
||||||
|
[COL_ID] = parse_id,
|
||||||
|
[COL_NAME] = parse_name,
|
||||||
|
[COL_TYPE] = parse_type,
|
||||||
|
[COL_IP] = parse_ip,
|
||||||
|
[COL_OFFSET] = parse_offset,
|
||||||
|
};
|
||||||
|
|
||||||
static int parse_flash_layout(struct stm32prog_data *data,
|
static int parse_flash_layout(struct stm32prog_data *data,
|
||||||
ulong addr,
|
ulong addr,
|
||||||
ulong size)
|
ulong size)
|
||||||
{
|
{
|
||||||
return -ENODEV;
|
int column = 0, part_nb = 0, ret;
|
||||||
|
bool end_of_line, eof;
|
||||||
|
char *p, *start, *last, *col;
|
||||||
|
struct stm32prog_part_t *part;
|
||||||
|
int part_list_size;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
data->part_nb = 0;
|
||||||
|
|
||||||
|
/* check if STM32image is detected */
|
||||||
|
if (!stm32prog_header_check((struct raw_header_s *)addr,
|
||||||
|
&data->header)) {
|
||||||
|
u32 checksum;
|
||||||
|
|
||||||
|
addr = addr + BL_HEADER_SIZE;
|
||||||
|
size = data->header.image_length;
|
||||||
|
|
||||||
|
checksum = stm32prog_header_checksum(addr, &data->header);
|
||||||
|
if (checksum != data->header.image_checksum) {
|
||||||
|
stm32prog_err("Layout: invalid checksum : 0x%x expected 0x%x",
|
||||||
|
checksum, data->header.image_checksum);
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!size)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
start = (char *)addr;
|
||||||
|
last = start + size;
|
||||||
|
|
||||||
|
*last = 0x0; /* force null terminated string */
|
||||||
|
pr_debug("flash layout =\n%s\n", start);
|
||||||
|
|
||||||
|
/* calculate expected number of partitions */
|
||||||
|
part_list_size = 1;
|
||||||
|
p = start;
|
||||||
|
while (*p && (p < last)) {
|
||||||
|
if (*p++ == '\n') {
|
||||||
|
part_list_size++;
|
||||||
|
if (p < last && *p == '#')
|
||||||
|
part_list_size--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (part_list_size > PHASE_LAST_USER) {
|
||||||
|
stm32prog_err("Layout: too many partition (%d)",
|
||||||
|
part_list_size);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
part = calloc(sizeof(struct stm32prog_part_t), part_list_size);
|
||||||
|
if (!part) {
|
||||||
|
stm32prog_err("Layout: alloc failed");
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
data->part_array = part;
|
||||||
|
|
||||||
|
/* main parsing loop */
|
||||||
|
i = 1;
|
||||||
|
eof = false;
|
||||||
|
p = start;
|
||||||
|
col = start; /* 1st column */
|
||||||
|
end_of_line = false;
|
||||||
|
while (!eof) {
|
||||||
|
switch (*p) {
|
||||||
|
/* CR is ignored and replaced by NULL character */
|
||||||
|
case '\r':
|
||||||
|
*p = '\0';
|
||||||
|
p++;
|
||||||
|
continue;
|
||||||
|
case '\0':
|
||||||
|
end_of_line = true;
|
||||||
|
eof = true;
|
||||||
|
break;
|
||||||
|
case '\n':
|
||||||
|
end_of_line = true;
|
||||||
|
break;
|
||||||
|
case '\t':
|
||||||
|
break;
|
||||||
|
case '#':
|
||||||
|
/* comment line is skipped */
|
||||||
|
if (column == 0 && p == col) {
|
||||||
|
while ((p < last) && *p)
|
||||||
|
if (*p++ == '\n')
|
||||||
|
break;
|
||||||
|
col = p;
|
||||||
|
i++;
|
||||||
|
if (p >= last || !*p) {
|
||||||
|
eof = true;
|
||||||
|
end_of_line = true;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
/* fall through */
|
||||||
|
/* by default continue with the next character */
|
||||||
|
default:
|
||||||
|
p++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* replace by \0: allow string parsing for each column */
|
||||||
|
*p = '\0';
|
||||||
|
p++;
|
||||||
|
if (p >= last) {
|
||||||
|
eof = true;
|
||||||
|
end_of_line = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* skip empty line and multiple TAB in tsv file */
|
||||||
|
if (strlen(col) == 0) {
|
||||||
|
col = p;
|
||||||
|
/* skip empty line */
|
||||||
|
if (column == 0 && end_of_line) {
|
||||||
|
end_of_line = false;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (column < COL_NB_STM32) {
|
||||||
|
ret = parse[column](data, i, col, part);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* save the beginning of the next column */
|
||||||
|
column++;
|
||||||
|
col = p;
|
||||||
|
|
||||||
|
if (!end_of_line)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* end of the line detected */
|
||||||
|
end_of_line = false;
|
||||||
|
|
||||||
|
if (column < COL_NB_STM32) {
|
||||||
|
stm32prog_err("Layout line %d: no enought column", i);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
column = 0;
|
||||||
|
part_nb++;
|
||||||
|
part++;
|
||||||
|
i++;
|
||||||
|
if (part_nb >= part_list_size) {
|
||||||
|
part = NULL;
|
||||||
|
if (!eof) {
|
||||||
|
stm32prog_err("Layout: no enought memory for %d part",
|
||||||
|
part_nb);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
data->part_nb = part_nb;
|
||||||
|
if (data->part_nb == 0) {
|
||||||
|
stm32prog_err("Layout: no partition found");
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int __init part_cmp(void *priv, struct list_head *a, struct list_head *b)
|
static int __init part_cmp(void *priv, struct list_head *a, struct list_head *b)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue