0ef21055ee
This patch makes a number of minor changes to the ftdump debugging tool. * There was an endian bug in one place, which this fixes. * We now use const qualifiers in a number of places where we can * ftdump can now be instructed to read from stdin by giving "-" as the filename. * The buffer into which the blob is read is increased from 16k to 64k, and is now dynamically allocated. * ftdump now emits source in dts-v1 format Since ftdump is little used these days, these fixes are arguably of little use. On the other hand, I already did the work of making the changes some time back, so I guess we might as well fold these small fixes and improvements in. Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
208 lines
4.5 KiB
C
208 lines
4.5 KiB
C
/*
|
|
* ftdump.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>
|
|
|
|
#define FTDUMP_BUF_SIZE 65536
|
|
|
|
#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 int is_printable_string(const void *data, int len)
|
|
{
|
|
const char *s = data;
|
|
const char *ss;
|
|
|
|
/* zero length is not */
|
|
if (len == 0)
|
|
return 0;
|
|
|
|
/* must terminate with zero */
|
|
if (s[len - 1] != '\0')
|
|
return 0;
|
|
|
|
ss = s;
|
|
while (*s && isprint(*s))
|
|
s++;
|
|
|
|
/* not zero, or not done yet */
|
|
if (*s != '\0' || (s + 1 - ss) < len)
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
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 (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[])
|
|
{
|
|
FILE *fp;
|
|
char *buf;
|
|
int size;
|
|
|
|
if (argc < 2) {
|
|
fprintf(stderr, "supply input filename\n");
|
|
return 5;
|
|
}
|
|
|
|
if (strcmp(argv[1], "-") == 0) {
|
|
fp = stdin;
|
|
} else {
|
|
fp = fopen(argv[1], "rb");
|
|
if (fp == NULL) {
|
|
fprintf(stderr, "unable to open %s\n", argv[1]);
|
|
return 10;
|
|
}
|
|
}
|
|
|
|
buf = malloc(FTDUMP_BUF_SIZE);
|
|
if (!buf) {
|
|
fprintf(stderr, "Couldn't allocate %d byte buffer\n", FTDUMP_BUF_SIZE);
|
|
return 10;
|
|
}
|
|
|
|
size = fread(buf, 1, FTDUMP_BUF_SIZE, fp);
|
|
if (size == FTDUMP_BUF_SIZE) {
|
|
fprintf(stderr, "file too large (maximum is %d bytes)\n", FTDUMP_BUF_SIZE);
|
|
return 10;
|
|
}
|
|
|
|
dump_blob(buf);
|
|
|
|
fclose(fp);
|
|
|
|
return 0;
|
|
}
|