xref: /aosp_15_r20/system/libufdt/ufdt_overlay.c (revision 13e8728f0cffde9369df671f7b293a048a99c7ed)
1*13e8728fSAndroid Build Coastguard Worker /*-
2*13e8728fSAndroid Build Coastguard Worker  * Copyright (c) 2015 Oleksandr Tymoshenko <[email protected]>
3*13e8728fSAndroid Build Coastguard Worker  * All rights reserved.
4*13e8728fSAndroid Build Coastguard Worker  *
5*13e8728fSAndroid Build Coastguard Worker  * This software was developed by Semihalf under sponsorship from
6*13e8728fSAndroid Build Coastguard Worker  * the FreeBSD Foundation.
7*13e8728fSAndroid Build Coastguard Worker  *
8*13e8728fSAndroid Build Coastguard Worker  * Redistribution and use in source and binary forms, with or without
9*13e8728fSAndroid Build Coastguard Worker  * modification, are permitted provided that the following conditions
10*13e8728fSAndroid Build Coastguard Worker  * are met:
11*13e8728fSAndroid Build Coastguard Worker  * 1. Redistributions of source code must retain the above copyright
12*13e8728fSAndroid Build Coastguard Worker  *    notice, this list of conditions and the following disclaimer.
13*13e8728fSAndroid Build Coastguard Worker  * 2. Redistributions in binary form must reproduce the above copyright
14*13e8728fSAndroid Build Coastguard Worker  *    notice, this list of conditions and the following disclaimer in the
15*13e8728fSAndroid Build Coastguard Worker  *    documentation and/or other materials provided with the distribution.
16*13e8728fSAndroid Build Coastguard Worker  *
17*13e8728fSAndroid Build Coastguard Worker  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18*13e8728fSAndroid Build Coastguard Worker  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19*13e8728fSAndroid Build Coastguard Worker  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20*13e8728fSAndroid Build Coastguard Worker  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21*13e8728fSAndroid Build Coastguard Worker  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22*13e8728fSAndroid Build Coastguard Worker  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23*13e8728fSAndroid Build Coastguard Worker  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24*13e8728fSAndroid Build Coastguard Worker  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25*13e8728fSAndroid Build Coastguard Worker  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26*13e8728fSAndroid Build Coastguard Worker  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27*13e8728fSAndroid Build Coastguard Worker  * SUCH DAMAGE.
28*13e8728fSAndroid Build Coastguard Worker  */
29*13e8728fSAndroid Build Coastguard Worker 
30*13e8728fSAndroid Build Coastguard Worker #include "ufdt_overlay.h"
31*13e8728fSAndroid Build Coastguard Worker #include "libufdt.h"
32*13e8728fSAndroid Build Coastguard Worker #include "ufdt_node_pool.h"
33*13e8728fSAndroid Build Coastguard Worker #include "ufdt_overlay_internal.h"
34*13e8728fSAndroid Build Coastguard Worker 
35*13e8728fSAndroid Build Coastguard Worker /*
36*13e8728fSAndroid Build Coastguard Worker  * The original version of fdt_overlay.c is slow in searching for particular
37*13e8728fSAndroid Build Coastguard Worker  * nodes and adding subnodes/properties due to the operations on flattened
38*13e8728fSAndroid Build Coastguard Worker  * device tree (FDT).
39*13e8728fSAndroid Build Coastguard Worker  *
40*13e8728fSAndroid Build Coastguard Worker  * Here we introduce `libufdt` which builds a real tree structure (named
41*13e8728fSAndroid Build Coastguard Worker  * ufdt -- unflattned device tree) from FDT. In the real tree, we can perform
42*13e8728fSAndroid Build Coastguard Worker  * certain operations (e.g., merge 2 subtrees, search for a node by path) in
43*13e8728fSAndroid Build Coastguard Worker  * almost optimal time complexity with acceptable additional memory usage.
44*13e8728fSAndroid Build Coastguard Worker  *
45*13e8728fSAndroid Build Coastguard Worker  * This file is the improved version of fdt_overlay.c by using the real tree
46*13e8728fSAndroid Build Coastguard Worker  * structure defined in libufdt.
47*13e8728fSAndroid Build Coastguard Worker  *
48*13e8728fSAndroid Build Coastguard Worker  * How the device tree overlay works and some
49*13e8728fSAndroid Build Coastguard Worker  * special terms (e.g., fixups, local fixups, fragment, etc)
50*13e8728fSAndroid Build Coastguard Worker  * are described in the document
51*13e8728fSAndroid Build Coastguard Worker  * external/dtc/Documentation/dt-object-internal.txt.
52*13e8728fSAndroid Build Coastguard Worker  */
53*13e8728fSAndroid Build Coastguard Worker 
54*13e8728fSAndroid Build Coastguard Worker /* BEGIN of operations about phandles in ufdt. */
55*13e8728fSAndroid Build Coastguard Worker 
56*13e8728fSAndroid Build Coastguard Worker /*
57*13e8728fSAndroid Build Coastguard Worker  * Increases u32 value at pos by offset.
58*13e8728fSAndroid Build Coastguard Worker  */
fdt_increase_u32(void * pos,uint32_t offset)59*13e8728fSAndroid Build Coastguard Worker static void fdt_increase_u32(void *pos, uint32_t offset) {
60*13e8728fSAndroid Build Coastguard Worker   uint32_t val;
61*13e8728fSAndroid Build Coastguard Worker 
62*13e8728fSAndroid Build Coastguard Worker   dto_memcpy(&val, pos, sizeof(val));
63*13e8728fSAndroid Build Coastguard Worker   val = cpu_to_fdt32(fdt32_to_cpu(val) + offset);
64*13e8728fSAndroid Build Coastguard Worker   dto_memcpy(pos, &val, sizeof(val));
65*13e8728fSAndroid Build Coastguard Worker }
66*13e8728fSAndroid Build Coastguard Worker 
67*13e8728fSAndroid Build Coastguard Worker /*
68*13e8728fSAndroid Build Coastguard Worker  * Gets the max phandle of a given ufdt.
69*13e8728fSAndroid Build Coastguard Worker  */
ufdt_get_max_phandle(struct ufdt * tree)70*13e8728fSAndroid Build Coastguard Worker uint32_t ufdt_get_max_phandle(struct ufdt *tree) {
71*13e8728fSAndroid Build Coastguard Worker   struct ufdt_static_phandle_table sorted_table = tree->phandle_table;
72*13e8728fSAndroid Build Coastguard Worker   if (sorted_table.len > 0)
73*13e8728fSAndroid Build Coastguard Worker     return sorted_table.data[sorted_table.len - 1].phandle;
74*13e8728fSAndroid Build Coastguard Worker   else
75*13e8728fSAndroid Build Coastguard Worker     return 0;
76*13e8728fSAndroid Build Coastguard Worker }
77*13e8728fSAndroid Build Coastguard Worker 
78*13e8728fSAndroid Build Coastguard Worker /*
79*13e8728fSAndroid Build Coastguard Worker  * Tries to increase the phandle value of a node
80*13e8728fSAndroid Build Coastguard Worker  * if the phandle exists.
81*13e8728fSAndroid Build Coastguard Worker  */
ufdt_node_try_increase_phandle(struct ufdt_node * node,uint32_t offset)82*13e8728fSAndroid Build Coastguard Worker static void ufdt_node_try_increase_phandle(struct ufdt_node *node,
83*13e8728fSAndroid Build Coastguard Worker                                            uint32_t offset) {
84*13e8728fSAndroid Build Coastguard Worker   int len = 0;
85*13e8728fSAndroid Build Coastguard Worker   char *prop_data = ufdt_node_get_fdt_prop_data_by_name(node, "phandle", &len);
86*13e8728fSAndroid Build Coastguard Worker   if (prop_data != NULL && len == sizeof(fdt32_t)) {
87*13e8728fSAndroid Build Coastguard Worker     fdt_increase_u32(prop_data, offset);
88*13e8728fSAndroid Build Coastguard Worker   }
89*13e8728fSAndroid Build Coastguard Worker   prop_data = ufdt_node_get_fdt_prop_data_by_name(node, "linux,phandle", &len);
90*13e8728fSAndroid Build Coastguard Worker   if (prop_data != NULL && len == sizeof(fdt32_t)) {
91*13e8728fSAndroid Build Coastguard Worker     fdt_increase_u32(prop_data, offset);
92*13e8728fSAndroid Build Coastguard Worker   }
93*13e8728fSAndroid Build Coastguard Worker }
94*13e8728fSAndroid Build Coastguard Worker 
95*13e8728fSAndroid Build Coastguard Worker /*
96*13e8728fSAndroid Build Coastguard Worker  * Increases all phandles by offset in a ufdt
97*13e8728fSAndroid Build Coastguard Worker  * in O(n) time.
98*13e8728fSAndroid Build Coastguard Worker  */
ufdt_try_increase_phandle(struct ufdt * tree,uint32_t offset)99*13e8728fSAndroid Build Coastguard Worker void ufdt_try_increase_phandle(struct ufdt *tree, uint32_t offset) {
100*13e8728fSAndroid Build Coastguard Worker   struct ufdt_static_phandle_table sorted_table = tree->phandle_table;
101*13e8728fSAndroid Build Coastguard Worker   int i;
102*13e8728fSAndroid Build Coastguard Worker 
103*13e8728fSAndroid Build Coastguard Worker   for (i = 0; i < sorted_table.len; i++) {
104*13e8728fSAndroid Build Coastguard Worker     struct ufdt_node *target_node = sorted_table.data[i].node;
105*13e8728fSAndroid Build Coastguard Worker 
106*13e8728fSAndroid Build Coastguard Worker     ufdt_node_try_increase_phandle(target_node, offset);
107*13e8728fSAndroid Build Coastguard Worker   }
108*13e8728fSAndroid Build Coastguard Worker }
109*13e8728fSAndroid Build Coastguard Worker 
110*13e8728fSAndroid Build Coastguard Worker /* END of operations about phandles in ufdt. */
111*13e8728fSAndroid Build Coastguard Worker 
112*13e8728fSAndroid Build Coastguard Worker /*
113*13e8728fSAndroid Build Coastguard Worker  * In the overlay_tree, there are some references (phandle)
114*13e8728fSAndroid Build Coastguard Worker  * pointing to somewhere in the main_tree.
115*13e8728fSAndroid Build Coastguard Worker  * Fix-up operations is to resolve the right address
116*13e8728fSAndroid Build Coastguard Worker  * in the overlay_tree.
117*13e8728fSAndroid Build Coastguard Worker  */
118*13e8728fSAndroid Build Coastguard Worker 
119*13e8728fSAndroid Build Coastguard Worker /* BEGIN of doing fixup in the overlay ufdt. */
120*13e8728fSAndroid Build Coastguard Worker 
121*13e8728fSAndroid Build Coastguard Worker /*
122*13e8728fSAndroid Build Coastguard Worker  * Returns exact memory location specified by fixup in format
123*13e8728fSAndroid Build Coastguard Worker  * /path/to/node:property:offset.
124*13e8728fSAndroid Build Coastguard Worker  * A property might contain multiple values and the offset is used to locate a
125*13e8728fSAndroid Build Coastguard Worker  * reference inside the property.
126*13e8728fSAndroid Build Coastguard Worker  * e.g.,
127*13e8728fSAndroid Build Coastguard Worker  * "property"=<1, 2, &ref, 4>, we can use /path/to/node:property:8 to get ref,
128*13e8728fSAndroid Build Coastguard Worker  * where 8 is sizeof(uint32) + sizeof(unit32).
129*13e8728fSAndroid Build Coastguard Worker  */
ufdt_get_fixup_location(struct ufdt * tree,const char * fixup)130*13e8728fSAndroid Build Coastguard Worker void *ufdt_get_fixup_location(struct ufdt *tree, const char *fixup) {
131*13e8728fSAndroid Build Coastguard Worker   char *path, *prop_ptr, *offset_ptr, *end_ptr;
132*13e8728fSAndroid Build Coastguard Worker   int prop_offset, prop_len;
133*13e8728fSAndroid Build Coastguard Worker   const char *prop_data;
134*13e8728fSAndroid Build Coastguard Worker   char path_buf[1024];
135*13e8728fSAndroid Build Coastguard Worker   char *path_mem = NULL;
136*13e8728fSAndroid Build Coastguard Worker 
137*13e8728fSAndroid Build Coastguard Worker   size_t fixup_len = strlen(fixup) + 1;
138*13e8728fSAndroid Build Coastguard Worker   if (fixup_len > sizeof(path_buf)) {
139*13e8728fSAndroid Build Coastguard Worker     path_mem = dto_malloc(fixup_len);
140*13e8728fSAndroid Build Coastguard Worker     path = path_mem;
141*13e8728fSAndroid Build Coastguard Worker   } else {
142*13e8728fSAndroid Build Coastguard Worker     path = path_buf;
143*13e8728fSAndroid Build Coastguard Worker   }
144*13e8728fSAndroid Build Coastguard Worker   dto_memcpy(path, fixup, fixup_len);
145*13e8728fSAndroid Build Coastguard Worker 
146*13e8728fSAndroid Build Coastguard Worker   prop_ptr = dto_strchr(path, ':');
147*13e8728fSAndroid Build Coastguard Worker   if (prop_ptr == NULL) {
148*13e8728fSAndroid Build Coastguard Worker     dto_error("Missing property part in '%s'\n", path);
149*13e8728fSAndroid Build Coastguard Worker     goto fail;
150*13e8728fSAndroid Build Coastguard Worker   }
151*13e8728fSAndroid Build Coastguard Worker 
152*13e8728fSAndroid Build Coastguard Worker   *prop_ptr = '\0';
153*13e8728fSAndroid Build Coastguard Worker   prop_ptr++;
154*13e8728fSAndroid Build Coastguard Worker 
155*13e8728fSAndroid Build Coastguard Worker   offset_ptr = dto_strchr(prop_ptr, ':');
156*13e8728fSAndroid Build Coastguard Worker   if (offset_ptr == NULL) {
157*13e8728fSAndroid Build Coastguard Worker     dto_error("Missing offset part in '%s'\n", path);
158*13e8728fSAndroid Build Coastguard Worker     goto fail;
159*13e8728fSAndroid Build Coastguard Worker   }
160*13e8728fSAndroid Build Coastguard Worker 
161*13e8728fSAndroid Build Coastguard Worker   *offset_ptr = '\0';
162*13e8728fSAndroid Build Coastguard Worker   offset_ptr++;
163*13e8728fSAndroid Build Coastguard Worker 
164*13e8728fSAndroid Build Coastguard Worker   prop_offset = dto_strtoul(offset_ptr, &end_ptr, 10 /* base */);
165*13e8728fSAndroid Build Coastguard Worker   if (*end_ptr != '\0') {
166*13e8728fSAndroid Build Coastguard Worker     dto_error("'%s' is not a valid number\n", offset_ptr);
167*13e8728fSAndroid Build Coastguard Worker     goto fail;
168*13e8728fSAndroid Build Coastguard Worker   }
169*13e8728fSAndroid Build Coastguard Worker 
170*13e8728fSAndroid Build Coastguard Worker   if (prop_offset < 0) {
171*13e8728fSAndroid Build Coastguard Worker     dto_error("'%s' is not a valid offset\n", offset_ptr);
172*13e8728fSAndroid Build Coastguard Worker     goto fail;
173*13e8728fSAndroid Build Coastguard Worker   }
174*13e8728fSAndroid Build Coastguard Worker 
175*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node *target_node;
176*13e8728fSAndroid Build Coastguard Worker   target_node = ufdt_get_node_by_path(tree, path);
177*13e8728fSAndroid Build Coastguard Worker   if (target_node == NULL) {
178*13e8728fSAndroid Build Coastguard Worker     dto_error("Path '%s' not found\n", path);
179*13e8728fSAndroid Build Coastguard Worker     goto fail;
180*13e8728fSAndroid Build Coastguard Worker   }
181*13e8728fSAndroid Build Coastguard Worker 
182*13e8728fSAndroid Build Coastguard Worker   prop_data =
183*13e8728fSAndroid Build Coastguard Worker       ufdt_node_get_fdt_prop_data_by_name(target_node, prop_ptr, &prop_len);
184*13e8728fSAndroid Build Coastguard Worker   if (prop_data == NULL) {
185*13e8728fSAndroid Build Coastguard Worker     dto_error("Property '%s' not found in  '%s' node\n", prop_ptr, path);
186*13e8728fSAndroid Build Coastguard Worker     goto fail;
187*13e8728fSAndroid Build Coastguard Worker   }
188*13e8728fSAndroid Build Coastguard Worker   /*
189*13e8728fSAndroid Build Coastguard Worker    * Note that prop_offset is the offset inside the property data.
190*13e8728fSAndroid Build Coastguard Worker    */
191*13e8728fSAndroid Build Coastguard Worker   if (prop_len < (int)sizeof(uint32_t) ||
192*13e8728fSAndroid Build Coastguard Worker       prop_offset > prop_len - (int)sizeof(uint32_t)) {
193*13e8728fSAndroid Build Coastguard Worker     dto_error("%s: property length is too small for fixup\n", path);
194*13e8728fSAndroid Build Coastguard Worker     goto fail;
195*13e8728fSAndroid Build Coastguard Worker   }
196*13e8728fSAndroid Build Coastguard Worker 
197*13e8728fSAndroid Build Coastguard Worker   if (path_mem) dto_free(path_mem);
198*13e8728fSAndroid Build Coastguard Worker   return (char *)prop_data + prop_offset;
199*13e8728fSAndroid Build Coastguard Worker 
200*13e8728fSAndroid Build Coastguard Worker fail:
201*13e8728fSAndroid Build Coastguard Worker   if (path_mem) dto_free(path_mem);
202*13e8728fSAndroid Build Coastguard Worker   return NULL;
203*13e8728fSAndroid Build Coastguard Worker }
204*13e8728fSAndroid Build Coastguard Worker 
205*13e8728fSAndroid Build Coastguard Worker /*
206*13e8728fSAndroid Build Coastguard Worker  * Process one entry in __fixups__ { } node.
207*13e8728fSAndroid Build Coastguard Worker  * @fixups is property value, array of NUL-terminated strings
208*13e8728fSAndroid Build Coastguard Worker  *   with fixup locations.
209*13e8728fSAndroid Build Coastguard Worker  * @fixups_len length of the fixups array in bytes.
210*13e8728fSAndroid Build Coastguard Worker  * @phandle is value for these locations.
211*13e8728fSAndroid Build Coastguard Worker  */
ufdt_do_one_fixup(struct ufdt * tree,const char * fixups,int fixups_len,int phandle)212*13e8728fSAndroid Build Coastguard Worker int ufdt_do_one_fixup(struct ufdt *tree, const char *fixups, int fixups_len,
213*13e8728fSAndroid Build Coastguard Worker                       int phandle) {
214*13e8728fSAndroid Build Coastguard Worker   void *fixup_pos;
215*13e8728fSAndroid Build Coastguard Worker   uint32_t val;
216*13e8728fSAndroid Build Coastguard Worker 
217*13e8728fSAndroid Build Coastguard Worker   val = cpu_to_fdt32(phandle);
218*13e8728fSAndroid Build Coastguard Worker 
219*13e8728fSAndroid Build Coastguard Worker   while (fixups_len > 0) {
220*13e8728fSAndroid Build Coastguard Worker     fixup_pos = ufdt_get_fixup_location(tree, fixups);
221*13e8728fSAndroid Build Coastguard Worker     if (fixup_pos != NULL) {
222*13e8728fSAndroid Build Coastguard Worker       dto_memcpy(fixup_pos, &val, sizeof(val));
223*13e8728fSAndroid Build Coastguard Worker     } else {
224*13e8728fSAndroid Build Coastguard Worker       return -1;
225*13e8728fSAndroid Build Coastguard Worker     }
226*13e8728fSAndroid Build Coastguard Worker 
227*13e8728fSAndroid Build Coastguard Worker     fixups_len -= dto_strlen(fixups) + 1;
228*13e8728fSAndroid Build Coastguard Worker     fixups += dto_strlen(fixups) + 1;
229*13e8728fSAndroid Build Coastguard Worker   }
230*13e8728fSAndroid Build Coastguard Worker 
231*13e8728fSAndroid Build Coastguard Worker   return 0;
232*13e8728fSAndroid Build Coastguard Worker }
233*13e8728fSAndroid Build Coastguard Worker 
234*13e8728fSAndroid Build Coastguard Worker /*
235*13e8728fSAndroid Build Coastguard Worker  * Handle __fixups__ node in overlay tree.
236*13e8728fSAndroid Build Coastguard Worker  */
237*13e8728fSAndroid Build Coastguard Worker 
ufdt_overlay_do_fixups(struct ufdt * main_tree,struct ufdt * overlay_tree)238*13e8728fSAndroid Build Coastguard Worker int ufdt_overlay_do_fixups(struct ufdt *main_tree, struct ufdt *overlay_tree) {
239*13e8728fSAndroid Build Coastguard Worker   int len = 0;
240*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node *overlay_fixups_node =
241*13e8728fSAndroid Build Coastguard Worker       ufdt_get_node_by_path(overlay_tree, "/__fixups__");
242*13e8728fSAndroid Build Coastguard Worker   if (!overlay_fixups_node) {
243*13e8728fSAndroid Build Coastguard Worker     /* There is no __fixups__. Do nothing. */
244*13e8728fSAndroid Build Coastguard Worker     return 0;
245*13e8728fSAndroid Build Coastguard Worker   }
246*13e8728fSAndroid Build Coastguard Worker 
247*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node *main_symbols_node =
248*13e8728fSAndroid Build Coastguard Worker       ufdt_get_node_by_path(main_tree, "/__symbols__");
249*13e8728fSAndroid Build Coastguard Worker 
250*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node **it;
251*13e8728fSAndroid Build Coastguard Worker   for_each_prop(it, overlay_fixups_node) {
252*13e8728fSAndroid Build Coastguard Worker     /* Find the first property */
253*13e8728fSAndroid Build Coastguard Worker 
254*13e8728fSAndroid Build Coastguard Worker     /* Check __symbols__ is exist when we have any property in __fixups__ */
255*13e8728fSAndroid Build Coastguard Worker     if (!main_symbols_node) {
256*13e8728fSAndroid Build Coastguard Worker       dto_error("No node __symbols__ in main dtb.\n");
257*13e8728fSAndroid Build Coastguard Worker       return -1;
258*13e8728fSAndroid Build Coastguard Worker     }
259*13e8728fSAndroid Build Coastguard Worker     break;
260*13e8728fSAndroid Build Coastguard Worker   }
261*13e8728fSAndroid Build Coastguard Worker 
262*13e8728fSAndroid Build Coastguard Worker   for_each_prop(it, overlay_fixups_node) {
263*13e8728fSAndroid Build Coastguard Worker     /*
264*13e8728fSAndroid Build Coastguard Worker      * A property in __fixups__ looks like:
265*13e8728fSAndroid Build Coastguard Worker      * symbol_name =
266*13e8728fSAndroid Build Coastguard Worker      * "/path/to/node:prop:offset0\x00/path/to/node:prop:offset1..."
267*13e8728fSAndroid Build Coastguard Worker      * So we firstly find the node "symbol_name" and obtain its phandle in
268*13e8728fSAndroid Build Coastguard Worker      * __symbols__ of the main_tree.
269*13e8728fSAndroid Build Coastguard Worker      */
270*13e8728fSAndroid Build Coastguard Worker 
271*13e8728fSAndroid Build Coastguard Worker     struct ufdt_node *fixups = *it;
272*13e8728fSAndroid Build Coastguard Worker     char *symbol_path = ufdt_node_get_fdt_prop_data_by_name(
273*13e8728fSAndroid Build Coastguard Worker         main_symbols_node, ufdt_node_name(fixups), &len);
274*13e8728fSAndroid Build Coastguard Worker 
275*13e8728fSAndroid Build Coastguard Worker     if (!symbol_path) {
276*13e8728fSAndroid Build Coastguard Worker       dto_error("Couldn't find '%s' symbol in main dtb\n",
277*13e8728fSAndroid Build Coastguard Worker                 ufdt_node_name(fixups));
278*13e8728fSAndroid Build Coastguard Worker       return -1;
279*13e8728fSAndroid Build Coastguard Worker     }
280*13e8728fSAndroid Build Coastguard Worker 
281*13e8728fSAndroid Build Coastguard Worker     struct ufdt_node *symbol_node;
282*13e8728fSAndroid Build Coastguard Worker     symbol_node = ufdt_get_node_by_path(main_tree, symbol_path);
283*13e8728fSAndroid Build Coastguard Worker 
284*13e8728fSAndroid Build Coastguard Worker     if (!symbol_node) {
285*13e8728fSAndroid Build Coastguard Worker       dto_error("Couldn't find '%s' path in main dtb\n", symbol_path);
286*13e8728fSAndroid Build Coastguard Worker       return -1;
287*13e8728fSAndroid Build Coastguard Worker     }
288*13e8728fSAndroid Build Coastguard Worker 
289*13e8728fSAndroid Build Coastguard Worker     uint32_t phandle = ufdt_node_get_phandle(symbol_node);
290*13e8728fSAndroid Build Coastguard Worker 
291*13e8728fSAndroid Build Coastguard Worker     const char *fixups_paths = ufdt_node_get_fdt_prop_data(fixups, &len);
292*13e8728fSAndroid Build Coastguard Worker 
293*13e8728fSAndroid Build Coastguard Worker     if (ufdt_do_one_fixup(overlay_tree, fixups_paths, len, phandle) < 0) {
294*13e8728fSAndroid Build Coastguard Worker       dto_error("Failed one fixup in ufdt_do_one_fixup\n");
295*13e8728fSAndroid Build Coastguard Worker       return -1;
296*13e8728fSAndroid Build Coastguard Worker     }
297*13e8728fSAndroid Build Coastguard Worker   }
298*13e8728fSAndroid Build Coastguard Worker 
299*13e8728fSAndroid Build Coastguard Worker   return 0;
300*13e8728fSAndroid Build Coastguard Worker }
301*13e8728fSAndroid Build Coastguard Worker 
302*13e8728fSAndroid Build Coastguard Worker /* END of doing fixup in the overlay ufdt. */
303*13e8728fSAndroid Build Coastguard Worker 
304*13e8728fSAndroid Build Coastguard Worker /*
305*13e8728fSAndroid Build Coastguard Worker  * Here is to overlay all fragments in the overlay_tree to the main_tree.
306*13e8728fSAndroid Build Coastguard Worker  * What is "overlay fragment"? The main purpose is to add some subtrees to the
307*13e8728fSAndroid Build Coastguard Worker  * main_tree in order to complete the entire device tree.
308*13e8728fSAndroid Build Coastguard Worker  *
309*13e8728fSAndroid Build Coastguard Worker  * A fragment consists of two parts: 1. the subtree to be added 2. where it
310*13e8728fSAndroid Build Coastguard Worker  * should be added.
311*13e8728fSAndroid Build Coastguard Worker  *
312*13e8728fSAndroid Build Coastguard Worker  * Overlaying a fragment requires: 1. find the node in the main_tree 2. merge
313*13e8728fSAndroid Build Coastguard Worker  * the subtree into that node in the main_tree.
314*13e8728fSAndroid Build Coastguard Worker  */
315*13e8728fSAndroid Build Coastguard Worker 
316*13e8728fSAndroid Build Coastguard Worker /* BEGIN of applying fragments. */
317*13e8728fSAndroid Build Coastguard Worker 
318*13e8728fSAndroid Build Coastguard Worker /*
319*13e8728fSAndroid Build Coastguard Worker  * Overlay the overlay_node over target_node.
320*13e8728fSAndroid Build Coastguard Worker  */
ufdt_overlay_node(struct ufdt_node * target_node,struct ufdt_node * overlay_node,struct ufdt_node_pool * pool)321*13e8728fSAndroid Build Coastguard Worker static int ufdt_overlay_node(struct ufdt_node *target_node,
322*13e8728fSAndroid Build Coastguard Worker                              struct ufdt_node *overlay_node,
323*13e8728fSAndroid Build Coastguard Worker                              struct ufdt_node_pool *pool) {
324*13e8728fSAndroid Build Coastguard Worker   return ufdt_node_merge_into(target_node, overlay_node, pool);
325*13e8728fSAndroid Build Coastguard Worker }
326*13e8728fSAndroid Build Coastguard Worker 
ufdt_overlay_get_target(struct ufdt * tree,struct ufdt_node * frag_node,struct ufdt_node ** target_node)327*13e8728fSAndroid Build Coastguard Worker enum overlay_result ufdt_overlay_get_target(struct ufdt *tree,
328*13e8728fSAndroid Build Coastguard Worker                                             struct ufdt_node *frag_node,
329*13e8728fSAndroid Build Coastguard Worker                                             struct ufdt_node **target_node) {
330*13e8728fSAndroid Build Coastguard Worker   uint32_t target;
331*13e8728fSAndroid Build Coastguard Worker   const char *target_path;
332*13e8728fSAndroid Build Coastguard Worker   const void *val;
333*13e8728fSAndroid Build Coastguard Worker   *target_node = NULL;
334*13e8728fSAndroid Build Coastguard Worker 
335*13e8728fSAndroid Build Coastguard Worker   val = ufdt_node_get_fdt_prop_data_by_name(frag_node, "target", NULL);
336*13e8728fSAndroid Build Coastguard Worker   if (val) {
337*13e8728fSAndroid Build Coastguard Worker     dto_memcpy(&target, val, sizeof(target));
338*13e8728fSAndroid Build Coastguard Worker     target = fdt32_to_cpu(target);
339*13e8728fSAndroid Build Coastguard Worker     *target_node = ufdt_get_node_by_phandle(tree, target);
340*13e8728fSAndroid Build Coastguard Worker     if (*target_node == NULL) {
341*13e8728fSAndroid Build Coastguard Worker       dto_error("failed to find target %04x\n", target);
342*13e8728fSAndroid Build Coastguard Worker       return OVERLAY_RESULT_TARGET_INVALID;
343*13e8728fSAndroid Build Coastguard Worker     }
344*13e8728fSAndroid Build Coastguard Worker   }
345*13e8728fSAndroid Build Coastguard Worker 
346*13e8728fSAndroid Build Coastguard Worker   if (*target_node == NULL) {
347*13e8728fSAndroid Build Coastguard Worker     target_path =
348*13e8728fSAndroid Build Coastguard Worker         ufdt_node_get_fdt_prop_data_by_name(frag_node, "target-path", NULL);
349*13e8728fSAndroid Build Coastguard Worker     if (target_path == NULL) {
350*13e8728fSAndroid Build Coastguard Worker       return OVERLAY_RESULT_MISSING_TARGET;
351*13e8728fSAndroid Build Coastguard Worker     }
352*13e8728fSAndroid Build Coastguard Worker 
353*13e8728fSAndroid Build Coastguard Worker     *target_node = ufdt_get_node_by_path(tree, target_path);
354*13e8728fSAndroid Build Coastguard Worker     if (*target_node == NULL) {
355*13e8728fSAndroid Build Coastguard Worker       dto_error("failed to find target-path %s\n", target_path);
356*13e8728fSAndroid Build Coastguard Worker       return OVERLAY_RESULT_TARGET_PATH_INVALID;
357*13e8728fSAndroid Build Coastguard Worker     }
358*13e8728fSAndroid Build Coastguard Worker   }
359*13e8728fSAndroid Build Coastguard Worker 
360*13e8728fSAndroid Build Coastguard Worker   return OVERLAY_RESULT_OK;
361*13e8728fSAndroid Build Coastguard Worker }
362*13e8728fSAndroid Build Coastguard Worker 
363*13e8728fSAndroid Build Coastguard Worker /*
364*13e8728fSAndroid Build Coastguard Worker  * Apply one overlay fragment (subtree).
365*13e8728fSAndroid Build Coastguard Worker  */
ufdt_apply_fragment(struct ufdt * tree,struct ufdt_node * frag_node,struct ufdt_node_pool * pool)366*13e8728fSAndroid Build Coastguard Worker static enum overlay_result ufdt_apply_fragment(struct ufdt *tree,
367*13e8728fSAndroid Build Coastguard Worker                                                struct ufdt_node *frag_node,
368*13e8728fSAndroid Build Coastguard Worker                                                struct ufdt_node_pool *pool) {
369*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node *target_node = NULL;
370*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node *overlay_node = NULL;
371*13e8728fSAndroid Build Coastguard Worker 
372*13e8728fSAndroid Build Coastguard Worker   overlay_node = ufdt_node_get_node_by_path(frag_node, "__overlay__");
373*13e8728fSAndroid Build Coastguard Worker   if (overlay_node == NULL) {
374*13e8728fSAndroid Build Coastguard Worker     return OVERLAY_RESULT_MISSING_OVERLAY;
375*13e8728fSAndroid Build Coastguard Worker   }
376*13e8728fSAndroid Build Coastguard Worker 
377*13e8728fSAndroid Build Coastguard Worker   enum overlay_result result =
378*13e8728fSAndroid Build Coastguard Worker       ufdt_overlay_get_target(tree, frag_node, &target_node);
379*13e8728fSAndroid Build Coastguard Worker   if (target_node == NULL) {
380*13e8728fSAndroid Build Coastguard Worker     dto_error("Unable to resolve target for %s\n", ufdt_node_name(frag_node));
381*13e8728fSAndroid Build Coastguard Worker     return result;
382*13e8728fSAndroid Build Coastguard Worker   }
383*13e8728fSAndroid Build Coastguard Worker 
384*13e8728fSAndroid Build Coastguard Worker   int err = ufdt_overlay_node(target_node, overlay_node, pool);
385*13e8728fSAndroid Build Coastguard Worker 
386*13e8728fSAndroid Build Coastguard Worker   if (err < 0) {
387*13e8728fSAndroid Build Coastguard Worker     dto_error("failed to overlay node %s to target %s\n",
388*13e8728fSAndroid Build Coastguard Worker               ufdt_node_name(overlay_node), ufdt_node_name(target_node));
389*13e8728fSAndroid Build Coastguard Worker     return OVERLAY_RESULT_MERGE_FAIL;
390*13e8728fSAndroid Build Coastguard Worker   }
391*13e8728fSAndroid Build Coastguard Worker 
392*13e8728fSAndroid Build Coastguard Worker   return OVERLAY_RESULT_OK;
393*13e8728fSAndroid Build Coastguard Worker }
394*13e8728fSAndroid Build Coastguard Worker 
395*13e8728fSAndroid Build Coastguard Worker /*
396*13e8728fSAndroid Build Coastguard Worker  * Applies all fragments to the main_tree.
397*13e8728fSAndroid Build Coastguard Worker  */
ufdt_overlay_apply_fragments(struct ufdt * main_tree,struct ufdt * overlay_tree,struct ufdt_node_pool * pool)398*13e8728fSAndroid Build Coastguard Worker static int ufdt_overlay_apply_fragments(struct ufdt *main_tree,
399*13e8728fSAndroid Build Coastguard Worker                                         struct ufdt *overlay_tree,
400*13e8728fSAndroid Build Coastguard Worker                                         struct ufdt_node_pool *pool) {
401*13e8728fSAndroid Build Coastguard Worker   enum overlay_result ret;
402*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node **it;
403*13e8728fSAndroid Build Coastguard Worker   /*
404*13e8728fSAndroid Build Coastguard Worker    * This loop may iterate to subnodes that's not a fragment node.
405*13e8728fSAndroid Build Coastguard Worker    * We must fail for any other error.
406*13e8728fSAndroid Build Coastguard Worker    */
407*13e8728fSAndroid Build Coastguard Worker   for_each_node(it, overlay_tree->root) {
408*13e8728fSAndroid Build Coastguard Worker     ret = ufdt_apply_fragment(main_tree, *it, pool);
409*13e8728fSAndroid Build Coastguard Worker     if ((ret != OVERLAY_RESULT_OK) && (ret != OVERLAY_RESULT_MISSING_OVERLAY)) {
410*13e8728fSAndroid Build Coastguard Worker       dto_error("failed to apply overlay fragment %s ret: %d\n",
411*13e8728fSAndroid Build Coastguard Worker                 ufdt_node_name(*it), ret);
412*13e8728fSAndroid Build Coastguard Worker       return -1;
413*13e8728fSAndroid Build Coastguard Worker     }
414*13e8728fSAndroid Build Coastguard Worker   }
415*13e8728fSAndroid Build Coastguard Worker   return 0;
416*13e8728fSAndroid Build Coastguard Worker }
417*13e8728fSAndroid Build Coastguard Worker 
418*13e8728fSAndroid Build Coastguard Worker /* END of applying fragments. */
419*13e8728fSAndroid Build Coastguard Worker 
420*13e8728fSAndroid Build Coastguard Worker /*
421*13e8728fSAndroid Build Coastguard Worker  * Since the overlay_tree will be "merged" into the main_tree, some
422*13e8728fSAndroid Build Coastguard Worker  * references (e.g., phandle values that acts as an unique ID) need to be
423*13e8728fSAndroid Build Coastguard Worker  * updated so it won't lead to collision that different nodes have the same
424*13e8728fSAndroid Build Coastguard Worker  * phandle value.
425*13e8728fSAndroid Build Coastguard Worker  *
426*13e8728fSAndroid Build Coastguard Worker  * Two things need to be done:
427*13e8728fSAndroid Build Coastguard Worker  *
428*13e8728fSAndroid Build Coastguard Worker  * 1. ufdt_try_increase_phandle()
429*13e8728fSAndroid Build Coastguard Worker  * Update phandle (an unique integer ID of a node in the device tree) of each
430*13e8728fSAndroid Build Coastguard Worker  * node in the overlay_tree. To achieve this, we simply increase each phandle
431*13e8728fSAndroid Build Coastguard Worker  * values in the overlay_tree by the max phandle value of the main_tree.
432*13e8728fSAndroid Build Coastguard Worker  *
433*13e8728fSAndroid Build Coastguard Worker  * 2. ufdt_overlay_do_local_fixups()
434*13e8728fSAndroid Build Coastguard Worker  * If there are some reference in the overlay_tree that references nodes
435*13e8728fSAndroid Build Coastguard Worker  * inside the overlay_tree, we have to modify the reference value (address of
436*13e8728fSAndroid Build Coastguard Worker  * the referenced node: phandle) so that it corresponds to the right node inside
437*13e8728fSAndroid Build Coastguard Worker  * the overlay_tree. Where the reference exists is kept in __local_fixups__ node
438*13e8728fSAndroid Build Coastguard Worker  * in the overlay_tree.
439*13e8728fSAndroid Build Coastguard Worker  */
440*13e8728fSAndroid Build Coastguard Worker 
441*13e8728fSAndroid Build Coastguard Worker /* BEGIN of updating local references (phandle values) in the overlay ufdt. */
442*13e8728fSAndroid Build Coastguard Worker 
443*13e8728fSAndroid Build Coastguard Worker /*
444*13e8728fSAndroid Build Coastguard Worker  * local fixups
445*13e8728fSAndroid Build Coastguard Worker  */
ufdt_local_fixup_prop(struct ufdt_node * target_prop_node,struct ufdt_node * local_fixup_prop_node,uint32_t phandle_offset)446*13e8728fSAndroid Build Coastguard Worker static int ufdt_local_fixup_prop(struct ufdt_node *target_prop_node,
447*13e8728fSAndroid Build Coastguard Worker                                  struct ufdt_node *local_fixup_prop_node,
448*13e8728fSAndroid Build Coastguard Worker                                  uint32_t phandle_offset) {
449*13e8728fSAndroid Build Coastguard Worker   /*
450*13e8728fSAndroid Build Coastguard Worker    * prop_offsets_ptr should be a list of fdt32_t.
451*13e8728fSAndroid Build Coastguard Worker    * <offset0 offset1 offset2 ...>
452*13e8728fSAndroid Build Coastguard Worker    */
453*13e8728fSAndroid Build Coastguard Worker   char *prop_offsets_ptr;
454*13e8728fSAndroid Build Coastguard Worker   int len = 0;
455*13e8728fSAndroid Build Coastguard Worker   prop_offsets_ptr = ufdt_node_get_fdt_prop_data(local_fixup_prop_node, &len);
456*13e8728fSAndroid Build Coastguard Worker 
457*13e8728fSAndroid Build Coastguard Worker   if (prop_offsets_ptr == NULL || len % sizeof(fdt32_t) != 0) return -1;
458*13e8728fSAndroid Build Coastguard Worker 
459*13e8728fSAndroid Build Coastguard Worker   char *prop_data;
460*13e8728fSAndroid Build Coastguard Worker   int target_length = 0;
461*13e8728fSAndroid Build Coastguard Worker 
462*13e8728fSAndroid Build Coastguard Worker   prop_data = ufdt_node_get_fdt_prop_data(target_prop_node, &target_length);
463*13e8728fSAndroid Build Coastguard Worker 
464*13e8728fSAndroid Build Coastguard Worker   if (prop_data == NULL) return -1;
465*13e8728fSAndroid Build Coastguard Worker 
466*13e8728fSAndroid Build Coastguard Worker   int i;
467*13e8728fSAndroid Build Coastguard Worker   for (i = 0; i < len; i += sizeof(fdt32_t)) {
468*13e8728fSAndroid Build Coastguard Worker     int offset = fdt32_to_cpu(*(fdt32_t *)(prop_offsets_ptr + i));
469*13e8728fSAndroid Build Coastguard Worker     if (offset + sizeof(fdt32_t) > (size_t)target_length) return -1;
470*13e8728fSAndroid Build Coastguard Worker     fdt_increase_u32((prop_data + offset), phandle_offset);
471*13e8728fSAndroid Build Coastguard Worker   }
472*13e8728fSAndroid Build Coastguard Worker   return 0;
473*13e8728fSAndroid Build Coastguard Worker }
474*13e8728fSAndroid Build Coastguard Worker 
ufdt_local_fixup_node(struct ufdt_node * target_node,struct ufdt_node * local_fixups_node,uint32_t phandle_offset)475*13e8728fSAndroid Build Coastguard Worker static int ufdt_local_fixup_node(struct ufdt_node *target_node,
476*13e8728fSAndroid Build Coastguard Worker                                  struct ufdt_node *local_fixups_node,
477*13e8728fSAndroid Build Coastguard Worker                                  uint32_t phandle_offset) {
478*13e8728fSAndroid Build Coastguard Worker   if (local_fixups_node == NULL) return 0;
479*13e8728fSAndroid Build Coastguard Worker 
480*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node **it_local_fixups;
481*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node *sub_target_node;
482*13e8728fSAndroid Build Coastguard Worker 
483*13e8728fSAndroid Build Coastguard Worker   for_each_prop(it_local_fixups, local_fixups_node) {
484*13e8728fSAndroid Build Coastguard Worker     sub_target_node = ufdt_node_get_property_by_name(
485*13e8728fSAndroid Build Coastguard Worker         target_node, ufdt_node_name(*it_local_fixups));
486*13e8728fSAndroid Build Coastguard Worker 
487*13e8728fSAndroid Build Coastguard Worker     if (sub_target_node != NULL) {
488*13e8728fSAndroid Build Coastguard Worker       int err = ufdt_local_fixup_prop(sub_target_node, *it_local_fixups,
489*13e8728fSAndroid Build Coastguard Worker                                       phandle_offset);
490*13e8728fSAndroid Build Coastguard Worker       if (err < 0) return -1;
491*13e8728fSAndroid Build Coastguard Worker     } else {
492*13e8728fSAndroid Build Coastguard Worker       return -1;
493*13e8728fSAndroid Build Coastguard Worker     }
494*13e8728fSAndroid Build Coastguard Worker   }
495*13e8728fSAndroid Build Coastguard Worker 
496*13e8728fSAndroid Build Coastguard Worker   for_each_node(it_local_fixups, local_fixups_node) {
497*13e8728fSAndroid Build Coastguard Worker     sub_target_node = ufdt_node_get_node_by_path(
498*13e8728fSAndroid Build Coastguard Worker         target_node, ufdt_node_name(*it_local_fixups));
499*13e8728fSAndroid Build Coastguard Worker     if (sub_target_node != NULL) {
500*13e8728fSAndroid Build Coastguard Worker       int err = ufdt_local_fixup_node(sub_target_node, *it_local_fixups,
501*13e8728fSAndroid Build Coastguard Worker                                       phandle_offset);
502*13e8728fSAndroid Build Coastguard Worker       if (err < 0) return -1;
503*13e8728fSAndroid Build Coastguard Worker     } else {
504*13e8728fSAndroid Build Coastguard Worker       return -1;
505*13e8728fSAndroid Build Coastguard Worker     }
506*13e8728fSAndroid Build Coastguard Worker   }
507*13e8728fSAndroid Build Coastguard Worker 
508*13e8728fSAndroid Build Coastguard Worker   return 0;
509*13e8728fSAndroid Build Coastguard Worker }
510*13e8728fSAndroid Build Coastguard Worker 
511*13e8728fSAndroid Build Coastguard Worker /*
512*13e8728fSAndroid Build Coastguard Worker  * Handle __local_fixups__ node in overlay DTB
513*13e8728fSAndroid Build Coastguard Worker  * The __local_fixups__ format we expect is
514*13e8728fSAndroid Build Coastguard Worker  * __local_fixups__ {
515*13e8728fSAndroid Build Coastguard Worker  *   path {
516*13e8728fSAndroid Build Coastguard Worker  *    to {
517*13e8728fSAndroid Build Coastguard Worker  *      local_ref1 = <offset>;
518*13e8728fSAndroid Build Coastguard Worker  *    };
519*13e8728fSAndroid Build Coastguard Worker  *   };
520*13e8728fSAndroid Build Coastguard Worker  *   path2 {
521*13e8728fSAndroid Build Coastguard Worker  *    to2 {
522*13e8728fSAndroid Build Coastguard Worker  *      local_ref2 = <offset1 offset2 ...>;
523*13e8728fSAndroid Build Coastguard Worker  *    };
524*13e8728fSAndroid Build Coastguard Worker  *   };
525*13e8728fSAndroid Build Coastguard Worker  * };
526*13e8728fSAndroid Build Coastguard Worker  *
527*13e8728fSAndroid Build Coastguard Worker  * which follows the dtc patch from:
528*13e8728fSAndroid Build Coastguard Worker  * https://marc.info/?l=devicetree&m=144061468601974&w=4
529*13e8728fSAndroid Build Coastguard Worker  */
ufdt_overlay_do_local_fixups(struct ufdt * tree,uint32_t phandle_offset)530*13e8728fSAndroid Build Coastguard Worker int ufdt_overlay_do_local_fixups(struct ufdt *tree, uint32_t phandle_offset) {
531*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node *overlay_node = ufdt_get_node_by_path(tree, "/");
532*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node *local_fixups_node =
533*13e8728fSAndroid Build Coastguard Worker       ufdt_get_node_by_path(tree, "/__local_fixups__");
534*13e8728fSAndroid Build Coastguard Worker 
535*13e8728fSAndroid Build Coastguard Worker   int err =
536*13e8728fSAndroid Build Coastguard Worker       ufdt_local_fixup_node(overlay_node, local_fixups_node, phandle_offset);
537*13e8728fSAndroid Build Coastguard Worker 
538*13e8728fSAndroid Build Coastguard Worker   if (err < 0) return -1;
539*13e8728fSAndroid Build Coastguard Worker 
540*13e8728fSAndroid Build Coastguard Worker   return 0;
541*13e8728fSAndroid Build Coastguard Worker }
542*13e8728fSAndroid Build Coastguard Worker 
ufdt_overlay_local_ref_update(struct ufdt * main_tree,struct ufdt * overlay_tree)543*13e8728fSAndroid Build Coastguard Worker static int ufdt_overlay_local_ref_update(struct ufdt *main_tree,
544*13e8728fSAndroid Build Coastguard Worker                                          struct ufdt *overlay_tree) {
545*13e8728fSAndroid Build Coastguard Worker   uint32_t phandle_offset = 0;
546*13e8728fSAndroid Build Coastguard Worker 
547*13e8728fSAndroid Build Coastguard Worker   phandle_offset = ufdt_get_max_phandle(main_tree);
548*13e8728fSAndroid Build Coastguard Worker   if (phandle_offset > 0) {
549*13e8728fSAndroid Build Coastguard Worker     ufdt_try_increase_phandle(overlay_tree, phandle_offset);
550*13e8728fSAndroid Build Coastguard Worker   }
551*13e8728fSAndroid Build Coastguard Worker 
552*13e8728fSAndroid Build Coastguard Worker   int err = ufdt_overlay_do_local_fixups(overlay_tree, phandle_offset);
553*13e8728fSAndroid Build Coastguard Worker   if (err < 0) {
554*13e8728fSAndroid Build Coastguard Worker     dto_error("failed to perform local fixups in overlay\n");
555*13e8728fSAndroid Build Coastguard Worker     return -1;
556*13e8728fSAndroid Build Coastguard Worker   }
557*13e8728fSAndroid Build Coastguard Worker   return 0;
558*13e8728fSAndroid Build Coastguard Worker }
559*13e8728fSAndroid Build Coastguard Worker 
560*13e8728fSAndroid Build Coastguard Worker /* END of updating local references (phandle values) in the overlay ufdt. */
561*13e8728fSAndroid Build Coastguard Worker 
_ufdt_overlay_fdtps(struct ufdt * main_tree,const struct ufdt * overlay_tree)562*13e8728fSAndroid Build Coastguard Worker static int _ufdt_overlay_fdtps(struct ufdt *main_tree,
563*13e8728fSAndroid Build Coastguard Worker                                const struct ufdt *overlay_tree) {
564*13e8728fSAndroid Build Coastguard Worker   for (int i = 0; i < overlay_tree->num_used_fdtps; i++) {
565*13e8728fSAndroid Build Coastguard Worker     void *fdt = overlay_tree->fdtps[i];
566*13e8728fSAndroid Build Coastguard Worker     if (ufdt_add_fdt(main_tree, fdt) < 0) {
567*13e8728fSAndroid Build Coastguard Worker       return -1;
568*13e8728fSAndroid Build Coastguard Worker     }
569*13e8728fSAndroid Build Coastguard Worker   }
570*13e8728fSAndroid Build Coastguard Worker   return 0;
571*13e8728fSAndroid Build Coastguard Worker }
572*13e8728fSAndroid Build Coastguard Worker 
ufdt_overlay_apply(struct ufdt * main_tree,struct ufdt * overlay_tree,size_t overlay_length,struct ufdt_node_pool * pool)573*13e8728fSAndroid Build Coastguard Worker static int ufdt_overlay_apply(struct ufdt *main_tree, struct ufdt *overlay_tree,
574*13e8728fSAndroid Build Coastguard Worker                               size_t overlay_length,
575*13e8728fSAndroid Build Coastguard Worker                               struct ufdt_node_pool *pool) {
576*13e8728fSAndroid Build Coastguard Worker   if (_ufdt_overlay_fdtps(main_tree, overlay_tree) < 0) {
577*13e8728fSAndroid Build Coastguard Worker     dto_error("failed to add more fdt into main ufdt tree.\n");
578*13e8728fSAndroid Build Coastguard Worker     return -1;
579*13e8728fSAndroid Build Coastguard Worker   }
580*13e8728fSAndroid Build Coastguard Worker 
581*13e8728fSAndroid Build Coastguard Worker   if (overlay_length < sizeof(struct fdt_header)) {
582*13e8728fSAndroid Build Coastguard Worker     dto_error("Overlay_length %zu smaller than header size %zu\n",
583*13e8728fSAndroid Build Coastguard Worker               overlay_length, sizeof(struct fdt_header));
584*13e8728fSAndroid Build Coastguard Worker     return -1;
585*13e8728fSAndroid Build Coastguard Worker   }
586*13e8728fSAndroid Build Coastguard Worker 
587*13e8728fSAndroid Build Coastguard Worker   if (ufdt_overlay_local_ref_update(main_tree, overlay_tree) < 0) {
588*13e8728fSAndroid Build Coastguard Worker     dto_error("failed to perform local fixups in overlay\n");
589*13e8728fSAndroid Build Coastguard Worker     return -1;
590*13e8728fSAndroid Build Coastguard Worker   }
591*13e8728fSAndroid Build Coastguard Worker 
592*13e8728fSAndroid Build Coastguard Worker   if (ufdt_overlay_do_fixups(main_tree, overlay_tree) < 0) {
593*13e8728fSAndroid Build Coastguard Worker     dto_error("failed to perform fixups in overlay\n");
594*13e8728fSAndroid Build Coastguard Worker     return -1;
595*13e8728fSAndroid Build Coastguard Worker   }
596*13e8728fSAndroid Build Coastguard Worker   if (ufdt_overlay_apply_fragments(main_tree, overlay_tree, pool) < 0) {
597*13e8728fSAndroid Build Coastguard Worker     dto_error("failed to apply fragments\n");
598*13e8728fSAndroid Build Coastguard Worker     return -1;
599*13e8728fSAndroid Build Coastguard Worker   }
600*13e8728fSAndroid Build Coastguard Worker 
601*13e8728fSAndroid Build Coastguard Worker   return 0;
602*13e8728fSAndroid Build Coastguard Worker }
603*13e8728fSAndroid Build Coastguard Worker 
ufdt_install_blob(void * blob,size_t blob_size)604*13e8728fSAndroid Build Coastguard Worker struct fdt_header *ufdt_install_blob(void *blob, size_t blob_size) {
605*13e8728fSAndroid Build Coastguard Worker   struct fdt_header *pHeader;
606*13e8728fSAndroid Build Coastguard Worker   int err;
607*13e8728fSAndroid Build Coastguard Worker 
608*13e8728fSAndroid Build Coastguard Worker   dto_debug("ufdt_install_blob (0x%08jx)\n", (uintmax_t)blob);
609*13e8728fSAndroid Build Coastguard Worker 
610*13e8728fSAndroid Build Coastguard Worker   if (blob_size < sizeof(struct fdt_header)) {
611*13e8728fSAndroid Build Coastguard Worker     dto_error("Blob_size %zu smaller than the header size %zu\n", blob_size,
612*13e8728fSAndroid Build Coastguard Worker               sizeof(struct fdt_header));
613*13e8728fSAndroid Build Coastguard Worker     return NULL;
614*13e8728fSAndroid Build Coastguard Worker   }
615*13e8728fSAndroid Build Coastguard Worker 
616*13e8728fSAndroid Build Coastguard Worker   pHeader = (struct fdt_header *)blob;
617*13e8728fSAndroid Build Coastguard Worker   err = fdt_check_header(pHeader);
618*13e8728fSAndroid Build Coastguard Worker   if (err < 0) {
619*13e8728fSAndroid Build Coastguard Worker     if (err == -FDT_ERR_BADVERSION) {
620*13e8728fSAndroid Build Coastguard Worker       dto_error("incompatible blob version: %d, should be: %d\n",
621*13e8728fSAndroid Build Coastguard Worker                 fdt_version(pHeader), FDT_LAST_SUPPORTED_VERSION);
622*13e8728fSAndroid Build Coastguard Worker 
623*13e8728fSAndroid Build Coastguard Worker     } else {
624*13e8728fSAndroid Build Coastguard Worker       dto_error("error validating blob: %s\n", fdt_strerror(err));
625*13e8728fSAndroid Build Coastguard Worker     }
626*13e8728fSAndroid Build Coastguard Worker     return NULL;
627*13e8728fSAndroid Build Coastguard Worker   }
628*13e8728fSAndroid Build Coastguard Worker 
629*13e8728fSAndroid Build Coastguard Worker   return pHeader;
630*13e8728fSAndroid Build Coastguard Worker }
631*13e8728fSAndroid Build Coastguard Worker 
632*13e8728fSAndroid Build Coastguard Worker /*
633*13e8728fSAndroid Build Coastguard Worker * From Google, based on dt_overlay_apply() logic
634*13e8728fSAndroid Build Coastguard Worker * Will dto_malloc a new fdt blob and return it. Will not dto_free parameters.
635*13e8728fSAndroid Build Coastguard Worker */
ufdt_apply_overlay(struct fdt_header * main_fdt_header,size_t main_fdt_size,void * overlay_fdtp,size_t overlay_size)636*13e8728fSAndroid Build Coastguard Worker struct fdt_header *ufdt_apply_overlay(struct fdt_header *main_fdt_header,
637*13e8728fSAndroid Build Coastguard Worker                                  size_t main_fdt_size,
638*13e8728fSAndroid Build Coastguard Worker                                  void *overlay_fdtp,
639*13e8728fSAndroid Build Coastguard Worker                                  size_t overlay_size) {
640*13e8728fSAndroid Build Coastguard Worker   size_t out_fdt_size;
641*13e8728fSAndroid Build Coastguard Worker 
642*13e8728fSAndroid Build Coastguard Worker   if (main_fdt_header == NULL) {
643*13e8728fSAndroid Build Coastguard Worker     return NULL;
644*13e8728fSAndroid Build Coastguard Worker   }
645*13e8728fSAndroid Build Coastguard Worker 
646*13e8728fSAndroid Build Coastguard Worker   if (overlay_size < 8 || overlay_size != fdt_totalsize(overlay_fdtp)) {
647*13e8728fSAndroid Build Coastguard Worker     dto_error("Bad overlay size!\n");
648*13e8728fSAndroid Build Coastguard Worker     return NULL;
649*13e8728fSAndroid Build Coastguard Worker   }
650*13e8728fSAndroid Build Coastguard Worker   if (main_fdt_size < 8 || main_fdt_size != fdt_totalsize(main_fdt_header)) {
651*13e8728fSAndroid Build Coastguard Worker     dto_error("Bad fdt size!\n");
652*13e8728fSAndroid Build Coastguard Worker     return NULL;
653*13e8728fSAndroid Build Coastguard Worker   }
654*13e8728fSAndroid Build Coastguard Worker 
655*13e8728fSAndroid Build Coastguard Worker   out_fdt_size = fdt_totalsize(main_fdt_header) + overlay_size;
656*13e8728fSAndroid Build Coastguard Worker   /* It's actually more than enough */
657*13e8728fSAndroid Build Coastguard Worker   struct fdt_header *out_fdt_header = dto_malloc(out_fdt_size);
658*13e8728fSAndroid Build Coastguard Worker 
659*13e8728fSAndroid Build Coastguard Worker   if (out_fdt_header == NULL) {
660*13e8728fSAndroid Build Coastguard Worker     dto_error("failed to allocate memory for DTB blob with overlays\n");
661*13e8728fSAndroid Build Coastguard Worker     return NULL;
662*13e8728fSAndroid Build Coastguard Worker   }
663*13e8728fSAndroid Build Coastguard Worker 
664*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node_pool pool;
665*13e8728fSAndroid Build Coastguard Worker   ufdt_node_pool_construct(&pool);
666*13e8728fSAndroid Build Coastguard Worker   struct ufdt *main_tree = ufdt_from_fdt(main_fdt_header, main_fdt_size, &pool);
667*13e8728fSAndroid Build Coastguard Worker   struct ufdt *overlay_tree = ufdt_from_fdt(overlay_fdtp, overlay_size, &pool);
668*13e8728fSAndroid Build Coastguard Worker   int err = ufdt_overlay_apply(main_tree, overlay_tree, overlay_size, &pool);
669*13e8728fSAndroid Build Coastguard Worker   if (err < 0) {
670*13e8728fSAndroid Build Coastguard Worker     goto fail;
671*13e8728fSAndroid Build Coastguard Worker   }
672*13e8728fSAndroid Build Coastguard Worker 
673*13e8728fSAndroid Build Coastguard Worker   err = ufdt_to_fdt(main_tree, out_fdt_header, out_fdt_size);
674*13e8728fSAndroid Build Coastguard Worker   if (err < 0) {
675*13e8728fSAndroid Build Coastguard Worker     dto_error("Failed to dump the device tree to out_fdt_header\n");
676*13e8728fSAndroid Build Coastguard Worker     goto fail;
677*13e8728fSAndroid Build Coastguard Worker   }
678*13e8728fSAndroid Build Coastguard Worker 
679*13e8728fSAndroid Build Coastguard Worker   ufdt_destruct(overlay_tree, &pool);
680*13e8728fSAndroid Build Coastguard Worker   ufdt_destruct(main_tree, &pool);
681*13e8728fSAndroid Build Coastguard Worker   ufdt_node_pool_destruct(&pool);
682*13e8728fSAndroid Build Coastguard Worker 
683*13e8728fSAndroid Build Coastguard Worker   return out_fdt_header;
684*13e8728fSAndroid Build Coastguard Worker 
685*13e8728fSAndroid Build Coastguard Worker fail:
686*13e8728fSAndroid Build Coastguard Worker   ufdt_destruct(overlay_tree, &pool);
687*13e8728fSAndroid Build Coastguard Worker   ufdt_destruct(main_tree, &pool);
688*13e8728fSAndroid Build Coastguard Worker   ufdt_node_pool_destruct(&pool);
689*13e8728fSAndroid Build Coastguard Worker   dto_free(out_fdt_header);
690*13e8728fSAndroid Build Coastguard Worker 
691*13e8728fSAndroid Build Coastguard Worker   return NULL;
692*13e8728fSAndroid Build Coastguard Worker }
693*13e8728fSAndroid Build Coastguard Worker 
694*13e8728fSAndroid Build Coastguard Worker /*
695*13e8728fSAndroid Build Coastguard Worker  * Apply device tree `overlays` to `main_fdt_header` fdt buffer. (API is unstable)
696*13e8728fSAndroid Build Coastguard Worker  *
697*13e8728fSAndroid Build Coastguard Worker  * `main_fdt_header` is getting overrided by result tree, so it must
698*13e8728fSAndroid Build Coastguard Worker  * have enough space (provided by `main_fdt_buffer_size`) to store it.
699*13e8728fSAndroid Build Coastguard Worker  * `main_fdt_header` and all `overlays` must be 8 bytes aligned.
700*13e8728fSAndroid Build Coastguard Worker  *
701*13e8728fSAndroid Build Coastguard Worker  * `dto_malloc` is used for:
702*13e8728fSAndroid Build Coastguard Worker  * - ufdt structures around main fdt and overlays.
703*13e8728fSAndroid Build Coastguard Worker  * - result tree temporary buffer at most `main_fdt_buffer_size` size.
704*13e8728fSAndroid Build Coastguard Worker  *
705*13e8728fSAndroid Build Coastguard Worker  * TODO(b/362830550): expose a more comprehensive error type.
706*13e8728fSAndroid Build Coastguard Worker  * Returns 0 or -1 in case of error.
707*13e8728fSAndroid Build Coastguard Worker  */
ufdt_apply_multioverlay(struct fdt_header * main_fdt_header,size_t main_fdt_buffer_size,void * const * overlays,size_t overlays_count)708*13e8728fSAndroid Build Coastguard Worker int ufdt_apply_multioverlay(struct fdt_header *main_fdt_header,
709*13e8728fSAndroid Build Coastguard Worker                             size_t main_fdt_buffer_size, void *const *overlays,
710*13e8728fSAndroid Build Coastguard Worker                             size_t overlays_count) {
711*13e8728fSAndroid Build Coastguard Worker   void *temporary_buffer = NULL;
712*13e8728fSAndroid Build Coastguard Worker 
713*13e8728fSAndroid Build Coastguard Worker   if (main_fdt_header == NULL || fdt_check_header(main_fdt_header) != 0 ||
714*13e8728fSAndroid Build Coastguard Worker       overlays == NULL) {
715*13e8728fSAndroid Build Coastguard Worker     return -1;
716*13e8728fSAndroid Build Coastguard Worker   }
717*13e8728fSAndroid Build Coastguard Worker   if (overlays_count == 0) {
718*13e8728fSAndroid Build Coastguard Worker     return 0;
719*13e8728fSAndroid Build Coastguard Worker   }
720*13e8728fSAndroid Build Coastguard Worker 
721*13e8728fSAndroid Build Coastguard Worker   size_t result_size = fdt_totalsize(main_fdt_header);
722*13e8728fSAndroid Build Coastguard Worker   struct ufdt_node_pool pool;
723*13e8728fSAndroid Build Coastguard Worker   ufdt_node_pool_construct(&pool);
724*13e8728fSAndroid Build Coastguard Worker   struct ufdt *main_tree = ufdt_from_fdt(main_fdt_header, result_size, &pool);
725*13e8728fSAndroid Build Coastguard Worker 
726*13e8728fSAndroid Build Coastguard Worker   for (size_t i = 0; i < overlays_count; i++) {
727*13e8728fSAndroid Build Coastguard Worker     struct fdt_header *current_overlay = overlays[i];
728*13e8728fSAndroid Build Coastguard Worker     if (fdt_check_header(current_overlay) != 0) {
729*13e8728fSAndroid Build Coastguard Worker       dto_error("Failed to parse %zuth overlay header\n", i);
730*13e8728fSAndroid Build Coastguard Worker       goto error;
731*13e8728fSAndroid Build Coastguard Worker     }
732*13e8728fSAndroid Build Coastguard Worker 
733*13e8728fSAndroid Build Coastguard Worker     size_t overlay_size = fdt_totalsize(current_overlay);
734*13e8728fSAndroid Build Coastguard Worker     result_size += overlay_size;
735*13e8728fSAndroid Build Coastguard Worker 
736*13e8728fSAndroid Build Coastguard Worker     // prepare main tree by rebuilding phandle table. don't need to do so
737*13e8728fSAndroid Build Coastguard Worker     // for the first iteration since main_tree hasn't been updated yet
738*13e8728fSAndroid Build Coastguard Worker     if (i != 0) {
739*13e8728fSAndroid Build Coastguard Worker       main_tree->phandle_table = build_phandle_table(main_tree);
740*13e8728fSAndroid Build Coastguard Worker     }
741*13e8728fSAndroid Build Coastguard Worker 
742*13e8728fSAndroid Build Coastguard Worker     struct ufdt *overlay_tree =
743*13e8728fSAndroid Build Coastguard Worker         ufdt_from_fdt(current_overlay, overlay_size, &pool);
744*13e8728fSAndroid Build Coastguard Worker     int err = ufdt_overlay_apply(main_tree, overlay_tree, overlay_size, &pool);
745*13e8728fSAndroid Build Coastguard Worker     ufdt_destruct(overlay_tree, &pool);
746*13e8728fSAndroid Build Coastguard Worker     if (err < 0) {
747*13e8728fSAndroid Build Coastguard Worker       dto_error("Failed to apply overlay number: %d\n", i);
748*13e8728fSAndroid Build Coastguard Worker       goto error;
749*13e8728fSAndroid Build Coastguard Worker     }
750*13e8728fSAndroid Build Coastguard Worker   }
751*13e8728fSAndroid Build Coastguard Worker 
752*13e8728fSAndroid Build Coastguard Worker   if (result_size > main_fdt_buffer_size) {
753*13e8728fSAndroid Build Coastguard Worker     dto_error(
754*13e8728fSAndroid Build Coastguard Worker         "Not enough space in main_fdt to apply the overlays. Required %d, "
755*13e8728fSAndroid Build Coastguard Worker         "available: %d\n",
756*13e8728fSAndroid Build Coastguard Worker         result_size, main_fdt_buffer_size);
757*13e8728fSAndroid Build Coastguard Worker     goto error;
758*13e8728fSAndroid Build Coastguard Worker   }
759*13e8728fSAndroid Build Coastguard Worker 
760*13e8728fSAndroid Build Coastguard Worker   // ufdt tree has references to fdt buffer, so we cannot dump ufdt to
761*13e8728fSAndroid Build Coastguard Worker   // underlying fdt buffer directly. allocate intermediate buffer for that.
762*13e8728fSAndroid Build Coastguard Worker   temporary_buffer = dto_malloc(result_size);
763*13e8728fSAndroid Build Coastguard Worker   if (temporary_buffer == NULL) {
764*13e8728fSAndroid Build Coastguard Worker     dto_error("Failed to allocate memory for temporary buffer: %d\n",
765*13e8728fSAndroid Build Coastguard Worker               result_size);
766*13e8728fSAndroid Build Coastguard Worker     goto error;
767*13e8728fSAndroid Build Coastguard Worker   }
768*13e8728fSAndroid Build Coastguard Worker 
769*13e8728fSAndroid Build Coastguard Worker   int err = ufdt_to_fdt(main_tree, temporary_buffer, result_size);
770*13e8728fSAndroid Build Coastguard Worker   if (err < 0) {
771*13e8728fSAndroid Build Coastguard Worker     dto_error(
772*13e8728fSAndroid Build Coastguard Worker         "Failed to dump the result device tree to the temporary buffer\n");
773*13e8728fSAndroid Build Coastguard Worker     goto error;
774*13e8728fSAndroid Build Coastguard Worker   }
775*13e8728fSAndroid Build Coastguard Worker   ufdt_destruct(main_tree, &pool);
776*13e8728fSAndroid Build Coastguard Worker 
777*13e8728fSAndroid Build Coastguard Worker   dto_memcpy(main_fdt_header, temporary_buffer, result_size);
778*13e8728fSAndroid Build Coastguard Worker   dto_free(temporary_buffer);
779*13e8728fSAndroid Build Coastguard Worker   ufdt_node_pool_destruct(&pool);
780*13e8728fSAndroid Build Coastguard Worker 
781*13e8728fSAndroid Build Coastguard Worker   return 0;
782*13e8728fSAndroid Build Coastguard Worker 
783*13e8728fSAndroid Build Coastguard Worker error:
784*13e8728fSAndroid Build Coastguard Worker   dto_free(temporary_buffer);
785*13e8728fSAndroid Build Coastguard Worker   ufdt_destruct(main_tree, &pool);
786*13e8728fSAndroid Build Coastguard Worker   ufdt_node_pool_destruct(&pool);
787*13e8728fSAndroid Build Coastguard Worker 
788*13e8728fSAndroid Build Coastguard Worker   return -1;
789*13e8728fSAndroid Build Coastguard Worker }
790