Browse Source
When creating a tree with the sequential write functions, certain things have to be done in a certain order. You must create the memory reserve map and only then can you create the actual tree structure. The -FDT_ERR_BADSTATE return code is for if you try to do things out of order. However, we weren't checking that very thoroughly, so it was possible to generate a corrupted blob if, for example, you started calling fdt_begin_node() etc. before calling fdt_finish_reservemap(). This makes the state checking more thorough disallow that. Signed-off-by: David Gibson <david@gibson.dropbear.id.au> Tested-by: Alexey Kardashevskiy <aik@ozlabs.ru> Reviewed-by: Alexey Kardashevskiy <aik@ozlabs.ru> Reviewed-by: Simon Glass <sjg@chromium.org>main
David Gibson
7 years ago
5 changed files with 218 additions and 15 deletions
@ -0,0 +1,140 @@
@@ -0,0 +1,140 @@
|
||||
/* |
||||
* libfdt - Flat Device Tree manipulation |
||||
* Testcase for error handling with sequential write states |
||||
* Copyright (C) 2018 David Gibson, Red Hat Inc. |
||||
* |
||||
* 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 <stdlib.h> |
||||
#include <stdio.h> |
||||
#include <string.h> |
||||
#include <ctype.h> |
||||
#include <stdint.h> |
||||
|
||||
#include <libfdt.h> |
||||
|
||||
#include "tests.h" |
||||
#include "testdata.h" |
||||
|
||||
#define SPACE 65536 |
||||
|
||||
#define CHECK_OK(code) \ |
||||
do { \ |
||||
verbose_printf(" OK: %s\n", #code); \ |
||||
err = (code); \ |
||||
if (err) \ |
||||
FAIL(#code ": %s", fdt_strerror(err)); \ |
||||
} while (0) |
||||
|
||||
#define CHECK_BADSTATE(code) \ |
||||
do { \ |
||||
verbose_printf("BAD: %s\n", #code); \ |
||||
err = (code); \ |
||||
if (err == 0) \ |
||||
FAIL(#code ": succeeded in bad state"); \ |
||||
else if (err != -FDT_ERR_BADSTATE) \ |
||||
FAIL(#code ": %s", fdt_strerror(err)); \ |
||||
} while (0) |
||||
|
||||
int main(int argc, char *argv[]) |
||||
{ |
||||
void *fdt = NULL; |
||||
int err; |
||||
|
||||
test_init(argc, argv); |
||||
|
||||
fdt = xmalloc(SPACE); |
||||
|
||||
err = fdt_create(fdt, SPACE); |
||||
if (err) |
||||
FAIL("fdt_create(): %s", fdt_strerror(err)); |
||||
|
||||
/* Memory reserve state */ |
||||
|
||||
CHECK_BADSTATE(fdt_begin_node(fdt, "")); |
||||
CHECK_BADSTATE(fdt_property_string(fdt, "bad-str", "TEST_STRING_1")); |
||||
CHECK_BADSTATE(fdt_end_node(fdt)); |
||||
CHECK_BADSTATE(fdt_finish(fdt)); |
||||
|
||||
CHECK_OK(fdt_add_reservemap_entry(fdt, TEST_ADDR_1, TEST_SIZE_1)); |
||||
CHECK_OK(fdt_add_reservemap_entry(fdt, TEST_ADDR_2, TEST_SIZE_2)); |
||||
|
||||
CHECK_BADSTATE(fdt_begin_node(fdt, "")); |
||||
CHECK_BADSTATE(fdt_property_string(fdt, "bad-str", "TEST_STRING_1")); |
||||
CHECK_BADSTATE(fdt_end_node(fdt)); |
||||
CHECK_BADSTATE(fdt_finish(fdt)); |
||||
|
||||
CHECK_OK(fdt_finish_reservemap(fdt)); |
||||
|
||||
/* Structure state */ |
||||
|
||||
CHECK_BADSTATE(fdt_add_reservemap_entry(fdt, TEST_ADDR_1, TEST_SIZE_1)); |
||||
CHECK_BADSTATE(fdt_finish_reservemap(fdt)); |
||||
|
||||
CHECK_OK(fdt_begin_node(fdt, "")); |
||||
CHECK_OK(fdt_property_string(fdt, "compatible", "test_tree1")); |
||||
CHECK_OK(fdt_property_u32(fdt, "prop-int", TEST_VALUE_1)); |
||||
CHECK_OK(fdt_property_u64(fdt, "prop-int64", TEST_VALUE64_1)); |
||||
CHECK_OK(fdt_property_string(fdt, "prop-str", TEST_STRING_1)); |
||||
CHECK_OK(fdt_property_u32(fdt, "#address-cells", 1)); |
||||
CHECK_OK(fdt_property_u32(fdt, "#size-cells", 0)); |
||||
|
||||
CHECK_OK(fdt_begin_node(fdt, "subnode@1")); |
||||
CHECK_OK(fdt_property_string(fdt, "compatible", "subnode1")); |
||||
CHECK_OK(fdt_property_u32(fdt, "reg", 1)); |
||||
CHECK_OK(fdt_property_cell(fdt, "prop-int", TEST_VALUE_1)); |
||||
CHECK_OK(fdt_begin_node(fdt, "subsubnode")); |
||||
CHECK_OK(fdt_property(fdt, "compatible", "subsubnode1\0subsubnode", |
||||
23)); |
||||
CHECK_OK(fdt_property_cell(fdt, "prop-int", TEST_VALUE_1)); |
||||
CHECK_OK(fdt_end_node(fdt)); |
||||
CHECK_OK(fdt_begin_node(fdt, "ss1")); |
||||
CHECK_OK(fdt_end_node(fdt)); |
||||
CHECK_OK(fdt_end_node(fdt)); |
||||
|
||||
CHECK_OK(fdt_begin_node(fdt, "subnode@2")); |
||||
CHECK_OK(fdt_property_u32(fdt, "reg", 2)); |
||||
CHECK_OK(fdt_property_cell(fdt, "linux,phandle", PHANDLE_1)); |
||||
CHECK_OK(fdt_property_cell(fdt, "prop-int", TEST_VALUE_2)); |
||||
CHECK_OK(fdt_property_u32(fdt, "#address-cells", 1)); |
||||
CHECK_OK(fdt_property_u32(fdt, "#size-cells", 0)); |
||||
CHECK_OK(fdt_begin_node(fdt, "subsubnode@0")); |
||||
CHECK_OK(fdt_property_u32(fdt, "reg", 0)); |
||||
CHECK_OK(fdt_property_cell(fdt, "phandle", PHANDLE_2)); |
||||
CHECK_OK(fdt_property(fdt, "compatible", "subsubnode2\0subsubnode", |
||||
23)); |
||||
CHECK_OK(fdt_property_cell(fdt, "prop-int", TEST_VALUE_2)); |
||||
CHECK_OK(fdt_end_node(fdt)); |
||||
CHECK_OK(fdt_begin_node(fdt, "ss2")); |
||||
CHECK_OK(fdt_end_node(fdt)); |
||||
|
||||
CHECK_OK(fdt_end_node(fdt)); |
||||
|
||||
CHECK_OK(fdt_end_node(fdt)); |
||||
|
||||
CHECK_OK(fdt_finish(fdt)); |
||||
|
||||
/* Completed state */ |
||||
|
||||
CHECK_BADSTATE(fdt_add_reservemap_entry(fdt, TEST_ADDR_1, TEST_SIZE_1)); |
||||
CHECK_BADSTATE(fdt_finish_reservemap(fdt)); |
||||
CHECK_BADSTATE(fdt_begin_node(fdt, "")); |
||||
CHECK_BADSTATE(fdt_property_string(fdt, "bad-str", "TEST_STRING_1")); |
||||
CHECK_BADSTATE(fdt_end_node(fdt)); |
||||
CHECK_BADSTATE(fdt_finish(fdt)); |
||||
|
||||
PASS(); |
||||
} |
Loading…
Reference in new issue