121fdc27aSRafal Jaworowski /* 221fdc27aSRafal Jaworowski * libfdt - Flat Device Tree manipulation 321fdc27aSRafal Jaworowski * Copyright (C) 2006 David Gibson, IBM Corporation. 421fdc27aSRafal Jaworowski * 521fdc27aSRafal Jaworowski * libfdt is dual licensed: you can use it either under the terms of 621fdc27aSRafal Jaworowski * the GPL, or the BSD license, at your option. 721fdc27aSRafal Jaworowski * 821fdc27aSRafal Jaworowski * a) This library is free software; you can redistribute it and/or 921fdc27aSRafal Jaworowski * modify it under the terms of the GNU General Public License as 1021fdc27aSRafal Jaworowski * published by the Free Software Foundation; either version 2 of the 1121fdc27aSRafal Jaworowski * License, or (at your option) any later version. 1221fdc27aSRafal Jaworowski * 1321fdc27aSRafal Jaworowski * This library is distributed in the hope that it will be useful, 1421fdc27aSRafal Jaworowski * but WITHOUT ANY WARRANTY; without even the implied warranty of 1521fdc27aSRafal Jaworowski * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 1621fdc27aSRafal Jaworowski * GNU General Public License for more details. 1721fdc27aSRafal Jaworowski * 1821fdc27aSRafal Jaworowski * You should have received a copy of the GNU General Public 1921fdc27aSRafal Jaworowski * License along with this library; if not, write to the Free 2021fdc27aSRafal Jaworowski * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, 2121fdc27aSRafal Jaworowski * MA 02110-1301 USA 2221fdc27aSRafal Jaworowski * 2321fdc27aSRafal Jaworowski * Alternatively, 2421fdc27aSRafal Jaworowski * 2521fdc27aSRafal Jaworowski * b) Redistribution and use in source and binary forms, with or 2621fdc27aSRafal Jaworowski * without modification, are permitted provided that the following 2721fdc27aSRafal Jaworowski * conditions are met: 2821fdc27aSRafal Jaworowski * 2921fdc27aSRafal Jaworowski * 1. Redistributions of source code must retain the above 3021fdc27aSRafal Jaworowski * copyright notice, this list of conditions and the following 3121fdc27aSRafal Jaworowski * disclaimer. 3221fdc27aSRafal Jaworowski * 2. Redistributions in binary form must reproduce the above 3321fdc27aSRafal Jaworowski * copyright notice, this list of conditions and the following 3421fdc27aSRafal Jaworowski * disclaimer in the documentation and/or other materials 3521fdc27aSRafal Jaworowski * provided with the distribution. 3621fdc27aSRafal Jaworowski * 3721fdc27aSRafal Jaworowski * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 3821fdc27aSRafal Jaworowski * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, 3921fdc27aSRafal Jaworowski * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 4021fdc27aSRafal Jaworowski * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 4121fdc27aSRafal Jaworowski * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR 4221fdc27aSRafal Jaworowski * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 4321fdc27aSRafal Jaworowski * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 4421fdc27aSRafal Jaworowski * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 4521fdc27aSRafal Jaworowski * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 4621fdc27aSRafal Jaworowski * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 4721fdc27aSRafal Jaworowski * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR 4821fdc27aSRafal Jaworowski * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, 4921fdc27aSRafal Jaworowski * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 5021fdc27aSRafal Jaworowski */ 5121fdc27aSRafal Jaworowski #include "libfdt_env.h" 5221fdc27aSRafal Jaworowski 5321fdc27aSRafal Jaworowski #include <fdt.h> 5421fdc27aSRafal Jaworowski #include <libfdt.h> 5521fdc27aSRafal Jaworowski 5621fdc27aSRafal Jaworowski #include "libfdt_internal.h" 5721fdc27aSRafal Jaworowski 5821fdc27aSRafal Jaworowski int fdt_check_header(const void *fdt) 5921fdc27aSRafal Jaworowski { 6021fdc27aSRafal Jaworowski if (fdt_magic(fdt) == FDT_MAGIC) { 6121fdc27aSRafal Jaworowski /* Complete tree */ 6221fdc27aSRafal Jaworowski if (fdt_version(fdt) < FDT_FIRST_SUPPORTED_VERSION) 6321fdc27aSRafal Jaworowski return -FDT_ERR_BADVERSION; 6421fdc27aSRafal Jaworowski if (fdt_last_comp_version(fdt) > FDT_LAST_SUPPORTED_VERSION) 6521fdc27aSRafal Jaworowski return -FDT_ERR_BADVERSION; 6621fdc27aSRafal Jaworowski } else if (fdt_magic(fdt) == FDT_SW_MAGIC) { 6721fdc27aSRafal Jaworowski /* Unfinished sequential-write blob */ 6821fdc27aSRafal Jaworowski if (fdt_size_dt_struct(fdt) == 0) 6921fdc27aSRafal Jaworowski return -FDT_ERR_BADSTATE; 7021fdc27aSRafal Jaworowski } else { 7121fdc27aSRafal Jaworowski return -FDT_ERR_BADMAGIC; 7221fdc27aSRafal Jaworowski } 7321fdc27aSRafal Jaworowski 7421fdc27aSRafal Jaworowski return 0; 7521fdc27aSRafal Jaworowski } 7621fdc27aSRafal Jaworowski 7721fdc27aSRafal Jaworowski const void *fdt_offset_ptr(const void *fdt, int offset, unsigned int len) 7821fdc27aSRafal Jaworowski { 7921fdc27aSRafal Jaworowski const char *p; 8021fdc27aSRafal Jaworowski 8121fdc27aSRafal Jaworowski if (fdt_version(fdt) >= 0x11) 8221fdc27aSRafal Jaworowski if (((offset + len) < offset) 8321fdc27aSRafal Jaworowski || ((offset + len) > fdt_size_dt_struct(fdt))) 8421fdc27aSRafal Jaworowski return NULL; 8521fdc27aSRafal Jaworowski 8621fdc27aSRafal Jaworowski p = _fdt_offset_ptr(fdt, offset); 8721fdc27aSRafal Jaworowski 8821fdc27aSRafal Jaworowski if (p + len < p) 8921fdc27aSRafal Jaworowski return NULL; 9021fdc27aSRafal Jaworowski return p; 9121fdc27aSRafal Jaworowski } 9221fdc27aSRafal Jaworowski 9321fdc27aSRafal Jaworowski uint32_t fdt_next_tag(const void *fdt, int startoffset, int *nextoffset) 9421fdc27aSRafal Jaworowski { 9521fdc27aSRafal Jaworowski const uint32_t *tagp, *lenp; 9621fdc27aSRafal Jaworowski uint32_t tag; 9721fdc27aSRafal Jaworowski int offset = startoffset; 9821fdc27aSRafal Jaworowski const char *p; 9921fdc27aSRafal Jaworowski 10021fdc27aSRafal Jaworowski *nextoffset = -FDT_ERR_TRUNCATED; 10121fdc27aSRafal Jaworowski tagp = fdt_offset_ptr(fdt, offset, FDT_TAGSIZE); 10221fdc27aSRafal Jaworowski if (!tagp) 10321fdc27aSRafal Jaworowski return FDT_END; /* premature end */ 10421fdc27aSRafal Jaworowski tag = fdt32_to_cpu(*tagp); 10521fdc27aSRafal Jaworowski offset += FDT_TAGSIZE; 10621fdc27aSRafal Jaworowski 10721fdc27aSRafal Jaworowski *nextoffset = -FDT_ERR_BADSTRUCTURE; 10821fdc27aSRafal Jaworowski switch (tag) { 10921fdc27aSRafal Jaworowski case FDT_BEGIN_NODE: 11021fdc27aSRafal Jaworowski /* skip name */ 11121fdc27aSRafal Jaworowski do { 11221fdc27aSRafal Jaworowski p = fdt_offset_ptr(fdt, offset++, 1); 11321fdc27aSRafal Jaworowski } while (p && (*p != '\0')); 11421fdc27aSRafal Jaworowski if (!p) 11521fdc27aSRafal Jaworowski return FDT_END; /* premature end */ 11621fdc27aSRafal Jaworowski break; 11721fdc27aSRafal Jaworowski 11821fdc27aSRafal Jaworowski case FDT_PROP: 11921fdc27aSRafal Jaworowski lenp = fdt_offset_ptr(fdt, offset, sizeof(*lenp)); 12021fdc27aSRafal Jaworowski if (!lenp) 12121fdc27aSRafal Jaworowski return FDT_END; /* premature end */ 12221fdc27aSRafal Jaworowski /* skip-name offset, length and value */ 12321fdc27aSRafal Jaworowski offset += sizeof(struct fdt_property) - FDT_TAGSIZE 12421fdc27aSRafal Jaworowski + fdt32_to_cpu(*lenp); 12521fdc27aSRafal Jaworowski break; 12621fdc27aSRafal Jaworowski 12721fdc27aSRafal Jaworowski case FDT_END: 12821fdc27aSRafal Jaworowski case FDT_END_NODE: 12921fdc27aSRafal Jaworowski case FDT_NOP: 13021fdc27aSRafal Jaworowski break; 13121fdc27aSRafal Jaworowski 13221fdc27aSRafal Jaworowski default: 13321fdc27aSRafal Jaworowski return FDT_END; 13421fdc27aSRafal Jaworowski } 13521fdc27aSRafal Jaworowski 13621fdc27aSRafal Jaworowski if (!fdt_offset_ptr(fdt, startoffset, offset - startoffset)) 13721fdc27aSRafal Jaworowski return FDT_END; /* premature end */ 13821fdc27aSRafal Jaworowski 13921fdc27aSRafal Jaworowski *nextoffset = FDT_TAGALIGN(offset); 14021fdc27aSRafal Jaworowski return tag; 14121fdc27aSRafal Jaworowski } 14221fdc27aSRafal Jaworowski 14321fdc27aSRafal Jaworowski int _fdt_check_node_offset(const void *fdt, int offset) 14421fdc27aSRafal Jaworowski { 14521fdc27aSRafal Jaworowski if ((offset < 0) || (offset % FDT_TAGSIZE) 14621fdc27aSRafal Jaworowski || (fdt_next_tag(fdt, offset, &offset) != FDT_BEGIN_NODE)) 14721fdc27aSRafal Jaworowski return -FDT_ERR_BADOFFSET; 14821fdc27aSRafal Jaworowski 14921fdc27aSRafal Jaworowski return offset; 15021fdc27aSRafal Jaworowski } 15121fdc27aSRafal Jaworowski 152*52baf267SWarner Losh int _fdt_check_prop_offset(const void *fdt, int offset) 153*52baf267SWarner Losh { 154*52baf267SWarner Losh if ((offset < 0) || (offset % FDT_TAGSIZE) 155*52baf267SWarner Losh || (fdt_next_tag(fdt, offset, &offset) != FDT_PROP)) 156*52baf267SWarner Losh return -FDT_ERR_BADOFFSET; 157*52baf267SWarner Losh 158*52baf267SWarner Losh return offset; 159*52baf267SWarner Losh } 160*52baf267SWarner Losh 16121fdc27aSRafal Jaworowski int fdt_next_node(const void *fdt, int offset, int *depth) 16221fdc27aSRafal Jaworowski { 16321fdc27aSRafal Jaworowski int nextoffset = 0; 16421fdc27aSRafal Jaworowski uint32_t tag; 16521fdc27aSRafal Jaworowski 16621fdc27aSRafal Jaworowski if (offset >= 0) 16721fdc27aSRafal Jaworowski if ((nextoffset = _fdt_check_node_offset(fdt, offset)) < 0) 16821fdc27aSRafal Jaworowski return nextoffset; 16921fdc27aSRafal Jaworowski 17021fdc27aSRafal Jaworowski do { 17121fdc27aSRafal Jaworowski offset = nextoffset; 17221fdc27aSRafal Jaworowski tag = fdt_next_tag(fdt, offset, &nextoffset); 17321fdc27aSRafal Jaworowski 17421fdc27aSRafal Jaworowski switch (tag) { 17521fdc27aSRafal Jaworowski case FDT_PROP: 17621fdc27aSRafal Jaworowski case FDT_NOP: 17721fdc27aSRafal Jaworowski break; 17821fdc27aSRafal Jaworowski 17921fdc27aSRafal Jaworowski case FDT_BEGIN_NODE: 18021fdc27aSRafal Jaworowski if (depth) 18121fdc27aSRafal Jaworowski (*depth)++; 18221fdc27aSRafal Jaworowski break; 18321fdc27aSRafal Jaworowski 18421fdc27aSRafal Jaworowski case FDT_END_NODE: 18521fdc27aSRafal Jaworowski if (depth && ((--(*depth)) < 0)) 18621fdc27aSRafal Jaworowski return nextoffset; 18721fdc27aSRafal Jaworowski break; 18821fdc27aSRafal Jaworowski 18921fdc27aSRafal Jaworowski case FDT_END: 19021fdc27aSRafal Jaworowski if ((nextoffset >= 0) 19121fdc27aSRafal Jaworowski || ((nextoffset == -FDT_ERR_TRUNCATED) && !depth)) 19221fdc27aSRafal Jaworowski return -FDT_ERR_NOTFOUND; 19321fdc27aSRafal Jaworowski else 19421fdc27aSRafal Jaworowski return nextoffset; 19521fdc27aSRafal Jaworowski } 19621fdc27aSRafal Jaworowski } while (tag != FDT_BEGIN_NODE); 19721fdc27aSRafal Jaworowski 19821fdc27aSRafal Jaworowski return offset; 19921fdc27aSRafal Jaworowski } 20021fdc27aSRafal Jaworowski 20121fdc27aSRafal Jaworowski const char *_fdt_find_string(const char *strtab, int tabsize, const char *s) 20221fdc27aSRafal Jaworowski { 20321fdc27aSRafal Jaworowski int len = strlen(s) + 1; 20421fdc27aSRafal Jaworowski const char *last = strtab + tabsize - len; 20521fdc27aSRafal Jaworowski const char *p; 20621fdc27aSRafal Jaworowski 20721fdc27aSRafal Jaworowski for (p = strtab; p <= last; p++) 20821fdc27aSRafal Jaworowski if (memcmp(p, s, len) == 0) 20921fdc27aSRafal Jaworowski return p; 21021fdc27aSRafal Jaworowski return NULL; 21121fdc27aSRafal Jaworowski } 21221fdc27aSRafal Jaworowski 21321fdc27aSRafal Jaworowski int fdt_move(const void *fdt, void *buf, int bufsize) 21421fdc27aSRafal Jaworowski { 21521fdc27aSRafal Jaworowski FDT_CHECK_HEADER(fdt); 21621fdc27aSRafal Jaworowski 21721fdc27aSRafal Jaworowski if (fdt_totalsize(fdt) > bufsize) 21821fdc27aSRafal Jaworowski return -FDT_ERR_NOSPACE; 21921fdc27aSRafal Jaworowski 22021fdc27aSRafal Jaworowski memmove(buf, fdt, fdt_totalsize(fdt)); 22121fdc27aSRafal Jaworowski return 0; 22221fdc27aSRafal Jaworowski } 223