// SPDX-License-Identifier: GPL-2.0+ /* * Copyright (c) 2016 NextThing Co * Copyright (c) 2016 Free Electrons */ #include <common.h> #include <command.h> #include <errno.h> #include <fdt_support.h> #include <image.h> #include <log.h> #include <malloc.h> #include <linux/sizes.h> #include <test/ut.h> #include <test/overlay.h> #include <test/suites.h> /* 4k ought to be enough for anybody */ #define FDT_COPY_SIZE (4 * SZ_1K) extern u32 __dtb_test_fdt_base_begin; extern u32 __dtb_test_fdt_overlay_begin; extern u32 __dtb_test_fdt_overlay_stacked_begin; static void *fdt; static int ut_fdt_getprop_u32_by_index(void *fdt, const char *path, const char *name, int index, u32 *out) { const fdt32_t *val; int node_off; int len; node_off = fdt_path_offset(fdt, path); if (node_off < 0) return node_off; val = fdt_getprop(fdt, node_off, name, &len); if (!val || (len < (sizeof(uint32_t) * (index + 1)))) return -FDT_ERR_NOTFOUND; *out = fdt32_to_cpu(*(val + index)); return 0; } static int ut_fdt_getprop_u32(void *fdt, const char *path, const char *name, u32 *out) { return ut_fdt_getprop_u32_by_index(fdt, path, name, 0, out); } static int fdt_getprop_str(void *fdt, const char *path, const char *name, const char **out) { int node_off; int len; node_off = fdt_path_offset(fdt, path); if (node_off < 0) return node_off; *out = fdt_stringlist_get(fdt, node_off, name, 0, &len); return len < 0 ? len : 0; } static int fdt_overlay_change_int_property(struct unit_test_state *uts) { u32 val = 0; ut_assertok(ut_fdt_getprop_u32(fdt, "/test-node", "test-int-property", &val)); ut_asserteq(43, val); return CMD_RET_SUCCESS; } OVERLAY_TEST(fdt_overlay_change_int_property, 0); static int fdt_overlay_change_str_property(struct unit_test_state *uts) { const char *val = NULL; ut_assertok(fdt_getprop_str(fdt, "/test-node", "test-str-property", &val)); ut_asserteq_str("foobar", val); return CMD_RET_SUCCESS; } OVERLAY_TEST(fdt_overlay_change_str_property, 0); static int fdt_overlay_add_str_property(struct unit_test_state *uts) { const char *val = NULL; ut_assertok(fdt_getprop_str(fdt, "/test-node", "test-str-property-2", &val)); ut_asserteq_str("foobar2", val); return CMD_RET_SUCCESS; } OVERLAY_TEST(fdt_overlay_add_str_property, 0); static int fdt_overlay_add_node_by_phandle(struct unit_test_state *uts) { int off; off = fdt_path_offset(fdt, "/test-node/new-node"); ut_assert(off >= 0); ut_assertnonnull(fdt_getprop(fdt, off, "new-property", NULL)); return CMD_RET_SUCCESS; } OVERLAY_TEST(fdt_overlay_add_node_by_phandle, 0); static int fdt_overlay_add_node_by_path(struct unit_test_state *uts) { int off; off = fdt_path_offset(fdt, "/new-node"); ut_assert(off >= 0); ut_assertnonnull(fdt_getprop(fdt, off, "new-property", NULL)); return CMD_RET_SUCCESS; } OVERLAY_TEST(fdt_overlay_add_node_by_path, 0); static int fdt_overlay_add_subnode_property(struct unit_test_state *uts) { int off; off = fdt_path_offset(fdt, "/test-node/sub-test-node"); ut_assert(off >= 0); ut_assertnonnull(fdt_getprop(fdt, off, "sub-test-property", NULL)); ut_assertnonnull(fdt_getprop(fdt, off, "new-sub-test-property", NULL)); return CMD_RET_SUCCESS; } OVERLAY_TEST(fdt_overlay_add_subnode_property, 0); static int fdt_overlay_local_phandle(struct unit_test_state *uts) { uint32_t local_phandle; u32 val = 0; int off; off = fdt_path_offset(fdt, "/new-local-node"); ut_assert(off >= 0); local_phandle = fdt_get_phandle(fdt, off); ut_assert(local_phandle); ut_assertok(ut_fdt_getprop_u32_by_index(fdt, "/", "test-several-phandle", 0, &val)); ut_asserteq(local_phandle, val); ut_assertok(ut_fdt_getprop_u32_by_index(fdt, "/", "test-several-phandle", 1, &val)); ut_asserteq(local_phandle, val); return CMD_RET_SUCCESS; } OVERLAY_TEST(fdt_overlay_local_phandle, 0); static int fdt_overlay_local_phandles(struct unit_test_state *uts) { uint32_t local_phandle, test_phandle; u32 val = 0; int off; off = fdt_path_offset(fdt, "/new-local-node"); ut_assert(off >= 0); local_phandle = fdt_get_phandle(fdt, off); ut_assert(local_phandle); off = fdt_path_offset(fdt, "/test-node"); ut_assert(off >= 0); test_phandle = fdt_get_phandle(fdt, off); ut_assert(test_phandle); ut_assertok(ut_fdt_getprop_u32_by_index(fdt, "/", "test-phandle", 0, &val)); ut_asserteq(test_phandle, val); ut_assertok(ut_fdt_getprop_u32_by_index(fdt, "/", "test-phandle", 1, &val)); ut_asserteq(local_phandle, val); return CMD_RET_SUCCESS; } OVERLAY_TEST(fdt_overlay_local_phandles, 0); static int fdt_overlay_stacked(struct unit_test_state *uts) { u32 val = 0; ut_assertok(ut_fdt_getprop_u32(fdt, "/new-local-node", "stacked-test-int-property", &val)); ut_asserteq(43, val); return CMD_RET_SUCCESS; } OVERLAY_TEST(fdt_overlay_stacked, 0); int do_ut_overlay(struct cmd_tbl *cmdtp, int flag, int argc, char *const argv[]) { struct unit_test *tests = ll_entry_start(struct unit_test, overlay_test); const int n_ents = ll_entry_count(struct unit_test, overlay_test); struct unit_test_state *uts; void *fdt_base = &__dtb_test_fdt_base_begin; void *fdt_overlay = &__dtb_test_fdt_overlay_begin; void *fdt_overlay_stacked = &__dtb_test_fdt_overlay_stacked_begin; void *fdt_overlay_copy, *fdt_overlay_stacked_copy; int ret = -ENOMEM; uts = calloc(1, sizeof(*uts)); if (!uts) return -ENOMEM; ut_assertok(fdt_check_header(fdt_base)); ut_assertok(fdt_check_header(fdt_overlay)); fdt = malloc(FDT_COPY_SIZE); if (!fdt) goto err1; fdt_overlay_copy = malloc(FDT_COPY_SIZE); if (!fdt_overlay_copy) goto err2; fdt_overlay_stacked_copy = malloc(FDT_COPY_SIZE); if (!fdt_overlay_stacked_copy) goto err3; /* * Resize the FDT to 4k so that we have room to operate on * * (and relocate it since the memory might be mapped * read-only) */ ut_assertok(fdt_open_into(fdt_base, fdt, FDT_COPY_SIZE)); /* * Resize the overlay to 4k so that we have room to operate on * * (and relocate it since the memory might be mapped * read-only) */ ut_assertok(fdt_open_into(fdt_overlay, fdt_overlay_copy, FDT_COPY_SIZE)); /* * Resize the stacked overlay to 4k so that we have room to operate on * * (and relocate it since the memory might be mapped * read-only) */ ut_assertok(fdt_open_into(fdt_overlay_stacked, fdt_overlay_stacked_copy, FDT_COPY_SIZE)); /* Apply the overlay */ ut_assertok(fdt_overlay_apply(fdt, fdt_overlay_copy)); /* Apply the stacked overlay */ ut_assertok(fdt_overlay_apply(fdt, fdt_overlay_stacked_copy)); ret = cmd_ut_category("overlay", "", tests, n_ents, argc, argv); free(fdt_overlay_stacked_copy); err3: free(fdt_overlay_copy); err2: free(fdt); err1: return ret; }