9497fcb742
The idea of "annotation" for firmware sections was pretty flexible, but in future we will want multiple attributes applied to same area. For example, indicate the section must be preserved when updating firmware so serial number or MAC address can be preserved. The solution here is to extend annotation so it can take multiple identifiers (flags) in a row. For example, to declare a 64KB COREBOOT section as CBFS using annotation: COREBOOT(CBFS)@0x0 64k If there's a new flag "PRESERVE" indicating the section must be preserved before update, we can declare it following CBFS flag: COREBOOT(CBFS PRESERVE)@0x0 64k The flags are directly parsed in fmd_parser, and stored in an union flashmap_flags. Output modules can choose to ignore or process the flags. Currently the only supported flag is "CBFS" (for backward compatible with annotation). There will be more new flags in follow up patches. BUG=chromium:936768 TEST=make; boots on x86 "google/eve" and arm "google/kukui" devices Change-Id: Ie2d99f570e6faff6ed3a4344d6af7526a4515fae Signed-off-by: Hung-Te Lin <hungte@chromium.org> Reviewed-on: https://review.coreboot.org/c/coreboot/+/31706 Reviewed-by: Julius Werner <jwerner@chromium.org> Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
145 lines
5.3 KiB
C
145 lines
5.3 KiB
C
/*
|
|
* fmd.h, parser frontend and utility functions 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.
|
|
*/
|
|
|
|
#ifndef FMD_H_
|
|
#define FMD_H_
|
|
|
|
#include <limits.h>
|
|
#include <stdbool.h>
|
|
#include <stddef.h>
|
|
#include <stdio.h>
|
|
|
|
#define FMD_NOTFOUND UINT_MAX
|
|
|
|
/**
|
|
* Flags used by flashmap_descriptor.
|
|
* These flags can be set by adding (NAME) after description name.
|
|
* For example, declaring a CBFS section named as COREBOOT for 16k:
|
|
* COREBOOT(CBFS) 16k
|
|
*/
|
|
union flashmap_flags {
|
|
struct {
|
|
int cbfs: 1; /* The section contains a CBFS area. */
|
|
} f;
|
|
int v;
|
|
};
|
|
|
|
struct flashmap_descriptor {
|
|
char *name;
|
|
bool offset_known;
|
|
/**
|
|
* Offset relative to that of the parent node.
|
|
* Exception: for the root node in the descriptor tree, it is optional.
|
|
* In this case, if absent, it indicates that the flash chip will not be
|
|
* memory mapped at runtime; otherwise, its value indicates the base
|
|
* address of the flash chip in the virtual address space rather than
|
|
* representing an offset into the flash image itself.
|
|
* It is an error to read this field unless offset_known is set.
|
|
*/
|
|
unsigned offset;
|
|
bool size_known;
|
|
/** It is an error to read this field unless size_known is set. */
|
|
unsigned size;
|
|
size_t list_len;
|
|
union flashmap_flags flags;
|
|
/** It is an error to dereference this array if list_len is 0. */
|
|
struct flashmap_descriptor **list;
|
|
};
|
|
|
|
/**
|
|
* **Client-defined** callback for flag "CBFS".
|
|
* This call is used to notify client code that the user has requested the given
|
|
* section node to be flagged with "CBFS". Implementations of this function
|
|
* should use their return type to tell the compiler whether the flag can be
|
|
* applied and can perform whatever actions are necessary.
|
|
* It's worth reiterating that this is only called on section nodes, and will
|
|
* never be called with the final, complete flashmap_descriptor because
|
|
* it is impossible to set flags for the image as a whole.
|
|
*
|
|
* @param flashmap_descriptor The section node with flag set
|
|
* @return Whether this flag can be applied
|
|
*/
|
|
bool fmd_process_flag_cbfs(const struct flashmap_descriptor *node);
|
|
|
|
/**
|
|
* Parse and validate a flashmap descriptor from the specified stream.
|
|
* As part of this process, any fields that were omitted in the input are
|
|
* inferred from whatever information is known, if possible. The result is a
|
|
* tree with all its offset and size fields filled, except possibly the former
|
|
* part of the root node in the case of non--memory mapped flash. If a syntax
|
|
* error causes the parser to fail, or if there is not enough information given
|
|
* in the input file to determine any single missing value, the specific error
|
|
* is reported to standard error and this function returns NULL.
|
|
*
|
|
* @param stream File from which to read the (partial) flashmap descriptor
|
|
* @return Populated flashmap descriptor tree, or NULL on failure
|
|
*/
|
|
struct flashmap_descriptor *fmd_create(FILE *stream);
|
|
|
|
/** @param victim Valid descriptor tree to be cleaned up, or NULL for no-op */
|
|
void fmd_cleanup(struct flashmap_descriptor *victim);
|
|
|
|
/**
|
|
* @param tree Must be non-NULL
|
|
* @return The number of nodes in the tree, including the root
|
|
*/
|
|
size_t fmd_count_nodes(const struct flashmap_descriptor *tree);
|
|
|
|
/**
|
|
* @param root The flashmap descriptor to search
|
|
* @param name The name of the sought-after section
|
|
* @return The desired section node, or NULL if none was found
|
|
*/
|
|
const struct flashmap_descriptor *fmd_find_node(
|
|
const struct flashmap_descriptor *root, const char *name);
|
|
|
|
/**
|
|
* @param root Parent node to whose start the "absolute" offset will be relative
|
|
* @param name The name of the node whose offset to determine
|
|
* @return The "absolute" offset, or FMD_NOTFOUND if the node wasn't found
|
|
*/
|
|
unsigned fmd_calc_absolute_offset(const struct flashmap_descriptor *root,
|
|
const char *name);
|
|
|
|
/** @param tree Must be non-NULL */
|
|
void fmd_print(const struct flashmap_descriptor *tree);
|
|
|
|
typedef struct flashmap_descriptor **flashmap_descriptor_iterator_t;
|
|
|
|
/*
|
|
* Run the subsequent statement once on each descendant of the specified node.
|
|
*
|
|
* @param iterator A flashmap_descriptor_iterator_t (automatically declared)
|
|
* @param parent The parent node of those over which the loop should iterate
|
|
*/
|
|
#define fmd_foreach_child_iterator(iterator, parent) \
|
|
for (flashmap_descriptor_iterator_t iterator = parent->list; \
|
|
iterator < parent->list + parent->list_len; ++iterator)
|
|
|
|
/*
|
|
* Run the subsequent statement once on each descendant of the specified node.
|
|
*
|
|
* @param child A struct flashmap_descriptor * (automatically declared)
|
|
* @param parent The parent node of those over which the loop should iterate
|
|
*/
|
|
#define fmd_foreach_child(child, parent) \
|
|
for (struct flashmap_descriptor **fmd_foreach_child_iterator_ = \
|
|
parent->list, *child = NULL; \
|
|
fmd_foreach_child_iterator_ < \
|
|
parent->list + parent->list_len && \
|
|
(child = *fmd_foreach_child_iterator_); \
|
|
++fmd_foreach_child_iterator_)
|
|
|
|
#endif
|