| // SPDX-License-Identifier: LGPL-2.1-or-later |
| /* |
| * libfdt - Flat Device Tree manipulation |
| * Test labels within values |
| * Copyright (C) 2008 David Gibson, IBM Corporation. |
| */ |
| |
| #include <stdlib.h> |
| #include <stdio.h> |
| #include <string.h> |
| #include <stdint.h> |
| #include <errno.h> |
| |
| #include <dlfcn.h> |
| |
| #include <libfdt.h> |
| |
| #include "tests.h" |
| #include "testdata.h" |
| |
| struct val_label { |
| const char *labelname; |
| int propoff; |
| }; |
| |
| static struct val_label labels1[] = { |
| { "start1", 0 }, |
| { "mid1", 2 }, |
| { "end1", -1 }, |
| }; |
| |
| static struct val_label labels2[] = { |
| { "start2", 0 }, |
| { "innerstart2", 0 }, |
| { "innermid2", 4 }, |
| { "innerend2", -1 }, |
| { "end2", -1 }, |
| }; |
| |
| static struct val_label labels3[] = { |
| { "start3", 0 }, |
| { "innerstart3", 0 }, |
| { "innermid3", 1 }, |
| { "innerend3", -1 }, |
| { "end3", -1 }, |
| }; |
| |
| static void check_prop_labels(void *sohandle, void *fdt, const char *name, |
| const struct val_label* labels, int n) |
| { |
| const struct fdt_property *prop; |
| const char *p; |
| int len; |
| int i; |
| |
| prop = fdt_get_property(fdt, 0, name, &len); |
| if (!prop) |
| FAIL("Couldn't locate property \"%s\"", name); |
| |
| p = dlsym(sohandle, name); |
| if (!p) |
| FAIL("Couldn't locate label symbol \"%s\"", name); |
| |
| if (p != (const char *)prop) |
| FAIL("Label \"%s\" does not point to correct property", name); |
| |
| for (i = 0; i < n; i++) { |
| int off = labels[i].propoff; |
| |
| if (off == -1) |
| off = len; |
| |
| p = dlsym(sohandle, labels[i].labelname); |
| if (!p) |
| FAIL("Couldn't locate label symbol \"%s\"", name); |
| |
| if ((p - prop->data) != off) |
| FAIL("Label \"%s\" points to offset %ld instead of %d" |
| "in property \"%s\"", labels[i].labelname, |
| (long)(p - prop->data), off, name); |
| } |
| } |
| |
| int main(int argc, char *argv[]) |
| { |
| void *sohandle; |
| void *fdt; |
| int err; |
| |
| test_init(argc, argv); |
| if (argc != 2) |
| CONFIG("Usage: %s <so file>", argv[0]); |
| |
| sohandle = dlopen(argv[1], RTLD_NOW); |
| if (!sohandle) |
| FAIL("Couldn't dlopen() %s", argv[1]); |
| |
| fdt = dlsym(sohandle, "dt_blob_start"); |
| if (!fdt) |
| FAIL("Couldn't locate \"dt_blob_start\" symbol in %s", |
| argv[1]); |
| |
| err = fdt_check_header(fdt); |
| if (err != 0) |
| FAIL("%s contains invalid tree: %s", argv[1], |
| fdt_strerror(err)); |
| |
| |
| check_prop_labels(sohandle, fdt, "prop1", labels1, ARRAY_SIZE(labels1)); |
| check_prop_labels(sohandle, fdt, "prop2", labels2, ARRAY_SIZE(labels2)); |
| check_prop_labels(sohandle, fdt, "prop3", labels3, ARRAY_SIZE(labels3)); |
| |
| PASS(); |
| } |