49a4450563
When updating firmware, it is very often that we may want to preserve few sections, for example vital product data (VPD) including serial number, calibration data and cache. A firmware updater has to hard-code the section names that need to be preserved and is hard to maintain. A better approach is to specify that in FMAP area flags (the `area_flag` field) using FMAP_AREA_PRESERVE. With this patchset, a FMD parser flag "PRESERVE" is introduced and will be converted to FMAP_AREA_PRESERVE when generating FMAP data (by fmap_from_fmd.c). For example, The FMD statement: RO_VPD(PRESERVE)@0x0 16k will generate an FMAP firmware section that: area_name = "RO_VPD" area_offset = 0 area_size = 16384 area_flags = FMAP_AREA_PRESERVE BUG=chromium:936768 TEST=make; boots on x86 "google/eve" and arm "google/kukui" devices Manually added 'PRESERVE' to some FMD files, and verify (by running fmap.py) the output coreboot.rom has FMAP_AREA_PRESERVE set Change-Id: I51e7d31029b98868a1cab0d26bf04a14db01b1c0 Signed-off-by: Hung-Te Lin <hungte@chromium.org> Reviewed-on: https://review.coreboot.org/c/coreboot/+/31707 Tested-by: build bot (Jenkins) <no-reply@coreboot.org> Reviewed-by: Julius Werner <jwerner@chromium.org>
78 lines
1.9 KiB
Text
78 lines
1.9 KiB
Text
/*
|
|
* fmd_scanner.l, scanner generator for flashmap descriptor language
|
|
*
|
|
* Copyright (C) 2015 Google, Inc.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; version 2 of the License.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*/
|
|
|
|
%{
|
|
#include "fmd_parser.h"
|
|
|
|
#include <assert.h>
|
|
#include <string.h>
|
|
|
|
int parse_integer(char *src, int base);
|
|
int copy_string(const char *src);
|
|
%}
|
|
|
|
%option noyywrap
|
|
%s FLAGS
|
|
|
|
MULTIPLIER [KMG]
|
|
|
|
%%
|
|
[[:space:]]+ /* Eat whitespace. */
|
|
#.*$ /* Eat comments. */
|
|
\( BEGIN(FLAGS); return *yytext;
|
|
<FLAGS>\) BEGIN(INITIAL); return *yytext;
|
|
<FLAGS>CBFS return FLAG_CBFS;
|
|
<FLAGS>PRESERVE return FLAG_PRESERVE;
|
|
0{MULTIPLIER}? |
|
|
[1-9][0-9]*{MULTIPLIER}? return parse_integer(yytext, 10);
|
|
0[0-9]+{MULTIPLIER}? return OCTAL;
|
|
0[xX][0-9a-fA-F]+{MULTIPLIER}? return parse_integer(yytext + 2, 16);
|
|
[^#@{}()[:space:]]* return copy_string(yytext);
|
|
. return *yytext;
|
|
|
|
%%
|
|
|
|
int parse_integer(char *src, int base)
|
|
{
|
|
char *multiplier = NULL;
|
|
unsigned val = strtoul(src, &multiplier, base);
|
|
|
|
if (*multiplier) {
|
|
switch(*multiplier) {
|
|
case 'K':
|
|
val *= 1024;
|
|
break;
|
|
case 'M':
|
|
val *= 1024*1024;
|
|
break;
|
|
case 'G':
|
|
val *= 1024*1024*1024;
|
|
break;
|
|
default:
|
|
// If we ever get here, the MULTIPLIER regex is allowing
|
|
// multiplier suffixes not handled by this code.
|
|
assert(false);
|
|
}
|
|
}
|
|
|
|
yylval.intval = val;
|
|
return INTEGER;
|
|
}
|
|
|
|
int copy_string(const char *src)
|
|
{
|
|
yylval.strval = strdup(src);
|
|
return STRING;
|
|
}
|