1*cd60bc56SAndroid Build Coastguard Worker // SPDX-License-Identifier: LGPL-2.1-or-later
2*cd60bc56SAndroid Build Coastguard Worker /*
3*cd60bc56SAndroid Build Coastguard Worker * libfdt - Flat Device Tree manipulation
4*cd60bc56SAndroid Build Coastguard Worker * Tests if two given dtbs are structurally equal (including order)
5*cd60bc56SAndroid Build Coastguard Worker * Copyright (C) 2007 David Gibson, IBM Corporation.
6*cd60bc56SAndroid Build Coastguard Worker */
7*cd60bc56SAndroid Build Coastguard Worker
8*cd60bc56SAndroid Build Coastguard Worker #include <stdlib.h>
9*cd60bc56SAndroid Build Coastguard Worker #include <stdio.h>
10*cd60bc56SAndroid Build Coastguard Worker #include <string.h>
11*cd60bc56SAndroid Build Coastguard Worker #include <stdint.h>
12*cd60bc56SAndroid Build Coastguard Worker
13*cd60bc56SAndroid Build Coastguard Worker #include <libfdt.h>
14*cd60bc56SAndroid Build Coastguard Worker
15*cd60bc56SAndroid Build Coastguard Worker #include "tests.h"
16*cd60bc56SAndroid Build Coastguard Worker #include "testdata.h"
17*cd60bc56SAndroid Build Coastguard Worker
18*cd60bc56SAndroid Build Coastguard Worker static int notequal; /* = 0 */
19*cd60bc56SAndroid Build Coastguard Worker
20*cd60bc56SAndroid Build Coastguard Worker #define MISMATCH(fmt, ...) \
21*cd60bc56SAndroid Build Coastguard Worker do { \
22*cd60bc56SAndroid Build Coastguard Worker if (notequal) \
23*cd60bc56SAndroid Build Coastguard Worker PASS(); \
24*cd60bc56SAndroid Build Coastguard Worker else \
25*cd60bc56SAndroid Build Coastguard Worker FAIL(fmt, ##__VA_ARGS__); \
26*cd60bc56SAndroid Build Coastguard Worker } while (0)
27*cd60bc56SAndroid Build Coastguard Worker
28*cd60bc56SAndroid Build Coastguard Worker #define MATCH() \
29*cd60bc56SAndroid Build Coastguard Worker do { \
30*cd60bc56SAndroid Build Coastguard Worker if (!notequal) \
31*cd60bc56SAndroid Build Coastguard Worker PASS(); \
32*cd60bc56SAndroid Build Coastguard Worker else \
33*cd60bc56SAndroid Build Coastguard Worker FAIL("Trees match which shouldn't"); \
34*cd60bc56SAndroid Build Coastguard Worker } while (0)
35*cd60bc56SAndroid Build Coastguard Worker
36*cd60bc56SAndroid Build Coastguard Worker #define CHECK(code) \
37*cd60bc56SAndroid Build Coastguard Worker { \
38*cd60bc56SAndroid Build Coastguard Worker err = (code); \
39*cd60bc56SAndroid Build Coastguard Worker if (err) \
40*cd60bc56SAndroid Build Coastguard Worker FAIL(#code ": %s", fdt_strerror(err)); \
41*cd60bc56SAndroid Build Coastguard Worker }
42*cd60bc56SAndroid Build Coastguard Worker
compare_mem_rsv(const void * fdt1,const void * fdt2)43*cd60bc56SAndroid Build Coastguard Worker static void compare_mem_rsv(const void *fdt1, const void *fdt2)
44*cd60bc56SAndroid Build Coastguard Worker {
45*cd60bc56SAndroid Build Coastguard Worker int i;
46*cd60bc56SAndroid Build Coastguard Worker uint64_t addr1, size1, addr2, size2;
47*cd60bc56SAndroid Build Coastguard Worker int err;
48*cd60bc56SAndroid Build Coastguard Worker
49*cd60bc56SAndroid Build Coastguard Worker if (fdt_num_mem_rsv(fdt1) != fdt_num_mem_rsv(fdt2))
50*cd60bc56SAndroid Build Coastguard Worker MISMATCH("Trees have different number of reserve entries");
51*cd60bc56SAndroid Build Coastguard Worker for (i = 0; i < fdt_num_mem_rsv(fdt1); i++) {
52*cd60bc56SAndroid Build Coastguard Worker CHECK(fdt_get_mem_rsv(fdt1, i, &addr1, &size1));
53*cd60bc56SAndroid Build Coastguard Worker CHECK(fdt_get_mem_rsv(fdt2, i, &addr2, &size2));
54*cd60bc56SAndroid Build Coastguard Worker
55*cd60bc56SAndroid Build Coastguard Worker if ((addr1 != addr2) || (size1 != size2))
56*cd60bc56SAndroid Build Coastguard Worker MISMATCH("Mismatch in reserve entry %d: "
57*cd60bc56SAndroid Build Coastguard Worker "(0x%llx, 0x%llx) != (0x%llx, 0x%llx)", i,
58*cd60bc56SAndroid Build Coastguard Worker (unsigned long long)addr1,
59*cd60bc56SAndroid Build Coastguard Worker (unsigned long long)size1,
60*cd60bc56SAndroid Build Coastguard Worker (unsigned long long)addr2,
61*cd60bc56SAndroid Build Coastguard Worker (unsigned long long)size2);
62*cd60bc56SAndroid Build Coastguard Worker }
63*cd60bc56SAndroid Build Coastguard Worker }
64*cd60bc56SAndroid Build Coastguard Worker
compare_structure(const void * fdt1,const void * fdt2)65*cd60bc56SAndroid Build Coastguard Worker static void compare_structure(const void *fdt1, const void *fdt2)
66*cd60bc56SAndroid Build Coastguard Worker {
67*cd60bc56SAndroid Build Coastguard Worker int nextoffset1 = 0, nextoffset2 = 0;
68*cd60bc56SAndroid Build Coastguard Worker int offset1, offset2;
69*cd60bc56SAndroid Build Coastguard Worker uint32_t tag1, tag2;
70*cd60bc56SAndroid Build Coastguard Worker const char *name1, *name2;
71*cd60bc56SAndroid Build Coastguard Worker int err;
72*cd60bc56SAndroid Build Coastguard Worker const struct fdt_property *prop1, *prop2;
73*cd60bc56SAndroid Build Coastguard Worker int len1, len2;
74*cd60bc56SAndroid Build Coastguard Worker
75*cd60bc56SAndroid Build Coastguard Worker while (1) {
76*cd60bc56SAndroid Build Coastguard Worker do {
77*cd60bc56SAndroid Build Coastguard Worker offset1 = nextoffset1;
78*cd60bc56SAndroid Build Coastguard Worker tag1 = fdt_next_tag(fdt1, offset1, &nextoffset1);
79*cd60bc56SAndroid Build Coastguard Worker } while (tag1 == FDT_NOP);
80*cd60bc56SAndroid Build Coastguard Worker do {
81*cd60bc56SAndroid Build Coastguard Worker offset2 = nextoffset2;
82*cd60bc56SAndroid Build Coastguard Worker tag2 = fdt_next_tag(fdt2, offset2, &nextoffset2);
83*cd60bc56SAndroid Build Coastguard Worker } while (tag2 == FDT_NOP);
84*cd60bc56SAndroid Build Coastguard Worker
85*cd60bc56SAndroid Build Coastguard Worker if (tag1 != tag2)
86*cd60bc56SAndroid Build Coastguard Worker MISMATCH("Tag mismatch (%d != %d) at (%d, %d)",
87*cd60bc56SAndroid Build Coastguard Worker tag1, tag2, offset1, offset2);
88*cd60bc56SAndroid Build Coastguard Worker
89*cd60bc56SAndroid Build Coastguard Worker switch (tag1) {
90*cd60bc56SAndroid Build Coastguard Worker case FDT_BEGIN_NODE:
91*cd60bc56SAndroid Build Coastguard Worker name1 = fdt_get_name(fdt1, offset1, &err);
92*cd60bc56SAndroid Build Coastguard Worker if (!name1)
93*cd60bc56SAndroid Build Coastguard Worker FAIL("fdt_get_name(fdt1, %d, ..): %s",
94*cd60bc56SAndroid Build Coastguard Worker offset1, fdt_strerror(err));
95*cd60bc56SAndroid Build Coastguard Worker name2 = fdt_get_name(fdt2, offset2, NULL);
96*cd60bc56SAndroid Build Coastguard Worker if (!name2)
97*cd60bc56SAndroid Build Coastguard Worker FAIL("fdt_get_name(fdt2, %d, ..): %s",
98*cd60bc56SAndroid Build Coastguard Worker offset2, fdt_strerror(err));
99*cd60bc56SAndroid Build Coastguard Worker
100*cd60bc56SAndroid Build Coastguard Worker if (!streq(name1, name2))
101*cd60bc56SAndroid Build Coastguard Worker MISMATCH("Name mismatch (\"%s\" != \"%s\") at (%d, %d)",
102*cd60bc56SAndroid Build Coastguard Worker name1, name2, offset1, offset2);
103*cd60bc56SAndroid Build Coastguard Worker break;
104*cd60bc56SAndroid Build Coastguard Worker
105*cd60bc56SAndroid Build Coastguard Worker case FDT_PROP:
106*cd60bc56SAndroid Build Coastguard Worker prop1 = fdt_offset_ptr(fdt1, offset1, sizeof(*prop1));
107*cd60bc56SAndroid Build Coastguard Worker if (!prop1)
108*cd60bc56SAndroid Build Coastguard Worker FAIL("Could get fdt1 property at %d", offset1);
109*cd60bc56SAndroid Build Coastguard Worker prop2 = fdt_offset_ptr(fdt2, offset2, sizeof(*prop2));
110*cd60bc56SAndroid Build Coastguard Worker if (!prop2)
111*cd60bc56SAndroid Build Coastguard Worker FAIL("Could get fdt2 property at %d", offset2);
112*cd60bc56SAndroid Build Coastguard Worker
113*cd60bc56SAndroid Build Coastguard Worker name1 = fdt_string(fdt1, fdt32_to_cpu(prop1->nameoff));
114*cd60bc56SAndroid Build Coastguard Worker name2 = fdt_string(fdt2, fdt32_to_cpu(prop2->nameoff));
115*cd60bc56SAndroid Build Coastguard Worker if (!streq(name1, name2))
116*cd60bc56SAndroid Build Coastguard Worker MISMATCH("Property name mismatch \"%s\" != \"%s\" "
117*cd60bc56SAndroid Build Coastguard Worker "at (%d, %d)", name1, name2, offset1, offset2);
118*cd60bc56SAndroid Build Coastguard Worker len1 = fdt32_to_cpu(prop1->len);
119*cd60bc56SAndroid Build Coastguard Worker len2 = fdt32_to_cpu(prop2->len);
120*cd60bc56SAndroid Build Coastguard Worker if (len1 != len2)
121*cd60bc56SAndroid Build Coastguard Worker MISMATCH("Property length mismatch %u != %u "
122*cd60bc56SAndroid Build Coastguard Worker "at (%d, %d)", len1, len2, offset1, offset2);
123*cd60bc56SAndroid Build Coastguard Worker
124*cd60bc56SAndroid Build Coastguard Worker if (memcmp(prop1->data, prop2->data, len1) != 0)
125*cd60bc56SAndroid Build Coastguard Worker MISMATCH("Property value mismatch at (%d, %d)",
126*cd60bc56SAndroid Build Coastguard Worker offset1, offset2);
127*cd60bc56SAndroid Build Coastguard Worker break;
128*cd60bc56SAndroid Build Coastguard Worker
129*cd60bc56SAndroid Build Coastguard Worker case FDT_END:
130*cd60bc56SAndroid Build Coastguard Worker return;
131*cd60bc56SAndroid Build Coastguard Worker }
132*cd60bc56SAndroid Build Coastguard Worker }
133*cd60bc56SAndroid Build Coastguard Worker }
134*cd60bc56SAndroid Build Coastguard Worker
main(int argc,char * argv[])135*cd60bc56SAndroid Build Coastguard Worker int main(int argc, char *argv[])
136*cd60bc56SAndroid Build Coastguard Worker {
137*cd60bc56SAndroid Build Coastguard Worker void *fdt1, *fdt2;
138*cd60bc56SAndroid Build Coastguard Worker uint32_t cpuid1, cpuid2;
139*cd60bc56SAndroid Build Coastguard Worker
140*cd60bc56SAndroid Build Coastguard Worker test_init(argc, argv);
141*cd60bc56SAndroid Build Coastguard Worker if ((argc != 3)
142*cd60bc56SAndroid Build Coastguard Worker && ((argc != 4) || !streq(argv[1], "-n")))
143*cd60bc56SAndroid Build Coastguard Worker CONFIG("Usage: %s [-n] <dtb file> <dtb file>", argv[0]);
144*cd60bc56SAndroid Build Coastguard Worker if (argc == 4)
145*cd60bc56SAndroid Build Coastguard Worker notequal = 1;
146*cd60bc56SAndroid Build Coastguard Worker
147*cd60bc56SAndroid Build Coastguard Worker fdt1 = load_blob(argv[argc-2]);
148*cd60bc56SAndroid Build Coastguard Worker fdt2 = load_blob(argv[argc-1]);
149*cd60bc56SAndroid Build Coastguard Worker
150*cd60bc56SAndroid Build Coastguard Worker compare_mem_rsv(fdt1, fdt2);
151*cd60bc56SAndroid Build Coastguard Worker compare_structure(fdt1, fdt2);
152*cd60bc56SAndroid Build Coastguard Worker
153*cd60bc56SAndroid Build Coastguard Worker cpuid1 = fdt_boot_cpuid_phys(fdt1);
154*cd60bc56SAndroid Build Coastguard Worker cpuid2 = fdt_boot_cpuid_phys(fdt2);
155*cd60bc56SAndroid Build Coastguard Worker if (cpuid1 != cpuid2)
156*cd60bc56SAndroid Build Coastguard Worker MISMATCH("boot_cpuid_phys mismatch 0x%x != 0x%x",
157*cd60bc56SAndroid Build Coastguard Worker cpuid1, cpuid2);
158*cd60bc56SAndroid Build Coastguard Worker
159*cd60bc56SAndroid Build Coastguard Worker MATCH();
160*cd60bc56SAndroid Build Coastguard Worker }
161