]> git.proxmox.com Git - mirror_edk2.git/blame - EmbeddedPkg/Library/FdtLib/fdt.c
EmbeddedPkg/FdtLib: Update FdtLib to v1.4.5
[mirror_edk2.git] / EmbeddedPkg / Library / FdtLib / fdt.c
CommitLineData
1e57a462 1/*\r
2 * libfdt - Flat Device Tree manipulation\r
3 * Copyright (C) 2006 David Gibson, IBM Corporation.\r
4 *\r
5 * libfdt is dual licensed: you can use it either under the terms of\r
6 * the GPL, or the BSD license, at your option.\r
7 *\r
8 * a) This library is free software; you can redistribute it and/or\r
9 * modify it under the terms of the GNU General Public License as\r
10 * published by the Free Software Foundation; either version 2 of the\r
11 * License, or (at your option) any later version.\r
12 *\r
13 * This library is distributed in the hope that it will be useful,\r
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of\r
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\r
16 * GNU General Public License for more details.\r
17 *\r
18 * You should have received a copy of the GNU General Public\r
19 * License along with this library; if not, write to the Free\r
20 * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston,\r
21 * MA 02110-1301 USA\r
22 *\r
23 * Alternatively,\r
24 *\r
25 * b) Redistribution and use in source and binary forms, with or\r
26 * without modification, are permitted provided that the following\r
27 * conditions are met:\r
28 *\r
29 * 1. Redistributions of source code must retain the above\r
30 * copyright notice, this list of conditions and the following\r
31 * disclaimer.\r
32 * 2. Redistributions in binary form must reproduce the above\r
33 * copyright notice, this list of conditions and the following\r
34 * disclaimer in the documentation and/or other materials\r
35 * provided with the distribution.\r
36 *\r
37 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND\r
38 * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,\r
39 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF\r
40 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE\r
41 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR\r
42 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,\r
43 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT\r
44 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;\r
45 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)\r
46 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN\r
47 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR\r
48 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,\r
49 * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\r
50 */\r
51#include "libfdt_env.h"\r
52\r
53#include <fdt.h>\r
54#include <libfdt.h>\r
55\r
56#include "libfdt_internal.h"\r
57\r
58int fdt_check_header(const void *fdt)\r
59{\r
60 if (fdt_magic(fdt) == FDT_MAGIC) {\r
61 /* Complete tree */\r
62 if (fdt_version(fdt) < FDT_FIRST_SUPPORTED_VERSION)\r
63 return -FDT_ERR_BADVERSION;\r
64 if (fdt_last_comp_version(fdt) > FDT_LAST_SUPPORTED_VERSION)\r
65 return -FDT_ERR_BADVERSION;\r
66 } else if (fdt_magic(fdt) == FDT_SW_MAGIC) {\r
67 /* Unfinished sequential-write blob */\r
68 if (fdt_size_dt_struct(fdt) == 0)\r
69 return -FDT_ERR_BADSTATE;\r
70 } else {\r
71 return -FDT_ERR_BADMAGIC;\r
72 }\r
73\r
74 return 0;\r
75}\r
76\r
77const void *fdt_offset_ptr(const void *fdt, int offset, unsigned int len)\r
78{\r
a0992390
PB
79 unsigned absoffset = offset + fdt_off_dt_struct(fdt);\r
80\r
81 if ((absoffset < offset)\r
82 || ((absoffset + len) < absoffset)\r
83 || (absoffset + len) > fdt_totalsize(fdt))\r
84 return NULL;\r
1e57a462 85\r
86 if (fdt_version(fdt) >= 0x11)\r
87 if (((offset + len) < offset)\r
88 || ((offset + len) > fdt_size_dt_struct(fdt)))\r
89 return NULL;\r
90\r
a0992390 91 return _fdt_offset_ptr(fdt, offset);\r
1e57a462 92}\r
93\r
94uint32_t fdt_next_tag(const void *fdt, int startoffset, int *nextoffset)\r
95{\r
3e8576dd 96 const fdt32_t *tagp, *lenp;\r
1e57a462 97 uint32_t tag;\r
98 int offset = startoffset;\r
99 const char *p;\r
100\r
101 *nextoffset = -FDT_ERR_TRUNCATED;\r
102 tagp = fdt_offset_ptr(fdt, offset, FDT_TAGSIZE);\r
103 if (!tagp)\r
104 return FDT_END; /* premature end */\r
105 tag = fdt32_to_cpu(*tagp);\r
106 offset += FDT_TAGSIZE;\r
107\r
108 *nextoffset = -FDT_ERR_BADSTRUCTURE;\r
109 switch (tag) {\r
110 case FDT_BEGIN_NODE:\r
111 /* skip name */\r
112 do {\r
113 p = fdt_offset_ptr(fdt, offset++, 1);\r
114 } while (p && (*p != '\0'));\r
115 if (!p)\r
116 return FDT_END; /* premature end */\r
117 break;\r
118\r
119 case FDT_PROP:\r
120 lenp = fdt_offset_ptr(fdt, offset, sizeof(*lenp));\r
121 if (!lenp)\r
122 return FDT_END; /* premature end */\r
123 /* skip-name offset, length and value */\r
124 offset += sizeof(struct fdt_property) - FDT_TAGSIZE\r
125 + fdt32_to_cpu(*lenp);\r
126 break;\r
127\r
128 case FDT_END:\r
129 case FDT_END_NODE:\r
130 case FDT_NOP:\r
131 break;\r
132\r
133 default:\r
134 return FDT_END;\r
135 }\r
136\r
137 if (!fdt_offset_ptr(fdt, startoffset, offset - startoffset))\r
138 return FDT_END; /* premature end */\r
139\r
140 *nextoffset = FDT_TAGALIGN(offset);\r
141 return tag;\r
142}\r
143\r
144int _fdt_check_node_offset(const void *fdt, int offset)\r
145{\r
146 if ((offset < 0) || (offset % FDT_TAGSIZE)\r
147 || (fdt_next_tag(fdt, offset, &offset) != FDT_BEGIN_NODE))\r
148 return -FDT_ERR_BADOFFSET;\r
149\r
150 return offset;\r
151}\r
152\r
153int _fdt_check_prop_offset(const void *fdt, int offset)\r
154{\r
155 if ((offset < 0) || (offset % FDT_TAGSIZE)\r
156 || (fdt_next_tag(fdt, offset, &offset) != FDT_PROP))\r
157 return -FDT_ERR_BADOFFSET;\r
158\r
159 return offset;\r
160}\r
161\r
162int fdt_next_node(const void *fdt, int offset, int *depth)\r
163{\r
164 int nextoffset = 0;\r
165 uint32_t tag;\r
166\r
167 if (offset >= 0)\r
168 if ((nextoffset = _fdt_check_node_offset(fdt, offset)) < 0)\r
169 return nextoffset;\r
170\r
171 do {\r
172 offset = nextoffset;\r
173 tag = fdt_next_tag(fdt, offset, &nextoffset);\r
174\r
175 switch (tag) {\r
176 case FDT_PROP:\r
177 case FDT_NOP:\r
178 break;\r
179\r
180 case FDT_BEGIN_NODE:\r
181 if (depth)\r
182 (*depth)++;\r
183 break;\r
184\r
185 case FDT_END_NODE:\r
186 if (depth && ((--(*depth)) < 0))\r
187 return nextoffset;\r
188 break;\r
189\r
190 case FDT_END:\r
191 if ((nextoffset >= 0)\r
192 || ((nextoffset == -FDT_ERR_TRUNCATED) && !depth))\r
193 return -FDT_ERR_NOTFOUND;\r
194 else\r
195 return nextoffset;\r
196 }\r
197 } while (tag != FDT_BEGIN_NODE);\r
198\r
199 return offset;\r
200}\r
201\r
3e8576dd
OM
202int fdt_first_subnode(const void *fdt, int offset)\r
203{\r
204 int depth = 0;\r
205\r
206 offset = fdt_next_node(fdt, offset, &depth);\r
207 if (offset < 0 || depth != 1)\r
208 return -FDT_ERR_NOTFOUND;\r
209\r
210 return offset;\r
211}\r
212\r
213int fdt_next_subnode(const void *fdt, int offset)\r
214{\r
215 int depth = 1;\r
216\r
217 /*\r
218 * With respect to the parent, the depth of the next subnode will be\r
219 * the same as the last.\r
220 */\r
221 do {\r
222 offset = fdt_next_node(fdt, offset, &depth);\r
223 if (offset < 0 || depth < 1)\r
224 return -FDT_ERR_NOTFOUND;\r
225 } while (depth > 1);\r
226\r
227 return offset;\r
228}\r
229\r
1e57a462 230const char *_fdt_find_string(const char *strtab, int tabsize, const char *s)\r
231{\r
232 int len = strlen(s) + 1;\r
233 const char *last = strtab + tabsize - len;\r
234 const char *p;\r
235\r
236 for (p = strtab; p <= last; p++)\r
237 if (memcmp(p, s, len) == 0)\r
238 return p;\r
239 return NULL;\r
240}\r
241\r
242int fdt_move(const void *fdt, void *buf, int bufsize)\r
243{\r
244 FDT_CHECK_HEADER(fdt);\r
245\r
246 if (fdt_totalsize(fdt) > bufsize)\r
247 return -FDT_ERR_NOSPACE;\r
248\r
249 memmove(buf, fdt, fdt_totalsize(fdt));\r
250 return 0;\r
251}\r