/* * libfdt - Flat Device Tree manipulation * Testcase for fdt_check_header * Copyright (C) 2018 David Gibson * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public License * as published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include <stdio.h> #include <libfdt.h> #include "tests.h" static void *dtdup(void *dt) { size_t bufsize = fdt_totalsize(dt); void *buf = xmalloc(bufsize); fdt_move(dt, buf, bufsize); return buf; } #define CHECK_MANGLE(exerr, code) \ do { \ void *fdt = dtdup(template); \ { code } \ err = fdt_check_header(fdt); \ verbose_printf("\"%s\" => %s\n", #code, fdt_strerror(err)); \ if (err != (exerr)) \ FAIL("fdt_check_header() didn't catch mangle %s", \ #code); \ free(fdt); \ } while (0) int main(int argc, char *argv[]) { void *template; int err; test_init(argc, argv); template = load_blob(argv[1]); /* Check that the base dt is valid before mangling it */ err = fdt_check_header(template); if (err != 0) FAIL("Base tree fails: %s", fdt_strerror(err)); /* Check a no-op mangle doesn't break things */ CHECK_MANGLE(0, ; ); /* Mess up the magic number */ CHECK_MANGLE(-FDT_ERR_BADMAGIC, fdt_set_magic(fdt, fdt_magic(fdt) ^ 0x1); ); CHECK_MANGLE(-FDT_ERR_BADMAGIC, fdt_set_magic(fdt, fdt_magic(fdt) ^ 0x80000000); ); /* Mess up the version */ CHECK_MANGLE(-FDT_ERR_BADVERSION, fdt_set_version(fdt, FDT_FIRST_SUPPORTED_VERSION - 1); fdt_set_last_comp_version(fdt, FDT_FIRST_SUPPORTED_VERSION - 1); ); CHECK_MANGLE(-FDT_ERR_BADVERSION, fdt_set_version(fdt, FDT_LAST_SUPPORTED_VERSION + 1); fdt_set_last_comp_version(fdt, FDT_LAST_SUPPORTED_VERSION + 1); ); CHECK_MANGLE(-FDT_ERR_BADVERSION, fdt_set_version(fdt, FDT_FIRST_SUPPORTED_VERSION); fdt_set_last_comp_version(fdt, FDT_LAST_SUPPORTED_VERSION); ); /* Out of bounds sizes */ CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_totalsize(fdt, FDT_V1_SIZE - 1); ); CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_totalsize(fdt, (uint32_t)INT_MAX + 1); ); /* Truncate within various blocks */ CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_totalsize(fdt, fdt_off_dt_struct(fdt) - 1); ); CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_totalsize(fdt, fdt_off_dt_strings(fdt) - 1); ); CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_totalsize(fdt, fdt_off_mem_rsvmap(fdt) - 1); ); CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_totalsize(fdt, fdt_off_dt_struct(fdt) + 1); ); CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_totalsize(fdt, fdt_off_dt_strings(fdt) + 1); ); CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_totalsize(fdt, fdt_off_mem_rsvmap(fdt) + 1); ); /* Negative block sizes */ CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_size_dt_struct(fdt, (uint32_t)-1); ); CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_size_dt_strings(fdt, (uint32_t)-1); ); CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_size_dt_struct(fdt, (uint32_t)INT_MIN); ); CHECK_MANGLE(-FDT_ERR_TRUNCATED, fdt_set_size_dt_strings(fdt, (uint32_t)INT_MIN); ); PASS(); }