687 lines
20 KiB
C
687 lines
20 KiB
C
/*-
|
|
* Copyright (c) 2015 Oleksandr Tymoshenko <gonzo@FreeBSD.org>
|
|
* All rights reserved.
|
|
*
|
|
* This software was developed by Semihalf under sponsorship from
|
|
* the FreeBSD Foundation.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
#include "ufdt_overlay.h"
|
|
#include "libufdt.h"
|
|
#include "ufdt_node_pool.h"
|
|
#include "ufdt_overlay_internal.h"
|
|
|
|
/*
|
|
* The original version of fdt_overlay.c is slow in searching for particular
|
|
* nodes and adding subnodes/properties due to the operations on flattened
|
|
* device tree (FDT).
|
|
*
|
|
* Here we introduce `libufdt` which builds a real tree structure (named
|
|
* ufdt -- unflattned device tree) from FDT. In the real tree, we can perform
|
|
* certain operations (e.g., merge 2 subtrees, search for a node by path) in
|
|
* almost optimal time complexity with acceptable additional memory usage.
|
|
*
|
|
* This file is the improved version of fdt_overlay.c by using the real tree
|
|
* structure defined in libufdt.
|
|
*
|
|
* How the device tree overlay works and some
|
|
* special terms (e.g., fixups, local fixups, fragment, etc)
|
|
* are described in the document
|
|
* external/dtc/Documentation/dt-object-internal.txt.
|
|
*/
|
|
|
|
/* BEGIN of operations about phandles in ufdt. */
|
|
|
|
/*
|
|
* Increases u32 value at pos by offset.
|
|
*/
|
|
static void fdt_increase_u32(void *pos, uint32_t offset) {
|
|
uint32_t val;
|
|
|
|
dto_memcpy(&val, pos, sizeof(val));
|
|
val = cpu_to_fdt32(fdt32_to_cpu(val) + offset);
|
|
dto_memcpy(pos, &val, sizeof(val));
|
|
}
|
|
|
|
/*
|
|
* Gets the max phandle of a given ufdt.
|
|
*/
|
|
uint32_t ufdt_get_max_phandle(struct ufdt *tree) {
|
|
struct ufdt_static_phandle_table sorted_table = tree->phandle_table;
|
|
if (sorted_table.len > 0)
|
|
return sorted_table.data[sorted_table.len - 1].phandle;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Tries to increase the phandle value of a node
|
|
* if the phandle exists.
|
|
*/
|
|
static void ufdt_node_try_increase_phandle(struct ufdt_node *node,
|
|
uint32_t offset) {
|
|
int len = 0;
|
|
char *prop_data = ufdt_node_get_fdt_prop_data_by_name(node, "phandle", &len);
|
|
if (prop_data != NULL && len == sizeof(fdt32_t)) {
|
|
fdt_increase_u32(prop_data, offset);
|
|
}
|
|
prop_data = ufdt_node_get_fdt_prop_data_by_name(node, "linux,phandle", &len);
|
|
if (prop_data != NULL && len == sizeof(fdt32_t)) {
|
|
fdt_increase_u32(prop_data, offset);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Increases all phandles by offset in a ufdt
|
|
* in O(n) time.
|
|
*/
|
|
void ufdt_try_increase_phandle(struct ufdt *tree, uint32_t offset) {
|
|
struct ufdt_static_phandle_table sorted_table = tree->phandle_table;
|
|
int i;
|
|
|
|
for (i = 0; i < sorted_table.len; i++) {
|
|
struct ufdt_node *target_node = sorted_table.data[i].node;
|
|
|
|
ufdt_node_try_increase_phandle(target_node, offset);
|
|
}
|
|
}
|
|
|
|
/* END of operations about phandles in ufdt. */
|
|
|
|
/*
|
|
* In the overlay_tree, there are some references (phandle)
|
|
* pointing to somewhere in the main_tree.
|
|
* Fix-up operations is to resolve the right address
|
|
* in the overlay_tree.
|
|
*/
|
|
|
|
/* BEGIN of doing fixup in the overlay ufdt. */
|
|
|
|
/*
|
|
* Returns exact memory location specified by fixup in format
|
|
* /path/to/node:property:offset.
|
|
* A property might contain multiple values and the offset is used to locate a
|
|
* reference inside the property.
|
|
* e.g.,
|
|
* "property"=<1, 2, &ref, 4>, we can use /path/to/node:property:8 to get ref,
|
|
* where 8 is sizeof(uint32) + sizeof(unit32).
|
|
*/
|
|
void *ufdt_get_fixup_location(struct ufdt *tree, const char *fixup) {
|
|
char *path, *prop_ptr, *offset_ptr, *end_ptr;
|
|
int prop_offset, prop_len;
|
|
const char *prop_data;
|
|
char path_buf[1024];
|
|
char *path_mem = NULL;
|
|
|
|
size_t fixup_len = strlen(fixup) + 1;
|
|
if (fixup_len > sizeof(path_buf)) {
|
|
path_mem = dto_malloc(fixup_len);
|
|
path = path_mem;
|
|
} else {
|
|
path = path_buf;
|
|
}
|
|
dto_memcpy(path, fixup, fixup_len);
|
|
|
|
prop_ptr = dto_strchr(path, ':');
|
|
if (prop_ptr == NULL) {
|
|
dto_error("Missing property part in '%s'\n", path);
|
|
goto fail;
|
|
}
|
|
|
|
*prop_ptr = '\0';
|
|
prop_ptr++;
|
|
|
|
offset_ptr = dto_strchr(prop_ptr, ':');
|
|
if (offset_ptr == NULL) {
|
|
dto_error("Missing offset part in '%s'\n", path);
|
|
goto fail;
|
|
}
|
|
|
|
*offset_ptr = '\0';
|
|
offset_ptr++;
|
|
|
|
prop_offset = dto_strtoul(offset_ptr, &end_ptr, 10 /* base */);
|
|
if (*end_ptr != '\0') {
|
|
dto_error("'%s' is not valid number\n", offset_ptr);
|
|
goto fail;
|
|
}
|
|
|
|
struct ufdt_node *target_node;
|
|
target_node = ufdt_get_node_by_path(tree, path);
|
|
if (target_node == NULL) {
|
|
dto_error("Path '%s' not found\n", path);
|
|
goto fail;
|
|
}
|
|
|
|
prop_data =
|
|
ufdt_node_get_fdt_prop_data_by_name(target_node, prop_ptr, &prop_len);
|
|
if (prop_data == NULL) {
|
|
dto_error("Property '%s' not found in '%s' node\n", prop_ptr, path);
|
|
goto fail;
|
|
}
|
|
/*
|
|
* Note that prop_offset is the offset inside the property data.
|
|
*/
|
|
if (prop_len < prop_offset + (int)sizeof(uint32_t)) {
|
|
dto_error("%s: property length is too small for fixup\n", path);
|
|
goto fail;
|
|
}
|
|
|
|
if (path_mem) dto_free(path_mem);
|
|
return (char *)prop_data + prop_offset;
|
|
|
|
fail:
|
|
if (path_mem) dto_free(path_mem);
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Process one entry in __fixups__ { } node.
|
|
* @fixups is property value, array of NUL-terminated strings
|
|
* with fixup locations.
|
|
* @fixups_len length of the fixups array in bytes.
|
|
* @phandle is value for these locations.
|
|
*/
|
|
int ufdt_do_one_fixup(struct ufdt *tree, const char *fixups, int fixups_len,
|
|
int phandle) {
|
|
void *fixup_pos;
|
|
uint32_t val;
|
|
|
|
val = cpu_to_fdt32(phandle);
|
|
|
|
while (fixups_len > 0) {
|
|
fixup_pos = ufdt_get_fixup_location(tree, fixups);
|
|
if (fixup_pos != NULL) {
|
|
dto_memcpy(fixup_pos, &val, sizeof(val));
|
|
} else {
|
|
return -1;
|
|
}
|
|
|
|
fixups_len -= dto_strlen(fixups) + 1;
|
|
fixups += dto_strlen(fixups) + 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Handle __fixups__ node in overlay tree.
|
|
*/
|
|
|
|
int ufdt_overlay_do_fixups(struct ufdt *main_tree, struct ufdt *overlay_tree) {
|
|
int len = 0;
|
|
struct ufdt_node *overlay_fixups_node =
|
|
ufdt_get_node_by_path(overlay_tree, "/__fixups__");
|
|
if (!overlay_fixups_node) {
|
|
/* There is no __fixups__. Do nothing. */
|
|
return 0;
|
|
}
|
|
|
|
struct ufdt_node *main_symbols_node =
|
|
ufdt_get_node_by_path(main_tree, "/__symbols__");
|
|
|
|
struct ufdt_node **it;
|
|
for_each_prop(it, overlay_fixups_node) {
|
|
/* Find the first property */
|
|
|
|
/* Check __symbols__ is exist when we have any property in __fixups__ */
|
|
if (!main_symbols_node) {
|
|
dto_error("No node __symbols__ in main dtb.\n");
|
|
return -1;
|
|
}
|
|
break;
|
|
}
|
|
|
|
for_each_prop(it, overlay_fixups_node) {
|
|
/*
|
|
* A property in __fixups__ looks like:
|
|
* symbol_name =
|
|
* "/path/to/node:prop:offset0\x00/path/to/node:prop:offset1..."
|
|
* So we firstly find the node "symbol_name" and obtain its phandle in
|
|
* __symbols__ of the main_tree.
|
|
*/
|
|
|
|
struct ufdt_node *fixups = *it;
|
|
char *symbol_path = ufdt_node_get_fdt_prop_data_by_name(
|
|
main_symbols_node, ufdt_node_name(fixups), &len);
|
|
|
|
if (!symbol_path) {
|
|
dto_error("Couldn't find '%s' symbol in main dtb\n",
|
|
ufdt_node_name(fixups));
|
|
return -1;
|
|
}
|
|
|
|
struct ufdt_node *symbol_node;
|
|
symbol_node = ufdt_get_node_by_path(main_tree, symbol_path);
|
|
|
|
if (!symbol_node) {
|
|
dto_error("Couldn't find '%s' path in main dtb\n", symbol_path);
|
|
return -1;
|
|
}
|
|
|
|
uint32_t phandle = ufdt_node_get_phandle(symbol_node);
|
|
|
|
const char *fixups_paths = ufdt_node_get_fdt_prop_data(fixups, &len);
|
|
|
|
if (ufdt_do_one_fixup(overlay_tree, fixups_paths, len, phandle) < 0) {
|
|
dto_error("Failed one fixup in ufdt_do_one_fixup\n");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* END of doing fixup in the overlay ufdt. */
|
|
|
|
/*
|
|
* Here is to overlay all fragments in the overlay_tree to the main_tree.
|
|
* What is "overlay fragment"? The main purpose is to add some subtrees to the
|
|
* main_tree in order to complete the entire device tree.
|
|
*
|
|
* A fragment consists of two parts: 1. the subtree to be added 2. where it
|
|
* should be added.
|
|
*
|
|
* Overlaying a fragment requires: 1. find the node in the main_tree 2. merge
|
|
* the subtree into that node in the main_tree.
|
|
*/
|
|
|
|
/* BEGIN of applying fragments. */
|
|
|
|
/*
|
|
* Overlay the overlay_node over target_node.
|
|
*/
|
|
static int ufdt_overlay_node(struct ufdt_node *target_node,
|
|
struct ufdt_node *overlay_node,
|
|
struct ufdt_node_pool *pool) {
|
|
return ufdt_node_merge_into(target_node, overlay_node, pool);
|
|
}
|
|
|
|
enum overlay_result ufdt_overlay_get_target(struct ufdt *tree,
|
|
struct ufdt_node *frag_node,
|
|
struct ufdt_node **target_node) {
|
|
uint32_t target;
|
|
const char *target_path;
|
|
const void *val;
|
|
*target_node = NULL;
|
|
|
|
val = ufdt_node_get_fdt_prop_data_by_name(frag_node, "target", NULL);
|
|
if (val) {
|
|
dto_memcpy(&target, val, sizeof(target));
|
|
target = fdt32_to_cpu(target);
|
|
*target_node = ufdt_get_node_by_phandle(tree, target);
|
|
if (*target_node == NULL) {
|
|
dto_error("failed to find target %04x\n", target);
|
|
return OVERLAY_RESULT_TARGET_INVALID;
|
|
}
|
|
}
|
|
|
|
if (*target_node == NULL) {
|
|
target_path =
|
|
ufdt_node_get_fdt_prop_data_by_name(frag_node, "target-path", NULL);
|
|
if (target_path == NULL) {
|
|
return OVERLAY_RESULT_MISSING_TARGET;
|
|
}
|
|
|
|
*target_node = ufdt_get_node_by_path(tree, target_path);
|
|
if (*target_node == NULL) {
|
|
dto_error("failed to find target-path %s\n", target_path);
|
|
return OVERLAY_RESULT_TARGET_PATH_INVALID;
|
|
}
|
|
}
|
|
|
|
return OVERLAY_RESULT_OK;
|
|
}
|
|
|
|
/*
|
|
* Apply one overlay fragment (subtree).
|
|
*/
|
|
static enum overlay_result ufdt_apply_fragment(struct ufdt *tree,
|
|
struct ufdt_node *frag_node,
|
|
struct ufdt_node_pool *pool) {
|
|
struct ufdt_node *target_node = NULL;
|
|
struct ufdt_node *overlay_node = NULL;
|
|
|
|
overlay_node = ufdt_node_get_node_by_path(frag_node, "__overlay__");
|
|
if (overlay_node == NULL) {
|
|
return OVERLAY_RESULT_MISSING_OVERLAY;
|
|
}
|
|
|
|
enum overlay_result result =
|
|
ufdt_overlay_get_target(tree, frag_node, &target_node);
|
|
if (target_node == NULL) {
|
|
dto_error("Unable to resolve target for %s\n", ufdt_node_name(frag_node));
|
|
return result;
|
|
}
|
|
|
|
int err = ufdt_overlay_node(target_node, overlay_node, pool);
|
|
|
|
if (err < 0) {
|
|
dto_error("failed to overlay node %s to target %s\n",
|
|
ufdt_node_name(overlay_node), ufdt_node_name(target_node));
|
|
return OVERLAY_RESULT_MERGE_FAIL;
|
|
}
|
|
|
|
return OVERLAY_RESULT_OK;
|
|
}
|
|
|
|
/*
|
|
* Applies all fragments to the main_tree.
|
|
*/
|
|
static int ufdt_overlay_apply_fragments(struct ufdt *main_tree,
|
|
struct ufdt *overlay_tree,
|
|
struct ufdt_node_pool *pool) {
|
|
enum overlay_result ret;
|
|
struct ufdt_node **it;
|
|
/*
|
|
* This loop may iterate to subnodes that's not a fragment node.
|
|
* We must fail for any other error.
|
|
*/
|
|
for_each_node(it, overlay_tree->root) {
|
|
ret = ufdt_apply_fragment(main_tree, *it, pool);
|
|
if ((ret != OVERLAY_RESULT_OK) && (ret != OVERLAY_RESULT_MISSING_OVERLAY)) {
|
|
dto_error("failed to apply overlay fragment %s ret: %d\n",
|
|
ufdt_node_name(*it), ret);
|
|
return -1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* END of applying fragments. */
|
|
|
|
/*
|
|
* Since the overlay_tree will be "merged" into the main_tree, some
|
|
* references (e.g., phandle values that acts as an unique ID) need to be
|
|
* updated so it won't lead to collision that different nodes have the same
|
|
* phandle value.
|
|
*
|
|
* Two things need to be done:
|
|
*
|
|
* 1. ufdt_try_increase_phandle()
|
|
* Update phandle (an unique integer ID of a node in the device tree) of each
|
|
* node in the overlay_tree. To achieve this, we simply increase each phandle
|
|
* values in the overlay_tree by the max phandle value of the main_tree.
|
|
*
|
|
* 2. ufdt_overlay_do_local_fixups()
|
|
* If there are some reference in the overlay_tree that references nodes
|
|
* inside the overlay_tree, we have to modify the reference value (address of
|
|
* the referenced node: phandle) so that it corresponds to the right node inside
|
|
* the overlay_tree. Where the reference exists is kept in __local_fixups__ node
|
|
* in the overlay_tree.
|
|
*/
|
|
|
|
/* BEGIN of updating local references (phandle values) in the overlay ufdt. */
|
|
|
|
/*
|
|
* local fixups
|
|
*/
|
|
static int ufdt_local_fixup_prop(struct ufdt_node *target_prop_node,
|
|
struct ufdt_node *local_fixup_prop_node,
|
|
uint32_t phandle_offset) {
|
|
/*
|
|
* prop_offsets_ptr should be a list of fdt32_t.
|
|
* <offset0 offset1 offset2 ...>
|
|
*/
|
|
char *prop_offsets_ptr;
|
|
int len = 0;
|
|
prop_offsets_ptr = ufdt_node_get_fdt_prop_data(local_fixup_prop_node, &len);
|
|
|
|
if (prop_offsets_ptr == NULL || len % sizeof(fdt32_t) != 0) return -1;
|
|
|
|
char *prop_data;
|
|
int target_length = 0;
|
|
|
|
prop_data = ufdt_node_get_fdt_prop_data(target_prop_node, &target_length);
|
|
|
|
if (prop_data == NULL) return -1;
|
|
|
|
int i;
|
|
for (i = 0; i < len; i += sizeof(fdt32_t)) {
|
|
int offset = fdt32_to_cpu(*(fdt32_t *)(prop_offsets_ptr + i));
|
|
if (offset + sizeof(fdt32_t) > (size_t)target_length) return -1;
|
|
fdt_increase_u32((prop_data + offset), phandle_offset);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int ufdt_local_fixup_node(struct ufdt_node *target_node,
|
|
struct ufdt_node *local_fixups_node,
|
|
uint32_t phandle_offset) {
|
|
if (local_fixups_node == NULL) return 0;
|
|
|
|
struct ufdt_node **it_local_fixups;
|
|
struct ufdt_node *sub_target_node;
|
|
|
|
for_each_prop(it_local_fixups, local_fixups_node) {
|
|
sub_target_node = ufdt_node_get_property_by_name(
|
|
target_node, ufdt_node_name(*it_local_fixups));
|
|
|
|
if (sub_target_node != NULL) {
|
|
int err = ufdt_local_fixup_prop(sub_target_node, *it_local_fixups,
|
|
phandle_offset);
|
|
if (err < 0) return -1;
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
for_each_node(it_local_fixups, local_fixups_node) {
|
|
sub_target_node = ufdt_node_get_node_by_path(
|
|
target_node, ufdt_node_name(*it_local_fixups));
|
|
if (sub_target_node != NULL) {
|
|
int err = ufdt_local_fixup_node(sub_target_node, *it_local_fixups,
|
|
phandle_offset);
|
|
if (err < 0) return -1;
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Handle __local_fixups__ node in overlay DTB
|
|
* The __local_fixups__ format we expect is
|
|
* __local_fixups__ {
|
|
* path {
|
|
* to {
|
|
* local_ref1 = <offset>;
|
|
* };
|
|
* };
|
|
* path2 {
|
|
* to2 {
|
|
* local_ref2 = <offset1 offset2 ...>;
|
|
* };
|
|
* };
|
|
* };
|
|
*
|
|
* which follows the dtc patch from:
|
|
* https://marc.info/?l=devicetree&m=144061468601974&w=4
|
|
*/
|
|
int ufdt_overlay_do_local_fixups(struct ufdt *tree, uint32_t phandle_offset) {
|
|
struct ufdt_node *overlay_node = ufdt_get_node_by_path(tree, "/");
|
|
struct ufdt_node *local_fixups_node =
|
|
ufdt_get_node_by_path(tree, "/__local_fixups__");
|
|
|
|
int err =
|
|
ufdt_local_fixup_node(overlay_node, local_fixups_node, phandle_offset);
|
|
|
|
if (err < 0) return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ufdt_overlay_local_ref_update(struct ufdt *main_tree,
|
|
struct ufdt *overlay_tree) {
|
|
uint32_t phandle_offset = 0;
|
|
|
|
phandle_offset = ufdt_get_max_phandle(main_tree);
|
|
if (phandle_offset > 0) {
|
|
ufdt_try_increase_phandle(overlay_tree, phandle_offset);
|
|
}
|
|
|
|
int err = ufdt_overlay_do_local_fixups(overlay_tree, phandle_offset);
|
|
if (err < 0) {
|
|
dto_error("failed to perform local fixups in overlay\n");
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* END of updating local references (phandle values) in the overlay ufdt. */
|
|
|
|
static int _ufdt_overlay_fdtps(struct ufdt *main_tree,
|
|
const struct ufdt *overlay_tree) {
|
|
for (int i = 0; i < overlay_tree->num_used_fdtps; i++) {
|
|
void *fdt = overlay_tree->fdtps[i];
|
|
if (ufdt_add_fdt(main_tree, fdt) < 0) {
|
|
return -1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int ufdt_overlay_apply(struct ufdt *main_tree, struct ufdt *overlay_tree,
|
|
size_t overlay_length,
|
|
struct ufdt_node_pool *pool) {
|
|
if (_ufdt_overlay_fdtps(main_tree, overlay_tree) < 0) {
|
|
dto_error("failed to add more fdt into main ufdt tree.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (overlay_length < sizeof(struct fdt_header)) {
|
|
dto_error("Overlay_length %zu smaller than header size %zu\n",
|
|
overlay_length, sizeof(struct fdt_header));
|
|
return -1;
|
|
}
|
|
|
|
if (ufdt_overlay_local_ref_update(main_tree, overlay_tree) < 0) {
|
|
dto_error("failed to perform local fixups in overlay\n");
|
|
return -1;
|
|
}
|
|
|
|
if (ufdt_overlay_do_fixups(main_tree, overlay_tree) < 0) {
|
|
dto_error("failed to perform fixups in overlay\n");
|
|
return -1;
|
|
}
|
|
if (ufdt_overlay_apply_fragments(main_tree, overlay_tree, pool) < 0) {
|
|
dto_error("failed to apply fragments\n");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct fdt_header *ufdt_install_blob(void *blob, size_t blob_size) {
|
|
struct fdt_header *pHeader;
|
|
int err;
|
|
|
|
dto_debug("ufdt_install_blob (0x%08jx)\n", (uintmax_t)blob);
|
|
|
|
if (blob_size < sizeof(struct fdt_header)) {
|
|
dto_error("Blob_size %zu smaller than the header size %zu\n", blob_size,
|
|
sizeof(struct fdt_header));
|
|
return NULL;
|
|
}
|
|
|
|
pHeader = (struct fdt_header *)blob;
|
|
err = fdt_check_header(pHeader);
|
|
if (err < 0) {
|
|
if (err == -FDT_ERR_BADVERSION) {
|
|
dto_error("incompatible blob version: %d, should be: %d",
|
|
fdt_version(pHeader), FDT_LAST_SUPPORTED_VERSION);
|
|
|
|
} else {
|
|
dto_error("error validating blob: %s", fdt_strerror(err));
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
return pHeader;
|
|
}
|
|
|
|
/*
|
|
* From Google, based on dt_overlay_apply() logic
|
|
* Will dto_malloc a new fdt blob and return it. Will not dto_free parameters.
|
|
*/
|
|
struct fdt_header *ufdt_apply_overlay(struct fdt_header *main_fdt_header,
|
|
size_t main_fdt_size,
|
|
void *overlay_fdtp,
|
|
size_t overlay_size) {
|
|
size_t out_fdt_size;
|
|
|
|
if (main_fdt_header == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
if (overlay_size < 8 || overlay_size != fdt_totalsize(overlay_fdtp)) {
|
|
dto_error("Bad overlay size!\n");
|
|
return NULL;
|
|
}
|
|
if (main_fdt_size < 8 || main_fdt_size != fdt_totalsize(main_fdt_header)) {
|
|
dto_error("Bad fdt size!\n");
|
|
return NULL;
|
|
}
|
|
|
|
out_fdt_size = fdt_totalsize(main_fdt_header) + overlay_size;
|
|
/* It's actually more than enough */
|
|
struct fdt_header *out_fdt_header = dto_malloc(out_fdt_size);
|
|
|
|
if (out_fdt_header == NULL) {
|
|
dto_error("failed to allocate memory for DTB blob with overlays\n");
|
|
return NULL;
|
|
}
|
|
|
|
struct ufdt_node_pool pool;
|
|
ufdt_node_pool_construct(&pool);
|
|
struct ufdt *main_tree = ufdt_from_fdt(main_fdt_header, main_fdt_size, &pool);
|
|
struct ufdt *overlay_tree = ufdt_from_fdt(overlay_fdtp, overlay_size, &pool);
|
|
int err = ufdt_overlay_apply(main_tree, overlay_tree, overlay_size, &pool);
|
|
if (err < 0) {
|
|
goto fail;
|
|
}
|
|
|
|
err = ufdt_to_fdt(main_tree, out_fdt_header, out_fdt_size);
|
|
if (err < 0) {
|
|
dto_error("Failed to dump the device tree to out_fdt_header\n");
|
|
goto fail;
|
|
}
|
|
|
|
ufdt_destruct(overlay_tree, &pool);
|
|
ufdt_destruct(main_tree, &pool);
|
|
ufdt_node_pool_destruct(&pool);
|
|
|
|
return out_fdt_header;
|
|
|
|
fail:
|
|
ufdt_destruct(overlay_tree, &pool);
|
|
ufdt_destruct(main_tree, &pool);
|
|
ufdt_node_pool_destruct(&pool);
|
|
dto_free(out_fdt_header);
|
|
|
|
return NULL;
|
|
}
|