| /* | 
 |  * fdtdump.c - Contributed by Pantelis Antoniou <pantelis.antoniou AT gmail.com> | 
 |  */ | 
 |  | 
 | #include <stdint.h> | 
 | #include <stdio.h> | 
 | #include <stdlib.h> | 
 | #include <string.h> | 
 | #include <ctype.h> | 
 |  | 
 | #include <fdt.h> | 
 | #include <libfdt_env.h> | 
 |  | 
 | #include "util.h" | 
 |  | 
 | #define ALIGN(x, a)	(((x) + ((a) - 1)) & ~((a) - 1)) | 
 | #define PALIGN(p, a)	((void *)(ALIGN((unsigned long)(p), (a)))) | 
 | #define GET_CELL(p)	(p += 4, *((const uint32_t *)(p-4))) | 
 |  | 
 | static void print_data(const char *data, int len) | 
 | { | 
 | 	int i; | 
 | 	const char *p = data; | 
 |  | 
 | 	/* no data, don't print */ | 
 | 	if (len == 0) | 
 | 		return; | 
 |  | 
 | 	if (util_is_printable_string(data, len)) { | 
 | 		printf(" = \"%s\"", (const char *)data); | 
 | 	} else if ((len % 4) == 0) { | 
 | 		printf(" = <"); | 
 | 		for (i = 0; i < len; i += 4) | 
 | 			printf("0x%08x%s", fdt32_to_cpu(GET_CELL(p)), | 
 | 			       i < (len - 4) ? " " : ""); | 
 | 		printf(">"); | 
 | 	} else { | 
 | 		printf(" = ["); | 
 | 		for (i = 0; i < len; i++) | 
 | 			printf("%02x%s", *p++, i < len - 1 ? " " : ""); | 
 | 		printf("]"); | 
 | 	} | 
 | } | 
 |  | 
 | static void dump_blob(void *blob) | 
 | { | 
 | 	struct fdt_header *bph = blob; | 
 | 	uint32_t off_mem_rsvmap = fdt32_to_cpu(bph->off_mem_rsvmap); | 
 | 	uint32_t off_dt = fdt32_to_cpu(bph->off_dt_struct); | 
 | 	uint32_t off_str = fdt32_to_cpu(bph->off_dt_strings); | 
 | 	struct fdt_reserve_entry *p_rsvmap = | 
 | 		(struct fdt_reserve_entry *)((char *)blob + off_mem_rsvmap); | 
 | 	const char *p_struct = (const char *)blob + off_dt; | 
 | 	const char *p_strings = (const char *)blob + off_str; | 
 | 	uint32_t version = fdt32_to_cpu(bph->version); | 
 | 	uint32_t totalsize = fdt32_to_cpu(bph->totalsize); | 
 | 	uint32_t tag; | 
 | 	const char *p, *s, *t; | 
 | 	int depth, sz, shift; | 
 | 	int i; | 
 | 	uint64_t addr, size; | 
 |  | 
 | 	depth = 0; | 
 | 	shift = 4; | 
 |  | 
 | 	printf("/dts-v1/;\n"); | 
 | 	printf("// magic:\t\t0x%x\n", fdt32_to_cpu(bph->magic)); | 
 | 	printf("// totalsize:\t\t0x%x (%d)\n", totalsize, totalsize); | 
 | 	printf("// off_dt_struct:\t0x%x\n", off_dt); | 
 | 	printf("// off_dt_strings:\t0x%x\n", off_str); | 
 | 	printf("// off_mem_rsvmap:\t0x%x\n", off_mem_rsvmap); | 
 | 	printf("// version:\t\t%d\n", version); | 
 | 	printf("// last_comp_version:\t%d\n", | 
 | 	       fdt32_to_cpu(bph->last_comp_version)); | 
 | 	if (version >= 2) | 
 | 		printf("// boot_cpuid_phys:\t0x%x\n", | 
 | 		       fdt32_to_cpu(bph->boot_cpuid_phys)); | 
 |  | 
 | 	if (version >= 3) | 
 | 		printf("// size_dt_strings:\t0x%x\n", | 
 | 		       fdt32_to_cpu(bph->size_dt_strings)); | 
 | 	if (version >= 17) | 
 | 		printf("// size_dt_struct:\t0x%x\n", | 
 | 		       fdt32_to_cpu(bph->size_dt_struct)); | 
 | 	printf("\n"); | 
 |  | 
 | 	for (i = 0; ; i++) { | 
 | 		addr = fdt64_to_cpu(p_rsvmap[i].address); | 
 | 		size = fdt64_to_cpu(p_rsvmap[i].size); | 
 | 		if (addr == 0 && size == 0) | 
 | 			break; | 
 |  | 
 | 		printf("/memreserve/ %llx %llx;\n", | 
 | 		       (unsigned long long)addr, (unsigned long long)size); | 
 | 	} | 
 |  | 
 | 	p = p_struct; | 
 | 	while ((tag = fdt32_to_cpu(GET_CELL(p))) != FDT_END) { | 
 |  | 
 | 		/* printf("tag: 0x%08x (%d)\n", tag, p - p_struct); */ | 
 |  | 
 | 		if (tag == FDT_BEGIN_NODE) { | 
 | 			s = p; | 
 | 			p = PALIGN(p + strlen(s) + 1, 4); | 
 |  | 
 | 			if (*s == '\0') | 
 | 				s = "/"; | 
 |  | 
 | 			printf("%*s%s {\n", depth * shift, "", s); | 
 |  | 
 | 			depth++; | 
 | 			continue; | 
 | 		} | 
 |  | 
 | 		if (tag == FDT_END_NODE) { | 
 | 			depth--; | 
 |  | 
 | 			printf("%*s};\n", depth * shift, ""); | 
 | 			continue; | 
 | 		} | 
 |  | 
 | 		if (tag == FDT_NOP) { | 
 | 			printf("%*s// [NOP]\n", depth * shift, ""); | 
 | 			continue; | 
 | 		} | 
 |  | 
 | 		if (tag != FDT_PROP) { | 
 | 			fprintf(stderr, "%*s ** Unknown tag 0x%08x\n", depth * shift, "", tag); | 
 | 			break; | 
 | 		} | 
 | 		sz = fdt32_to_cpu(GET_CELL(p)); | 
 | 		s = p_strings + fdt32_to_cpu(GET_CELL(p)); | 
 | 		if (version < 16 && sz >= 8) | 
 | 			p = PALIGN(p, 8); | 
 | 		t = p; | 
 |  | 
 | 		p = PALIGN(p + sz, 4); | 
 |  | 
 | 		printf("%*s%s", depth * shift, "", s); | 
 | 		print_data(t, sz); | 
 | 		printf(";\n"); | 
 | 	} | 
 | } | 
 |  | 
 |  | 
 | int main(int argc, char *argv[]) | 
 | { | 
 | 	char *buf; | 
 |  | 
 | 	if (argc < 2) { | 
 | 		fprintf(stderr, "supply input filename\n"); | 
 | 		return 5; | 
 | 	} | 
 |  | 
 | 	buf = utilfdt_read(argv[1]); | 
 | 	if (buf) | 
 | 		dump_blob(buf); | 
 | 	else | 
 | 		return 10; | 
 |  | 
 | 	return 0; | 
 | } |