]> git.karo-electronics.de Git - karo-tx-uboot.git/blob - lib/libfdt/fdt_region.c
arm64: mvebu: Add basic support for the Marvell Armada 7K/8K SoC
[karo-tx-uboot.git] / lib / libfdt / fdt_region.c
1 /*
2  * libfdt - Flat Device Tree manipulation
3  * Copyright (C) 2013 Google, Inc
4  * Written by Simon Glass <sjg@chromium.org>
5  * SPDX-License-Identifier:     GPL-2.0+ BSD-2-Clause
6  */
7
8 #include <libfdt_env.h>
9
10 #ifndef USE_HOSTCC
11 #include <fdt.h>
12 #include <libfdt.h>
13 #else
14 #include "fdt_host.h"
15 #endif
16
17 #include "libfdt_internal.h"
18
19 /**
20  * fdt_add_region() - Add a new region to our list
21  * @info:       State information
22  * @offset:     Start offset of region
23  * @size:       Size of region
24  *
25  * The region is added if there is space, but in any case we increment the
26  * count. If permitted, and the new region overlaps the last one, we merge
27  * them.
28  */
29 static int fdt_add_region(struct fdt_region_state *info, int offset, int size)
30 {
31         struct fdt_region *reg;
32
33         reg = info->region ? &info->region[info->count - 1] : NULL;
34         if (info->can_merge && info->count &&
35             info->count <= info->max_regions &&
36             reg && offset <= reg->offset + reg->size) {
37                 reg->size = offset + size - reg->offset;
38         } else if (info->count++ < info->max_regions) {
39                 if (reg) {
40                         reg++;
41                         reg->offset = offset;
42                         reg->size = size;
43                 }
44         } else {
45                 return -1;
46         }
47
48         return 0;
49 }
50
51 static int region_list_contains_offset(struct fdt_region_state *info,
52                                        const void *fdt, int target)
53 {
54         struct fdt_region *reg;
55         int num;
56
57         target += fdt_off_dt_struct(fdt);
58         for (reg = info->region, num = 0; num < info->count; reg++, num++) {
59                 if (target >= reg->offset && target < reg->offset + reg->size)
60                         return 1;
61         }
62
63         return 0;
64 }
65
66 int fdt_add_alias_regions(const void *fdt, struct fdt_region *region, int count,
67                           int max_regions, struct fdt_region_state *info)
68 {
69         int base = fdt_off_dt_struct(fdt);
70         int node, node_end, offset;
71         int did_alias_header;
72
73         node = fdt_subnode_offset(fdt, 0, "aliases");
74         if (node < 0)
75                 return -FDT_ERR_NOTFOUND;
76
77         /* The aliases node must come before the others */
78         node_end = fdt_next_subnode(fdt, node);
79         if (node_end <= 0)
80                 return -FDT_ERR_BADLAYOUT;
81         node_end -= sizeof(fdt32_t);
82
83         did_alias_header = 0;
84         info->region = region;
85         info->count = count;
86         info->can_merge = 0;
87         info->max_regions = max_regions;
88
89         for (offset = fdt_first_property_offset(fdt, node);
90              offset >= 0;
91              offset = fdt_next_property_offset(fdt, offset)) {
92                 const struct fdt_property *prop;
93                 const char *name;
94                 int target, next;
95
96                 prop = fdt_get_property_by_offset(fdt, offset, NULL);
97                 name = fdt_string(fdt, fdt32_to_cpu(prop->nameoff));
98                 target = fdt_path_offset(fdt, name);
99                 if (!region_list_contains_offset(info, fdt, target))
100                         continue;
101                 next = fdt_next_property_offset(fdt, offset);
102                 if (next < 0)
103                         next = node_end;
104
105                 if (!did_alias_header) {
106                         fdt_add_region(info, base + node, 12);
107                         did_alias_header = 1;
108                 }
109                 fdt_add_region(info, base + offset, next - offset);
110         }
111
112         /* Add the 'end' tag */
113         if (did_alias_header)
114                 fdt_add_region(info, base + node_end, sizeof(fdt32_t));
115
116         return info->count < max_regions ? info->count : -FDT_ERR_NOSPACE;
117 }
118
119 /**
120  * fdt_include_supernodes() - Include supernodes required by this node
121  * @info:       State information
122  * @depth:      Current stack depth
123  *
124  * When we decided to include a node or property which is not at the top
125  * level, this function forces the inclusion of higher level nodes. For
126  * example, given this tree:
127  *
128  * / {
129  *     testing {
130  *     }
131  * }
132  *
133  * If we decide to include testing then we need the root node to have a valid
134  * tree. This function adds those regions.
135  */
136 static int fdt_include_supernodes(struct fdt_region_state *info, int depth)
137 {
138         int base = fdt_off_dt_struct(info->fdt);
139         int start, stop_at;
140         int i;
141
142         /*
143          * Work down the stack looking for supernodes that we didn't include.
144          * The algortihm here is actually pretty simple, since we know that
145          * no previous subnode had to include these nodes, or if it did, we
146          * marked them as included (on the stack) already.
147          */
148         for (i = 0; i <= depth; i++) {
149                 if (!info->stack[i].included) {
150                         start = info->stack[i].offset;
151
152                         /* Add the FDT_BEGIN_NODE tag of this supernode */
153                         fdt_next_tag(info->fdt, start, &stop_at);
154                         if (fdt_add_region(info, base + start, stop_at - start))
155                                 return -1;
156
157                         /* Remember that this supernode is now included */
158                         info->stack[i].included = 1;
159                         info->can_merge = 1;
160                 }
161
162                 /* Force (later) generation of the FDT_END_NODE tag */
163                 if (!info->stack[i].want)
164                         info->stack[i].want = WANT_NODES_ONLY;
165         }
166
167         return 0;
168 }
169
170 enum {
171         FDT_DONE_NOTHING,
172         FDT_DONE_MEM_RSVMAP,
173         FDT_DONE_STRUCT,
174         FDT_DONE_END,
175         FDT_DONE_STRINGS,
176         FDT_DONE_ALL,
177 };
178
179 int fdt_first_region(const void *fdt,
180                 int (*h_include)(void *priv, const void *fdt, int offset,
181                                  int type, const char *data, int size),
182                 void *priv, struct fdt_region *region,
183                 char *path, int path_len, int flags,
184                 struct fdt_region_state *info)
185 {
186         struct fdt_region_ptrs *p = &info->ptrs;
187
188         /* Set up our state */
189         info->fdt = fdt;
190         info->can_merge = 1;
191         info->max_regions = 1;
192         info->start = -1;
193         p->want = WANT_NOTHING;
194         p->end = path;
195         *p->end = '\0';
196         p->nextoffset = 0;
197         p->depth = -1;
198         p->done = FDT_DONE_NOTHING;
199
200         return fdt_next_region(fdt, h_include, priv, region,
201                                path, path_len, flags, info);
202 }
203
204 /***********************************************************************
205  *
206  *      Theory of operation
207  *
208  * Note: in this description 'included' means that a node (or other part
209  * of the tree) should be included in the region list, i.e. it will have
210  * a region which covers its part of the tree.
211  *
212  * This function maintains some state from the last time it is called.
213  * It checks the next part of the tree that it is supposed to look at
214  * (p.nextoffset) to see if that should be included or not. When it
215  * finds something to include, it sets info->start to its offset. This
216  * marks the start of the region we want to include.
217  *
218  * Once info->start is set to the start (i.e. not -1), we continue
219  * scanning until we find something that we don't want included. This
220  * will be the end of a region. At this point we can close off the
221  * region and add it to the list. So we do so, and reset info->start
222  * to -1.
223  *
224  * One complication here is that we want to merge regions. So when we
225  * come to add another region later, we may in fact merge it with the
226  * previous one if one ends where the other starts.
227  *
228  * The function fdt_add_region() will return -1 if it fails to add the
229  * region, because we already have a region ready to be returned, and
230  * the new one cannot be merged in with it. In this case, we must return
231  * the region we found, and wait for another call to this function.
232  * When it comes, we will repeat the processing of the tag and again
233  * try to add a region. This time it will succeed.
234  *
235  * The current state of the pointers (stack, offset, etc.) is maintained
236  * in a ptrs member. At the start of every loop iteration we make a copy
237  * of it.  The copy is then updated as the tag is processed. Only if we
238  * get to the end of the loop iteration (and successfully call
239  * fdt_add_region() if we need to) can we commit the changes we have
240  * made to these pointers. For example, if we see an FDT_END_NODE tag,
241  * we will decrement the depth value. But if we need to add a region
242  * for this tag (let's say because the previous tag is included and this
243  * FDT_END_NODE tag is not included) then we will only commit the result
244  * if we were able to add the region. That allows us to retry again next
245  * time.
246  *
247  * We keep track of a variable called 'want' which tells us what we want
248  * to include when there is no specific information provided by the
249  * h_include function for a particular property. This basically handles
250  * the inclusion of properties which are pulled in by virtue of the node
251  * they are in. So if you include a node, its properties are also
252  * included.  In this case 'want' will be WANT_NODES_AND_PROPS. The
253  * FDT_REG_DIRECT_SUBNODES feature also makes use of 'want'. While we
254  * are inside the subnode, 'want' will be set to WANT_NODES_ONLY, so
255  * that only the subnode's FDT_BEGIN_NODE and FDT_END_NODE tags will be
256  * included, and properties will be skipped. If WANT_NOTHING is
257  * selected, then we will just rely on what the h_include() function
258  * tells us.
259  *
260  * Using 'want' we work out 'include', which tells us whether this
261  * current tag should be included or not. As you can imagine, if the
262  * value of 'include' changes, that means we are on a boundary between
263  * nodes to include and nodes to exclude. At this point we either close
264  * off a previous region and add it to the list, or mark the start of a
265  * new region.
266  *
267  * Apart from the nodes, we have mem_rsvmap, the FDT_END tag and the
268  * string list. Each of these dealt with as a whole (i.e. we create a
269  * region for each if it is to be included). For mem_rsvmap we don't
270  * allow it to merge with the first struct region. For the stringlist,
271  * we don't allow it to merge with the last struct region (which
272  * contains at minimum the FDT_END tag).
273  *
274  *********************************************************************/
275
276 int fdt_next_region(const void *fdt,
277                 int (*h_include)(void *priv, const void *fdt, int offset,
278                                  int type, const char *data, int size),
279                 void *priv, struct fdt_region *region,
280                 char *path, int path_len, int flags,
281                 struct fdt_region_state *info)
282 {
283         int base = fdt_off_dt_struct(fdt);
284         int last_node = 0;
285         const char *str;
286
287         info->region = region;
288         info->count = 0;
289         if (info->ptrs.done < FDT_DONE_MEM_RSVMAP &&
290             (flags & FDT_REG_ADD_MEM_RSVMAP)) {
291                 /* Add the memory reserve map into its own region */
292                 if (fdt_add_region(info, fdt_off_mem_rsvmap(fdt),
293                                    fdt_off_dt_struct(fdt) -
294                                    fdt_off_mem_rsvmap(fdt)))
295                         return 0;
296                 info->can_merge = 0;    /* Don't allow merging with this */
297                 info->ptrs.done = FDT_DONE_MEM_RSVMAP;
298         }
299
300         /*
301          * Work through the tags one by one, deciding whether each needs to
302          * be included or not. We set the variable 'include' to indicate our
303          * decision. 'want' is used to track what we want to include - it
304          * allows us to pick up all the properties (and/or subnode tags) of
305          * a node.
306          */
307         while (info->ptrs.done < FDT_DONE_STRUCT) {
308                 const struct fdt_property *prop;
309                 struct fdt_region_ptrs p;
310                 const char *name;
311                 int include = 0;
312                 int stop_at = 0;
313                 uint32_t tag;
314                 int offset;
315                 int val;
316                 int len;
317
318                 /*
319                  * Make a copy of our pointers. If we make it to the end of
320                  * this block then we will commit them back to info->ptrs.
321                  * Otherwise we can try again from the same starting state
322                  * next time we are called.
323                  */
324                 p = info->ptrs;
325
326                 /*
327                  * Find the tag, and the offset of the next one. If we need to
328                  * stop including tags, then by default we stop *after*
329                  * including the current tag
330                  */
331                 offset = p.nextoffset;
332                 tag = fdt_next_tag(fdt, offset, &p.nextoffset);
333                 stop_at = p.nextoffset;
334
335                 switch (tag) {
336                 case FDT_PROP:
337                         stop_at = offset;
338                         prop = fdt_get_property_by_offset(fdt, offset, NULL);
339                         str = fdt_string(fdt, fdt32_to_cpu(prop->nameoff));
340                         val = h_include(priv, fdt, last_node, FDT_IS_PROP, str,
341                                             strlen(str) + 1);
342                         if (val == -1) {
343                                 include = p.want >= WANT_NODES_AND_PROPS;
344                         } else {
345                                 include = val;
346                                 /*
347                                  * Make sure we include the } for this block.
348                                  * It might be more correct to have this done
349                                  * by the call to fdt_include_supernodes() in
350                                  * the case where it adds the node we are
351                                  * currently in, but this is equivalent.
352                                  */
353                                 if ((flags & FDT_REG_SUPERNODES) && val &&
354                                     !p.want)
355                                         p.want = WANT_NODES_ONLY;
356                         }
357
358                         /* Value grepping is not yet supported */
359                         break;
360
361                 case FDT_NOP:
362                         include = p.want >= WANT_NODES_AND_PROPS;
363                         stop_at = offset;
364                         break;
365
366                 case FDT_BEGIN_NODE:
367                         last_node = offset;
368                         p.depth++;
369                         if (p.depth == FDT_MAX_DEPTH)
370                                 return -FDT_ERR_TOODEEP;
371                         name = fdt_get_name(fdt, offset, &len);
372                         if (p.end - path + 2 + len >= path_len)
373                                 return -FDT_ERR_NOSPACE;
374
375                         /* Build the full path of this node */
376                         if (p.end != path + 1)
377                                 *p.end++ = '/';
378                         strcpy(p.end, name);
379                         p.end += len;
380                         info->stack[p.depth].want = p.want;
381                         info->stack[p.depth].offset = offset;
382
383                         /*
384                          * If we are not intending to include this node unless
385                          * it matches, make sure we stop *before* its tag.
386                          */
387                         if (p.want == WANT_NODES_ONLY ||
388                             !(flags & (FDT_REG_DIRECT_SUBNODES |
389                                        FDT_REG_ALL_SUBNODES))) {
390                                 stop_at = offset;
391                                 p.want = WANT_NOTHING;
392                         }
393                         val = h_include(priv, fdt, offset, FDT_IS_NODE, path,
394                                         p.end - path + 1);
395
396                         /* Include this if requested */
397                         if (val) {
398                                 p.want = (flags & FDT_REG_ALL_SUBNODES) ?
399                                         WANT_ALL_NODES_AND_PROPS :
400                                         WANT_NODES_AND_PROPS;
401                         }
402
403                         /* If not requested, decay our 'p.want' value */
404                         else if (p.want) {
405                                 if (p.want != WANT_ALL_NODES_AND_PROPS)
406                                         p.want--;
407
408                         /* Not including this tag, so stop now */
409                         } else {
410                                 stop_at = offset;
411                         }
412
413                         /*
414                          * Decide whether to include this tag, and update our
415                          * stack with the state for this node
416                          */
417                         include = p.want;
418                         info->stack[p.depth].included = include;
419                         break;
420
421                 case FDT_END_NODE:
422                         include = p.want;
423                         if (p.depth < 0)
424                                 return -FDT_ERR_BADSTRUCTURE;
425
426                         /*
427                          * If we don't want this node, stop right away, unless
428                          * we are including subnodes
429                          */
430                         if (!p.want && !(flags & FDT_REG_DIRECT_SUBNODES))
431                                 stop_at = offset;
432                         p.want = info->stack[p.depth].want;
433                         p.depth--;
434                         while (p.end > path && *--p.end != '/')
435                                 ;
436                         *p.end = '\0';
437                         break;
438
439                 case FDT_END:
440                         /* We always include the end tag */
441                         include = 1;
442                         p.done = FDT_DONE_STRUCT;
443                         break;
444                 }
445
446                 /* If this tag is to be included, mark it as region start */
447                 if (include && info->start == -1) {
448                         /* Include any supernodes required by this one */
449                         if (flags & FDT_REG_SUPERNODES) {
450                                 if (fdt_include_supernodes(info, p.depth))
451                                         return 0;
452                         }
453                         info->start = offset;
454                 }
455
456                 /*
457                  * If this tag is not to be included, finish up the current
458                  * region.
459                  */
460                 if (!include && info->start != -1) {
461                         if (fdt_add_region(info, base + info->start,
462                                            stop_at - info->start))
463                                 return 0;
464                         info->start = -1;
465                         info->can_merge = 1;
466                 }
467
468                 /* If we have made it this far, we can commit our pointers */
469                 info->ptrs = p;
470         }
471
472         /* Add a region for the END tag and a separate one for string table */
473         if (info->ptrs.done < FDT_DONE_END) {
474                 if (info->ptrs.nextoffset != fdt_size_dt_struct(fdt))
475                         return -FDT_ERR_BADSTRUCTURE;
476
477                 if (fdt_add_region(info, base + info->start,
478                                    info->ptrs.nextoffset - info->start))
479                         return 0;
480                 info->ptrs.done++;
481         }
482         if (info->ptrs.done < FDT_DONE_STRINGS) {
483                 if (flags & FDT_REG_ADD_STRING_TAB) {
484                         info->can_merge = 0;
485                         if (fdt_off_dt_strings(fdt) <
486                             base + info->ptrs.nextoffset)
487                                 return -FDT_ERR_BADLAYOUT;
488                         if (fdt_add_region(info, fdt_off_dt_strings(fdt),
489                                            fdt_size_dt_strings(fdt)))
490                                 return 0;
491                 }
492                 info->ptrs.done++;
493         }
494
495         return info->count > 0 ? 0 : -FDT_ERR_NOTFOUND;
496 }