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
@ -24,6 +24,17 @@
|
||||
|
||||
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)
|
||||
{
|
||||
static const char error_msg[] = "Unspecified";
|
||||
@ -34,11 +45,370 @@ char *stm32prog_get_error(struct stm32prog_data *data)
|
||||
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,
|
||||
ulong addr,
|
||||
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)
|
||||
|
Loading…
Reference in New Issue
Block a user