1 // SPDX-License-Identifier: GPL-2.0+
3 * (C) Copyright 2018 Linaro Ltd.
4 * Sam Protsenko <semen.protsenko@linaro.org>
7 #include <image-android-dt.h>
9 #include <linux/libfdt.h>
13 * Check if image header is correct.
15 * @param hdr_addr Start address of DT image
16 * Return: true if header is correct or false if header is incorrect
18 bool android_dt_check_header(ulong hdr_addr
)
20 const struct dt_table_header
*hdr
;
23 hdr
= map_sysmem(hdr_addr
, sizeof(*hdr
));
24 magic
= fdt32_to_cpu(hdr
->magic
);
27 return magic
== DT_TABLE_MAGIC
;
31 * Get the address of FDT (dtb or dtbo) in memory by its index in image.
33 * @param hdr_addr Start address of DT image
34 * @param index Index of desired FDT in image (starting from 0)
35 * @param[out] addr If not NULL, will contain address to specified FDT
36 * @param[out] size If not NULL, will contain size of specified FDT
38 * Return: true on success or false on error
40 bool android_dt_get_fdt_by_index(ulong hdr_addr
, u32 index
, ulong
*addr
,
43 const struct dt_table_header
*hdr
;
44 const struct dt_table_entry
*e
;
45 u32 entry_count
, entries_offset
, entry_size
;
47 u32 dt_offset
, dt_size
;
49 hdr
= map_sysmem(hdr_addr
, sizeof(*hdr
));
50 entry_count
= fdt32_to_cpu(hdr
->dt_entry_count
);
51 entries_offset
= fdt32_to_cpu(hdr
->dt_entries_offset
);
52 entry_size
= fdt32_to_cpu(hdr
->dt_entry_size
);
55 if (index
>= entry_count
) {
56 printf("Error: index >= dt_entry_count (%u >= %u)\n", index
,
61 e_addr
= hdr_addr
+ entries_offset
+ index
* entry_size
;
62 e
= map_sysmem(e_addr
, sizeof(*e
));
63 dt_offset
= fdt32_to_cpu(e
->dt_offset
);
64 dt_size
= fdt32_to_cpu(e
->dt_size
);
68 *addr
= hdr_addr
+ dt_offset
;
75 #if !defined(CONFIG_XPL_BUILD)
76 static void android_dt_print_fdt_info(const struct fdt_header
*fdt
)
80 const char *compatible
;
82 root_node_off
= fdt_path_offset(fdt
, "/");
83 if (root_node_off
< 0) {
84 printf("Error: Root node not found\n");
88 fdt_size
= fdt_totalsize(fdt
);
89 compatible
= fdt_getprop(fdt
, root_node_off
, "compatible",
92 printf(" (FDT)size = %d\n", fdt_size
);
93 printf(" (FDT)compatible = %s\n",
94 compatible
? compatible
: "(unknown)");
98 * Print information about DT image structure.
100 * @param hdr_addr Start address of DT image
102 void android_dt_print_contents(ulong hdr_addr
)
104 const struct dt_table_header
*hdr
;
105 u32 entry_count
, entries_offset
, entry_size
;
108 hdr
= map_sysmem(hdr_addr
, sizeof(*hdr
));
109 entry_count
= fdt32_to_cpu(hdr
->dt_entry_count
);
110 entries_offset
= fdt32_to_cpu(hdr
->dt_entries_offset
);
111 entry_size
= fdt32_to_cpu(hdr
->dt_entry_size
);
113 /* Print image header info */
114 printf("dt_table_header:\n");
115 printf(" magic = %08x\n", fdt32_to_cpu(hdr
->magic
));
116 printf(" total_size = %d\n", fdt32_to_cpu(hdr
->total_size
));
117 printf(" header_size = %d\n", fdt32_to_cpu(hdr
->header_size
));
118 printf(" dt_entry_size = %d\n", entry_size
);
119 printf(" dt_entry_count = %d\n", entry_count
);
120 printf(" dt_entries_offset = %d\n", entries_offset
);
121 printf(" page_size = %d\n", fdt32_to_cpu(hdr
->page_size
));
122 printf(" version = %d\n", fdt32_to_cpu(hdr
->version
));
126 /* Print image entries info */
127 for (i
= 0; i
< entry_count
; ++i
) {
128 const ulong e_addr
= hdr_addr
+ entries_offset
+ i
* entry_size
;
129 const struct dt_table_entry
*e
;
130 const struct fdt_header
*fdt
;
131 u32 dt_offset
, dt_size
;
134 e
= map_sysmem(e_addr
, sizeof(*e
));
135 dt_offset
= fdt32_to_cpu(e
->dt_offset
);
136 dt_size
= fdt32_to_cpu(e
->dt_size
);
138 printf("dt_table_entry[%d]:\n", i
);
139 printf(" dt_size = %d\n", dt_size
);
140 printf(" dt_offset = %d\n", dt_offset
);
141 printf(" id = %08x\n", fdt32_to_cpu(e
->id
));
142 printf(" rev = %08x\n", fdt32_to_cpu(e
->rev
));
143 for (j
= 0; j
< 4; ++j
) {
144 printf(" custom[%d] = %08x\n", j
,
145 fdt32_to_cpu(e
->custom
[j
]));
150 /* Print FDT info for this entry */
151 fdt
= map_sysmem(hdr_addr
+ dt_offset
, sizeof(*fdt
));
152 android_dt_print_fdt_info(fdt
);