/*- * Copyright (c) 2015 Oleksandr Tymoshenko * 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. * */ char *prop_offsets_ptr; int len = 0; prop_offsets_ptr = ufdt_node_get_fdt_prop_data(local_fixup_prop_node, &len); char *prop_data; int target_length = 0; prop_data = ufdt_node_get_fdt_prop_data(target_prop_node, &target_length); if (prop_offsets_ptr == NULL || 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 = ; * }; * }; * path2 { * to2 { * local_ref2 = ; * }; * }; * }; * * 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; }