scripts/dtc: Update to upstream version v1.4.7-57-gf267e674d145

This adds the following commits from upstream:

f267e674d145 checks: Fix crash with multiple source annotations
3616b9a811b6 checks: Use source position information for check failures
2bdbd07a1223 checks: Make each message output atomic
a1eff70c02cf util: Add xa{v}sprintf_append functions
82a52ce4573b libfdt: Add a test for fdt_getprop_by_offset()
607b8586b383 PEP8 / Flake8 cleanups for setup.py
f9c0a425b648 Remove broken objdir / srcdir support
5182b5e6f28c pylibfdt: Use common PREFIX variable
d45bf1f5f2a6 Refine make tests_clean target
99284c4db9cb Refine pylibfdt_clean target
a4629cfaedfb Refine libfdt_clean target
08380fc43aa2 tests: Use modern octal literals for Python
8113c00b99d3 pylibfdt: Allow switch to Python 3 via environment variable PYTHON
11738cf01f15 libfdt: Don't use memcpy to handle unaligned reads on ARM
86a288a73670 checks: Restructure check_msg to decrease indentation
5667e7ef9a9a annotations: add the annotation functionality
8e20ccf52f90 annotations: add positions
ca930e20bb54 tests: Don't lose errors from make checkm
43366bb4eeee tests: Property count valgrind errors in wrapped tests
5062516fb8cb srcpos: Remove srcpos_empty
a3143fafbf83 Revert "annotations: add positions"
403cc79f06a1 checks: Update SPI bus check for 'spi-slave'
baa1d2cf7894 annotations: add positions
ff2ad38f6a5a Merge remote-tracking branch 'origin/pr/18'
aa7254d9cb17 libfdt: return correct value if #size-cells property is not present
49903aed7783 use ptrdiff_t modifier for printing pointer differences
da2b691ccf68 treesource: Fix dts output for phandles in middle of a sequence of ints
8f8b77a0d62d tests: Wrap check_align() calls with base_run_test()
522d81d572f2 Fix dts output with a REF_PATH marker
e45198c98359 Added test cases for target references
0fcffda15e9f Merge nodes with local target label references
1e4a0928f3b3 pylibfdt: Don't have setup.py depend on where it's invoked from
ca399b14956f pylibfdt: Eliminate run_setup make function
98972f1b3e33 pylibfdt: Improved version extraction
7ba2be6cda5f pylibfdt: Don't silence setup.py when V=1
7691f9d39301 pylibfdt: Make SETUP make variable
855b9963def9 pylibfdt: Simpler CFLAGS handling
47cafbeeb977 pylibfdt: Link extension module with libfdt rather than rebuilding
dd695d6afb19 pylibfdt: Correctly set build output directory
59327523d0d8 pylibfdt: We don't need include files from the base directory
e84742aa7b93 checks: fix simple-bus compatible matching
8c59a97ce096 Fix missing labels when emitting dts format
d448f9a5fd94 Revert dts output formatting changes of spaces around brackets

Signed-off-by: Rob Herring <robh@kernel.org>
This commit is contained in:
Rob Herring 2018-11-28 18:37:35 -06:00
parent e8b1dee214
commit c2e7075ca8
17 changed files with 436 additions and 147 deletions

View File

@ -19,6 +19,7 @@
*/ */
#include "dtc.h" #include "dtc.h"
#include "srcpos.h"
#ifdef TRACE_CHECKS #ifdef TRACE_CHECKS
#define TRACE(c, ...) \ #define TRACE(c, ...) \
@ -78,23 +79,56 @@ static inline void PRINTF(5, 6) check_msg(struct check *c, struct dt_info *dti,
const char *fmt, ...) const char *fmt, ...)
{ {
va_list ap; va_list ap;
va_start(ap, fmt); char *str = NULL;
struct srcpos *pos = NULL;
char *file_str;
if ((c->warn && (quiet < 1)) if (!(c->warn && (quiet < 1)) && !(c->error && (quiet < 2)))
|| (c->error && (quiet < 2))) { return;
fprintf(stderr, "%s: %s (%s): ",
strcmp(dti->outname, "-") ? dti->outname : "<stdout>", if (prop && prop->srcpos)
(c->error) ? "ERROR" : "Warning", c->name); pos = prop->srcpos;
if (node) { else if (node && node->srcpos)
fprintf(stderr, "%s", node->fullpath); pos = node->srcpos;
if (prop)
fprintf(stderr, ":%s", prop->name); if (pos) {
fputs(": ", stderr); file_str = srcpos_string(pos);
} xasprintf(&str, "%s", file_str);
vfprintf(stderr, fmt, ap); free(file_str);
fprintf(stderr, "\n"); } else if (streq(dti->outname, "-")) {
xasprintf(&str, "<stdout>");
} else {
xasprintf(&str, "%s", dti->outname);
} }
xasprintf_append(&str, ": %s (%s): ",
(c->error) ? "ERROR" : "Warning", c->name);
if (node) {
if (prop)
xasprintf_append(&str, "%s:%s: ", node->fullpath, prop->name);
else
xasprintf_append(&str, "%s: ", node->fullpath);
}
va_start(ap, fmt);
xavsprintf_append(&str, fmt, ap);
va_end(ap); va_end(ap);
xasprintf_append(&str, "\n");
if (!prop && pos) {
pos = node->srcpos;
while (pos->next) {
pos = pos->next;
file_str = srcpos_string(pos);
xasprintf_append(&str, " also defined at %s\n", file_str);
free(file_str);
}
}
fputs(str, stderr);
} }
#define FAIL(c, dti, node, ...) \ #define FAIL(c, dti, node, ...) \
@ -910,7 +944,7 @@ static bool node_is_compatible(struct node *node, const char *compat)
for (str = prop->val.val, end = str + prop->val.len; str < end; for (str = prop->val.val, end = str + prop->val.len; str < end;
str += strnlen(str, end - str) + 1) { str += strnlen(str, end - str) + 1) {
if (strprefixeq(str, end - str, compat)) if (streq(str, compat))
return true; return true;
} }
return false; return false;
@ -921,7 +955,8 @@ static void check_simple_bus_bridge(struct check *c, struct dt_info *dti, struct
if (node_is_compatible(node, "simple-bus")) if (node_is_compatible(node, "simple-bus"))
node->bus = &simple_bus; node->bus = &simple_bus;
} }
WARNING(simple_bus_bridge, check_simple_bus_bridge, NULL, &addr_size_cells); WARNING(simple_bus_bridge, check_simple_bus_bridge, NULL,
&addr_size_cells, &compatible_is_string_list);
static void check_simple_bus_reg(struct check *c, struct dt_info *dti, struct node *node) static void check_simple_bus_reg(struct check *c, struct dt_info *dti, struct node *node)
{ {
@ -1035,6 +1070,7 @@ static const struct bus_type spi_bus = {
static void check_spi_bus_bridge(struct check *c, struct dt_info *dti, struct node *node) static void check_spi_bus_bridge(struct check *c, struct dt_info *dti, struct node *node)
{ {
int spi_addr_cells = 1;
if (strprefixeq(node->name, node->basenamelen, "spi")) { if (strprefixeq(node->name, node->basenamelen, "spi")) {
node->bus = &spi_bus; node->bus = &spi_bus;
@ -1063,7 +1099,9 @@ static void check_spi_bus_bridge(struct check *c, struct dt_info *dti, struct no
if (node->bus != &spi_bus || !node->children) if (node->bus != &spi_bus || !node->children)
return; return;
if (node_addr_cells(node) != 1) if (get_property(node, "spi-slave"))
spi_addr_cells = 0;
if (node_addr_cells(node) != spi_addr_cells)
FAIL(c, dti, node, "incorrect #address-cells for SPI bus"); FAIL(c, dti, node, "incorrect #address-cells for SPI bus");
if (node_size_cells(node) != 0) if (node_size_cells(node) != 0)
FAIL(c, dti, node, "incorrect #size-cells for SPI bus"); FAIL(c, dti, node, "incorrect #size-cells for SPI bus");
@ -1082,6 +1120,9 @@ static void check_spi_bus_reg(struct check *c, struct dt_info *dti, struct node
if (!node->parent || (node->parent->bus != &spi_bus)) if (!node->parent || (node->parent->bus != &spi_bus))
return; return;
if (get_property(node->parent, "spi-slave"))
return;
prop = get_property(node, "reg"); prop = get_property(node, "reg");
if (prop) if (prop)
cells = (cell_t *)prop->val.val; cells = (cell_t *)prop->val.val;

View File

@ -213,14 +213,14 @@ static void PRINTF(1, 2) lexical_error(const char *fmt, ...);
<*>\&{LABEL} { /* label reference */ <*>\&{LABEL} { /* label reference */
DPRINT("Ref: %s\n", yytext+1); DPRINT("Ref: %s\n", yytext+1);
yylval.labelref = xstrdup(yytext+1); yylval.labelref = xstrdup(yytext+1);
return DT_REF; return DT_LABEL_REF;
} }
<*>"&{/"{PATHCHAR}*\} { /* new-style path reference */ <*>"&{/"{PATHCHAR}*\} { /* new-style path reference */
yytext[yyleng-1] = '\0'; yytext[yyleng-1] = '\0';
DPRINT("Ref: %s\n", yytext+2); DPRINT("Ref: %s\n", yytext+2);
yylval.labelref = xstrdup(yytext+2); yylval.labelref = xstrdup(yytext+2);
return DT_REF; return DT_PATH_REF;
} }
<BYTESTRING>[0-9a-fA-F]{2} { <BYTESTRING>[0-9a-fA-F]{2} {

View File

@ -70,7 +70,8 @@ extern bool treesource_error;
%token <byte> DT_BYTE %token <byte> DT_BYTE
%token <data> DT_STRING %token <data> DT_STRING
%token <labelref> DT_LABEL %token <labelref> DT_LABEL
%token <labelref> DT_REF %token <labelref> DT_LABEL_REF
%token <labelref> DT_PATH_REF
%token DT_INCBIN %token DT_INCBIN
%type <data> propdata %type <data> propdata
@ -83,6 +84,7 @@ extern bool treesource_error;
%type <data> bytestring %type <data> bytestring
%type <prop> propdef %type <prop> propdef
%type <proplist> proplist %type <proplist> proplist
%type <labelref> dt_ref
%type <node> devicetree %type <node> devicetree
%type <node> nodedef %type <node> nodedef
@ -158,6 +160,8 @@ memreserve:
} }
; ;
dt_ref: DT_LABEL_REF | DT_PATH_REF;
devicetree: devicetree:
'/' nodedef '/' nodedef
{ {
@ -167,7 +171,7 @@ devicetree:
{ {
$$ = merge_nodes($1, $3); $$ = merge_nodes($1, $3);
} }
| DT_REF nodedef | dt_ref nodedef
{ {
/* /*
* We rely on the rule being always: * We rely on the rule being always:
@ -176,9 +180,12 @@ devicetree:
*/ */
if (!($<flags>-1 & DTSF_PLUGIN)) if (!($<flags>-1 & DTSF_PLUGIN))
ERROR(&@2, "Label or path %s not found", $1); ERROR(&@2, "Label or path %s not found", $1);
$$ = add_orphan_node(name_node(build_node(NULL, NULL), ""), $2, $1); $$ = add_orphan_node(
name_node(build_node(NULL, NULL, NULL),
""),
$2, $1);
} }
| devicetree DT_LABEL DT_REF nodedef | devicetree DT_LABEL dt_ref nodedef
{ {
struct node *target = get_node_by_ref($1, $3); struct node *target = get_node_by_ref($1, $3);
@ -189,7 +196,7 @@ devicetree:
ERROR(&@3, "Label or path %s not found", $3); ERROR(&@3, "Label or path %s not found", $3);
$$ = $1; $$ = $1;
} }
| devicetree DT_REF nodedef | devicetree DT_PATH_REF nodedef
{ {
/* /*
* We rely on the rule being always: * We rely on the rule being always:
@ -208,7 +215,26 @@ devicetree:
} }
$$ = $1; $$ = $1;
} }
| devicetree DT_DEL_NODE DT_REF ';' | devicetree DT_LABEL_REF nodedef
{
struct node *target = get_node_by_ref($1, $2);
if (target) {
merge_nodes(target, $3);
} else {
/*
* We rely on the rule being always:
* versioninfo plugindecl memreserves devicetree
* so $-1 is what we want (plugindecl)
*/
if ($<flags>-1 & DTSF_PLUGIN)
add_orphan_node($1, $3, $2);
else
ERROR(&@2, "Label or path %s not found", $2);
}
$$ = $1;
}
| devicetree DT_DEL_NODE dt_ref ';'
{ {
struct node *target = get_node_by_ref($1, $3); struct node *target = get_node_by_ref($1, $3);
@ -220,7 +246,7 @@ devicetree:
$$ = $1; $$ = $1;
} }
| devicetree DT_OMIT_NO_REF DT_REF ';' | devicetree DT_OMIT_NO_REF dt_ref ';'
{ {
struct node *target = get_node_by_ref($1, $3); struct node *target = get_node_by_ref($1, $3);
@ -237,7 +263,7 @@ devicetree:
nodedef: nodedef:
'{' proplist subnodes '}' ';' '{' proplist subnodes '}' ';'
{ {
$$ = build_node($2, $3); $$ = build_node($2, $3, &@$);
} }
; ;
@ -255,11 +281,11 @@ proplist:
propdef: propdef:
DT_PROPNODENAME '=' propdata ';' DT_PROPNODENAME '=' propdata ';'
{ {
$$ = build_property($1, $3); $$ = build_property($1, $3, &@$);
} }
| DT_PROPNODENAME ';' | DT_PROPNODENAME ';'
{ {
$$ = build_property($1, empty_data); $$ = build_property($1, empty_data, &@$);
} }
| DT_DEL_PROP DT_PROPNODENAME ';' | DT_DEL_PROP DT_PROPNODENAME ';'
{ {
@ -285,7 +311,7 @@ propdata:
{ {
$$ = data_merge($1, $3); $$ = data_merge($1, $3);
} }
| propdataprefix DT_REF | propdataprefix dt_ref
{ {
$1 = data_add_marker($1, TYPE_STRING, $2); $1 = data_add_marker($1, TYPE_STRING, $2);
$$ = data_add_marker($1, REF_PATH, $2); $$ = data_add_marker($1, REF_PATH, $2);
@ -383,7 +409,7 @@ arrayprefix:
$$.data = data_append_integer($1.data, $2, $1.bits); $$.data = data_append_integer($1.data, $2, $1.bits);
} }
| arrayprefix DT_REF | arrayprefix dt_ref
{ {
uint64_t val = ~0ULL >> (64 - $1.bits); uint64_t val = ~0ULL >> (64 - $1.bits);
@ -540,7 +566,7 @@ subnode:
} }
| DT_DEL_NODE DT_PROPNODENAME ';' | DT_DEL_NODE DT_PROPNODENAME ';'
{ {
$$ = name_node(build_node_delete(), $2); $$ = name_node(build_node_delete(&@$), $2);
} }
| DT_OMIT_NO_REF subnode | DT_OMIT_NO_REF subnode
{ {

View File

@ -35,6 +35,8 @@ int phandle_format = PHANDLE_EPAPR; /* Use linux,phandle or phandle properties *
int generate_symbols; /* enable symbols & fixup support */ int generate_symbols; /* enable symbols & fixup support */
int generate_fixups; /* suppress generation of fixups on symbol support */ int generate_fixups; /* suppress generation of fixups on symbol support */
int auto_label_aliases; /* auto generate labels -> aliases */ int auto_label_aliases; /* auto generate labels -> aliases */
int annotate; /* Level of annotation: 1 for input source location
>1 for full input source location. */
static int is_power_of_2(int x) static int is_power_of_2(int x)
{ {
@ -60,7 +62,7 @@ static void fill_fullpaths(struct node *tree, const char *prefix)
/* Usage related data. */ /* Usage related data. */
static const char usage_synopsis[] = "dtc [options] <input file>"; static const char usage_synopsis[] = "dtc [options] <input file>";
static const char usage_short_opts[] = "qI:O:o:V:d:R:S:p:a:fb:i:H:sW:E:@Ahv"; static const char usage_short_opts[] = "qI:O:o:V:d:R:S:p:a:fb:i:H:sW:E:@AThv";
static struct option const usage_long_opts[] = { static struct option const usage_long_opts[] = {
{"quiet", no_argument, NULL, 'q'}, {"quiet", no_argument, NULL, 'q'},
{"in-format", a_argument, NULL, 'I'}, {"in-format", a_argument, NULL, 'I'},
@ -81,6 +83,7 @@ static struct option const usage_long_opts[] = {
{"error", a_argument, NULL, 'E'}, {"error", a_argument, NULL, 'E'},
{"symbols", no_argument, NULL, '@'}, {"symbols", no_argument, NULL, '@'},
{"auto-alias", no_argument, NULL, 'A'}, {"auto-alias", no_argument, NULL, 'A'},
{"annotate", no_argument, NULL, 'T'},
{"help", no_argument, NULL, 'h'}, {"help", no_argument, NULL, 'h'},
{"version", no_argument, NULL, 'v'}, {"version", no_argument, NULL, 'v'},
{NULL, no_argument, NULL, 0x0}, {NULL, no_argument, NULL, 0x0},
@ -117,6 +120,7 @@ static const char * const usage_opts_help[] = {
"\n\tEnable/disable errors (prefix with \"no-\")", "\n\tEnable/disable errors (prefix with \"no-\")",
"\n\tEnable generation of symbols", "\n\tEnable generation of symbols",
"\n\tEnable auto-alias of labels", "\n\tEnable auto-alias of labels",
"\n\tAnnotate output .dts with input source file and line (-T -T for more details)",
"\n\tPrint this help and exit", "\n\tPrint this help and exit",
"\n\tPrint version and exit", "\n\tPrint version and exit",
NULL, NULL,
@ -264,6 +268,9 @@ int main(int argc, char *argv[])
case 'A': case 'A':
auto_label_aliases = 1; auto_label_aliases = 1;
break; break;
case 'T':
annotate++;
break;
case 'h': case 'h':
usage(NULL); usage(NULL);
@ -302,6 +309,8 @@ int main(int argc, char *argv[])
outform = "dts"; outform = "dts";
} }
} }
if (annotate && (!streq(inform, "dts") || !streq(outform, "dts")))
die("--annotate requires -I dts -O dts\n");
if (streq(inform, "dts")) if (streq(inform, "dts"))
dti = dt_from_source(arg); dti = dt_from_source(arg);
else if (streq(inform, "fs")) else if (streq(inform, "fs"))

View File

@ -58,6 +58,7 @@ extern int phandle_format; /* Use linux,phandle or phandle properties */
extern int generate_symbols; /* generate symbols for nodes with labels */ extern int generate_symbols; /* generate symbols for nodes with labels */
extern int generate_fixups; /* generate fixups */ extern int generate_fixups; /* generate fixups */
extern int auto_label_aliases; /* auto generate labels -> aliases */ extern int auto_label_aliases; /* auto generate labels -> aliases */
extern int annotate; /* annotate .dts with input source location */
#define PHANDLE_LEGACY 0x1 #define PHANDLE_LEGACY 0x1
#define PHANDLE_EPAPR 0x2 #define PHANDLE_EPAPR 0x2
@ -158,6 +159,7 @@ struct property {
struct property *next; struct property *next;
struct label *labels; struct label *labels;
struct srcpos *srcpos;
}; };
struct node { struct node {
@ -177,6 +179,7 @@ struct node {
struct label *labels; struct label *labels;
const struct bus_type *bus; const struct bus_type *bus;
struct srcpos *srcpos;
bool omit_if_unused, is_referenced; bool omit_if_unused, is_referenced;
}; };
@ -205,13 +208,15 @@ struct node {
void add_label(struct label **labels, char *label); void add_label(struct label **labels, char *label);
void delete_labels(struct label **labels); void delete_labels(struct label **labels);
struct property *build_property(char *name, struct data val); struct property *build_property(char *name, struct data val,
struct srcpos *srcpos);
struct property *build_property_delete(char *name); struct property *build_property_delete(char *name);
struct property *chain_property(struct property *first, struct property *list); struct property *chain_property(struct property *first, struct property *list);
struct property *reverse_properties(struct property *first); struct property *reverse_properties(struct property *first);
struct node *build_node(struct property *proplist, struct node *children); struct node *build_node(struct property *proplist, struct node *children,
struct node *build_node_delete(void); struct srcpos *srcpos);
struct node *build_node_delete(struct srcpos *srcpos);
struct node *name_node(struct node *node, char *name); struct node *name_node(struct node *node, char *name);
struct node *omit_node_if_unused(struct node *node); struct node *omit_node_if_unused(struct node *node);
struct node *reference_node(struct node *node); struct node *reference_node(struct node *node);

View File

@ -692,7 +692,7 @@ static struct property *flat_read_property(struct inbuf *dtbuf,
val = flat_read_data(dtbuf, proplen); val = flat_read_data(dtbuf, proplen);
return build_property(name, val); return build_property(name, val, NULL);
} }
@ -750,7 +750,7 @@ static struct node *unflatten_tree(struct inbuf *dtbuf,
char *flatname; char *flatname;
uint32_t val; uint32_t val;
node = build_node(NULL, NULL); node = build_node(NULL, NULL, NULL);
flatname = flat_read_string(dtbuf); flatname = flat_read_string(dtbuf);

View File

@ -34,7 +34,7 @@ static struct node *read_fstree(const char *dirname)
if (!d) if (!d)
die("Couldn't opendir() \"%s\": %s\n", dirname, strerror(errno)); die("Couldn't opendir() \"%s\": %s\n", dirname, strerror(errno));
tree = build_node(NULL, NULL); tree = build_node(NULL, NULL, NULL);
while ((de = readdir(d)) != NULL) { while ((de = readdir(d)) != NULL) {
char *tmpname; char *tmpname;
@ -60,7 +60,8 @@ static struct node *read_fstree(const char *dirname)
} else { } else {
prop = build_property(xstrdup(de->d_name), prop = build_property(xstrdup(de->d_name),
data_copy_file(pfile, data_copy_file(pfile,
st.st_size)); st.st_size),
NULL);
add_property(tree, prop); add_property(tree, prop);
fclose(pfile); fclose(pfile);
} }

View File

@ -9,3 +9,7 @@ LIBFDT_VERSION = version.lds
LIBFDT_SRCS = fdt.c fdt_ro.c fdt_wip.c fdt_sw.c fdt_rw.c fdt_strerror.c fdt_empty_tree.c \ LIBFDT_SRCS = fdt.c fdt_ro.c fdt_wip.c fdt_sw.c fdt_rw.c fdt_strerror.c fdt_empty_tree.c \
fdt_addresses.c fdt_overlay.c fdt_addresses.c fdt_overlay.c
LIBFDT_OBJS = $(LIBFDT_SRCS:%.c=%.o) LIBFDT_OBJS = $(LIBFDT_SRCS:%.c=%.o)
libfdt_clean:
@$(VECHO) CLEAN "(libfdt)"
rm -f $(STD_CLEANFILES:%=$(LIBFDT_dir)/%)

View File

@ -64,7 +64,7 @@ static int fdt_cells(const void *fdt, int nodeoffset, const char *name)
c = fdt_getprop(fdt, nodeoffset, name, &len); c = fdt_getprop(fdt, nodeoffset, name, &len);
if (!c) if (!c)
return 2; return len;
if (len != sizeof(*c)) if (len != sizeof(*c))
return -FDT_ERR_BADNCELLS; return -FDT_ERR_BADNCELLS;
@ -78,10 +78,20 @@ static int fdt_cells(const void *fdt, int nodeoffset, const char *name)
int fdt_address_cells(const void *fdt, int nodeoffset) int fdt_address_cells(const void *fdt, int nodeoffset)
{ {
return fdt_cells(fdt, nodeoffset, "#address-cells"); int val;
val = fdt_cells(fdt, nodeoffset, "#address-cells");
if (val == -FDT_ERR_NOTFOUND)
return 2;
return val;
} }
int fdt_size_cells(const void *fdt, int nodeoffset) int fdt_size_cells(const void *fdt, int nodeoffset)
{ {
return fdt_cells(fdt, nodeoffset, "#size-cells"); int val;
val = fdt_cells(fdt, nodeoffset, "#size-cells");
if (val == -FDT_ERR_NOTFOUND)
return 1;
return val;
} }

View File

@ -163,18 +163,26 @@ uint32_t fdt_next_tag(const void *fdt, int offset, int *nextoffset);
static inline uint32_t fdt32_ld(const fdt32_t *p) static inline uint32_t fdt32_ld(const fdt32_t *p)
{ {
fdt32_t v; const uint8_t *bp = (const uint8_t *)p;
memcpy(&v, p, sizeof(v)); return ((uint32_t)bp[0] << 24)
return fdt32_to_cpu(v); | ((uint32_t)bp[1] << 16)
| ((uint32_t)bp[2] << 8)
| bp[3];
} }
static inline uint64_t fdt64_ld(const fdt64_t *p) static inline uint64_t fdt64_ld(const fdt64_t *p)
{ {
fdt64_t v; const uint8_t *bp = (const uint8_t *)p;
memcpy(&v, p, sizeof(v)); return ((uint64_t)bp[0] << 56)
return fdt64_to_cpu(v); | ((uint64_t)bp[1] << 48)
| ((uint64_t)bp[2] << 40)
| ((uint64_t)bp[3] << 32)
| ((uint64_t)bp[4] << 24)
| ((uint64_t)bp[5] << 16)
| ((uint64_t)bp[6] << 8)
| bp[7];
} }
/**********************************************************************/ /**********************************************************************/
@ -1145,7 +1153,7 @@ int fdt_address_cells(const void *fdt, int nodeoffset);
* *
* returns: * returns:
* 0 <= n < FDT_MAX_NCELLS, on success * 0 <= n < FDT_MAX_NCELLS, on success
* 2, if the node has no #size-cells property * 1, if the node has no #size-cells property
* -FDT_ERR_BADNCELLS, if the node has a badly formatted or invalid * -FDT_ERR_BADNCELLS, if the node has a badly formatted or invalid
* #size-cells property * #size-cells property
* -FDT_ERR_BADMAGIC, * -FDT_ERR_BADMAGIC,

View File

@ -19,6 +19,7 @@
*/ */
#include "dtc.h" #include "dtc.h"
#include "srcpos.h"
/* /*
* Tree building functions * Tree building functions
@ -50,7 +51,8 @@ void delete_labels(struct label **labels)
label->deleted = 1; label->deleted = 1;
} }
struct property *build_property(char *name, struct data val) struct property *build_property(char *name, struct data val,
struct srcpos *srcpos)
{ {
struct property *new = xmalloc(sizeof(*new)); struct property *new = xmalloc(sizeof(*new));
@ -58,6 +60,7 @@ struct property *build_property(char *name, struct data val)
new->name = name; new->name = name;
new->val = val; new->val = val;
new->srcpos = srcpos_copy(srcpos);
return new; return new;
} }
@ -97,7 +100,8 @@ struct property *reverse_properties(struct property *first)
return head; return head;
} }
struct node *build_node(struct property *proplist, struct node *children) struct node *build_node(struct property *proplist, struct node *children,
struct srcpos *srcpos)
{ {
struct node *new = xmalloc(sizeof(*new)); struct node *new = xmalloc(sizeof(*new));
struct node *child; struct node *child;
@ -106,6 +110,7 @@ struct node *build_node(struct property *proplist, struct node *children)
new->proplist = reverse_properties(proplist); new->proplist = reverse_properties(proplist);
new->children = children; new->children = children;
new->srcpos = srcpos_copy(srcpos);
for_each_child(new, child) { for_each_child(new, child) {
child->parent = new; child->parent = new;
@ -114,13 +119,14 @@ struct node *build_node(struct property *proplist, struct node *children)
return new; return new;
} }
struct node *build_node_delete(void) struct node *build_node_delete(struct srcpos *srcpos)
{ {
struct node *new = xmalloc(sizeof(*new)); struct node *new = xmalloc(sizeof(*new));
memset(new, 0, sizeof(*new)); memset(new, 0, sizeof(*new));
new->deleted = 1; new->deleted = 1;
new->srcpos = srcpos_copy(srcpos);
return new; return new;
} }
@ -183,6 +189,8 @@ struct node *merge_nodes(struct node *old_node, struct node *new_node)
old_prop->val = new_prop->val; old_prop->val = new_prop->val;
old_prop->deleted = 0; old_prop->deleted = 0;
free(old_prop->srcpos);
old_prop->srcpos = new_prop->srcpos;
free(new_prop); free(new_prop);
new_prop = NULL; new_prop = NULL;
break; break;
@ -223,6 +231,8 @@ struct node *merge_nodes(struct node *old_node, struct node *new_node)
add_child(old_node, new_child); add_child(old_node, new_child);
} }
old_node->srcpos = srcpos_extend(old_node->srcpos, new_node->srcpos);
/* The new node contents are now merged into the old node. Free /* The new node contents are now merged into the old node. Free
* the new node. */ * the new node. */
free(new_node); free(new_node);
@ -241,18 +251,18 @@ struct node * add_orphan_node(struct node *dt, struct node *new_node, char *ref)
if (ref[0] == '/') { if (ref[0] == '/') {
d = data_append_data(d, ref, strlen(ref) + 1); d = data_append_data(d, ref, strlen(ref) + 1);
p = build_property("target-path", d); p = build_property("target-path", d, NULL);
} else { } else {
d = data_add_marker(d, REF_PHANDLE, ref); d = data_add_marker(d, REF_PHANDLE, ref);
d = data_append_integer(d, 0xffffffff, 32); d = data_append_integer(d, 0xffffffff, 32);
p = build_property("target", d); p = build_property("target", d, NULL);
} }
xasprintf(&name, "fragment@%u", xasprintf(&name, "fragment@%u",
next_orphan_fragment++); next_orphan_fragment++);
name_node(new_node, "__overlay__"); name_node(new_node, "__overlay__");
node = build_node(p, new_node); node = build_node(p, new_node, NULL);
name_node(node, name); name_node(node, name);
add_child(dt, node); add_child(dt, node);
@ -351,7 +361,7 @@ void append_to_property(struct node *node,
p->val = d; p->val = d;
} else { } else {
d = data_append_data(empty_data, data, len); d = data_append_data(empty_data, data, len);
p = build_property(name, d); p = build_property(name, d, NULL);
add_property(node, p); add_property(node, p);
} }
} }
@ -609,11 +619,11 @@ cell_t get_node_phandle(struct node *root, struct node *node)
if (!get_property(node, "linux,phandle") if (!get_property(node, "linux,phandle")
&& (phandle_format & PHANDLE_LEGACY)) && (phandle_format & PHANDLE_LEGACY))
add_property(node, build_property("linux,phandle", d)); add_property(node, build_property("linux,phandle", d, NULL));
if (!get_property(node, "phandle") if (!get_property(node, "phandle")
&& (phandle_format & PHANDLE_EPAPR)) && (phandle_format & PHANDLE_EPAPR))
add_property(node, build_property("phandle", d)); add_property(node, build_property("phandle", d, NULL));
/* If the node *does* have a phandle property, we must /* If the node *does* have a phandle property, we must
* be dealing with a self-referencing phandle, which will be * be dealing with a self-referencing phandle, which will be
@ -787,7 +797,7 @@ static struct node *build_and_name_child_node(struct node *parent, char *name)
{ {
struct node *node; struct node *node;
node = build_node(NULL, NULL); node = build_node(NULL, NULL, NULL);
name_node(node, xstrdup(name)); name_node(node, xstrdup(name));
add_child(parent, node); add_child(parent, node);
@ -849,7 +859,8 @@ static void generate_label_tree_internal(struct dt_info *dti,
/* insert it */ /* insert it */
p = build_property(l->label, p = build_property(l->label,
data_copy_mem(node->fullpath, data_copy_mem(node->fullpath,
strlen(node->fullpath) + 1)); strlen(node->fullpath) + 1),
NULL);
add_property(an, p); add_property(an, p);
} }

View File

@ -33,6 +33,9 @@ struct search_path {
/* This is the list of directories that we search for source files */ /* This is the list of directories that we search for source files */
static struct search_path *search_path_head, **search_path_tail; static struct search_path *search_path_head, **search_path_tail;
/* Detect infinite include recursion. */
#define MAX_SRCFILE_DEPTH (100)
static int srcfile_depth; /* = 0 */
static char *get_dirname(const char *path) static char *get_dirname(const char *path)
{ {
@ -51,11 +54,51 @@ static char *get_dirname(const char *path)
FILE *depfile; /* = NULL */ FILE *depfile; /* = NULL */
struct srcfile_state *current_srcfile; /* = NULL */ struct srcfile_state *current_srcfile; /* = NULL */
static char *initial_path; /* = NULL */
static int initial_pathlen; /* = 0 */
static bool initial_cpp = true;
/* Detect infinite include recursion. */ static void set_initial_path(char *fname)
#define MAX_SRCFILE_DEPTH (100) {
static int srcfile_depth; /* = 0 */ int i, len = strlen(fname);
xasprintf(&initial_path, "%s", fname);
initial_pathlen = 0;
for (i = 0; i != len; i++)
if (initial_path[i] == '/')
initial_pathlen++;
}
static char *shorten_to_initial_path(char *fname)
{
char *p1, *p2, *prevslash1 = NULL;
int slashes = 0;
for (p1 = fname, p2 = initial_path; *p1 && *p2; p1++, p2++) {
if (*p1 != *p2)
break;
if (*p1 == '/') {
prevslash1 = p1;
slashes++;
}
}
p1 = prevslash1 + 1;
if (prevslash1) {
int diff = initial_pathlen - slashes, i, j;
int restlen = strlen(fname) - (p1 - fname);
char *res;
res = xmalloc((3 * diff) + restlen + 1);
for (i = 0, j = 0; i != diff; i++) {
res[j++] = '.';
res[j++] = '.';
res[j++] = '/';
}
strcpy(res + j, p1);
return res;
}
return NULL;
}
/** /**
* Try to open a file in a given directory. * Try to open a file in a given directory.
@ -157,6 +200,9 @@ void srcfile_push(const char *fname)
srcfile->colno = 1; srcfile->colno = 1;
current_srcfile = srcfile; current_srcfile = srcfile;
if (srcfile_depth == 1)
set_initial_path(srcfile->name);
} }
bool srcfile_pop(void) bool srcfile_pop(void)
@ -197,18 +243,6 @@ void srcfile_add_search_path(const char *dirname)
search_path_tail = &node->next; search_path_tail = &node->next;
} }
/*
* The empty source position.
*/
struct srcpos srcpos_empty = {
.first_line = 0,
.first_column = 0,
.last_line = 0,
.last_column = 0,
.file = NULL,
};
void srcpos_update(struct srcpos *pos, const char *text, int len) void srcpos_update(struct srcpos *pos, const char *text, int len)
{ {
int i; int i;
@ -234,13 +268,35 @@ struct srcpos *
srcpos_copy(struct srcpos *pos) srcpos_copy(struct srcpos *pos)
{ {
struct srcpos *pos_new; struct srcpos *pos_new;
struct srcfile_state *srcfile_state;
if (!pos)
return NULL;
pos_new = xmalloc(sizeof(struct srcpos)); pos_new = xmalloc(sizeof(struct srcpos));
assert(pos->next == NULL);
memcpy(pos_new, pos, sizeof(struct srcpos)); memcpy(pos_new, pos, sizeof(struct srcpos));
/* allocate without free */
srcfile_state = xmalloc(sizeof(struct srcfile_state));
memcpy(srcfile_state, pos->file, sizeof(struct srcfile_state));
pos_new->file = srcfile_state;
return pos_new; return pos_new;
} }
struct srcpos *srcpos_extend(struct srcpos *pos, struct srcpos *newtail)
{
struct srcpos *p;
if (!pos)
return newtail;
for (p = pos; p->next != NULL; p = p->next);
p->next = newtail;
return pos;
}
char * char *
srcpos_string(struct srcpos *pos) srcpos_string(struct srcpos *pos)
{ {
@ -266,6 +322,68 @@ srcpos_string(struct srcpos *pos)
return pos_str; return pos_str;
} }
static char *
srcpos_string_comment(struct srcpos *pos, bool first_line, int level)
{
char *pos_str, *fname, *first, *rest;
bool fresh_fname = false;
if (!pos) {
if (level > 1) {
xasprintf(&pos_str, "<no-file>:<no-line>");
return pos_str;
} else {
return NULL;
}
}
if (!pos->file)
fname = "<no-file>";
else if (!pos->file->name)
fname = "<no-filename>";
else if (level > 1)
fname = pos->file->name;
else {
fname = shorten_to_initial_path(pos->file->name);
if (fname)
fresh_fname = true;
else
fname = pos->file->name;
}
if (level > 1)
xasprintf(&first, "%s:%d:%d-%d:%d", fname,
pos->first_line, pos->first_column,
pos->last_line, pos->last_column);
else
xasprintf(&first, "%s:%d", fname,
first_line ? pos->first_line : pos->last_line);
if (fresh_fname)
free(fname);
if (pos->next != NULL) {
rest = srcpos_string_comment(pos->next, first_line, level);
xasprintf(&pos_str, "%s, %s", first, rest);
free(first);
free(rest);
} else {
pos_str = first;
}
return pos_str;
}
char *srcpos_string_first(struct srcpos *pos, int level)
{
return srcpos_string_comment(pos, true, level);
}
char *srcpos_string_last(struct srcpos *pos, int level)
{
return srcpos_string_comment(pos, false, level);
}
void srcpos_verror(struct srcpos *pos, const char *prefix, void srcpos_verror(struct srcpos *pos, const char *prefix,
const char *fmt, va_list va) const char *fmt, va_list va)
{ {
@ -294,4 +412,9 @@ void srcpos_set_line(char *f, int l)
{ {
current_srcfile->name = f; current_srcfile->name = f;
current_srcfile->lineno = l; current_srcfile->lineno = l;
if (initial_cpp) {
initial_cpp = false;
set_initial_path(f);
}
} }

View File

@ -74,6 +74,7 @@ struct srcpos {
int last_line; int last_line;
int last_column; int last_column;
struct srcfile_state *file; struct srcfile_state *file;
struct srcpos *next;
}; };
#define YYLTYPE struct srcpos #define YYLTYPE struct srcpos
@ -93,19 +94,18 @@ struct srcpos {
YYRHSLOC(Rhs, 0).last_column; \ YYRHSLOC(Rhs, 0).last_column; \
(Current).file = YYRHSLOC (Rhs, 0).file; \ (Current).file = YYRHSLOC (Rhs, 0).file; \
} \ } \
(Current).next = NULL; \
} while (0) } while (0)
/*
* Fictional source position used for IR nodes that are
* created without otherwise knowing a true source position.
* For example,constant definitions from the command line.
*/
extern struct srcpos srcpos_empty;
extern void srcpos_update(struct srcpos *pos, const char *text, int len); extern void srcpos_update(struct srcpos *pos, const char *text, int len);
extern struct srcpos *srcpos_copy(struct srcpos *pos); extern struct srcpos *srcpos_copy(struct srcpos *pos);
extern struct srcpos *srcpos_extend(struct srcpos *new_srcpos,
struct srcpos *old_srcpos);
extern char *srcpos_string(struct srcpos *pos); extern char *srcpos_string(struct srcpos *pos);
extern char *srcpos_string_first(struct srcpos *pos, int level);
extern char *srcpos_string_last(struct srcpos *pos, int level);
extern void PRINTF(3, 0) srcpos_verror(struct srcpos *pos, const char *prefix, extern void PRINTF(3, 0) srcpos_verror(struct srcpos *pos, const char *prefix,
const char *fmt, va_list va); const char *fmt, va_list va);

View File

@ -64,6 +64,10 @@ static bool isstring(char c)
static void write_propval_string(FILE *f, const char *s, size_t len) static void write_propval_string(FILE *f, const char *s, size_t len)
{ {
const char *end = s + len - 1; const char *end = s + len - 1;
if (!len)
return;
assert(*end == '\0'); assert(*end == '\0');
fprintf(f, "\""); fprintf(f, "\"");
@ -118,18 +122,20 @@ static void write_propval_int(FILE *f, const char *p, size_t len, size_t width)
for (; p < end; p += width) { for (; p < end; p += width) {
switch (width) { switch (width) {
case 1: case 1:
fprintf(f, " %02"PRIx8, *(const uint8_t*)p); fprintf(f, "%02"PRIx8, *(const uint8_t*)p);
break; break;
case 2: case 2:
fprintf(f, " 0x%02"PRIx16, fdt16_to_cpu(*(const fdt16_t*)p)); fprintf(f, "0x%02"PRIx16, fdt16_to_cpu(*(const fdt16_t*)p));
break; break;
case 4: case 4:
fprintf(f, " 0x%02"PRIx32, fdt32_to_cpu(*(const fdt32_t*)p)); fprintf(f, "0x%02"PRIx32, fdt32_to_cpu(*(const fdt32_t*)p));
break; break;
case 8: case 8:
fprintf(f, " 0x%02"PRIx64, fdt64_to_cpu(*(const fdt64_t*)p)); fprintf(f, "0x%02"PRIx64, fdt64_to_cpu(*(const fdt64_t*)p));
break; break;
} }
if (p + width < end)
fputc(' ', f);
} }
} }
@ -162,10 +168,10 @@ static const char *delim_start[] = {
[TYPE_STRING] = "", [TYPE_STRING] = "",
}; };
static const char *delim_end[] = { static const char *delim_end[] = {
[TYPE_UINT8] = " ]", [TYPE_UINT8] = "]",
[TYPE_UINT16] = " >", [TYPE_UINT16] = ">",
[TYPE_UINT32] = " >", [TYPE_UINT32] = ">",
[TYPE_UINT64] = " >", [TYPE_UINT64] = ">",
[TYPE_STRING] = "", [TYPE_STRING] = "",
}; };
@ -208,13 +214,22 @@ static void write_propval(FILE *f, struct property *prop)
struct marker *m = prop->val.markers; struct marker *m = prop->val.markers;
struct marker dummy_marker; struct marker dummy_marker;
enum markertype emit_type = TYPE_NONE; enum markertype emit_type = TYPE_NONE;
char *srcstr;
if (len == 0) { if (len == 0) {
fprintf(f, ";\n"); fprintf(f, ";");
if (annotate) {
srcstr = srcpos_string_first(prop->srcpos, annotate);
if (srcstr) {
fprintf(f, " /* %s */", srcstr);
free(srcstr);
}
}
fprintf(f, "\n");
return; return;
} }
fprintf(f, " = "); fprintf(f, " =");
if (!next_type_marker(m)) { if (!next_type_marker(m)) {
/* data type information missing, need to guess */ /* data type information missing, need to guess */
@ -225,33 +240,24 @@ static void write_propval(FILE *f, struct property *prop)
m = &dummy_marker; m = &dummy_marker;
} }
struct marker *m_label = prop->val.markers;
for_each_marker(m) { for_each_marker(m) {
size_t chunk_len; size_t chunk_len = (m->next ? m->next->offset : len) - m->offset;
size_t data_len = type_marker_length(m) ? : len - m->offset;
const char *p = &prop->val.val[m->offset]; const char *p = &prop->val.val[m->offset];
if (!has_data_type_information(m)) if (has_data_type_information(m)) {
emit_type = m->type;
fprintf(f, " %s", delim_start[emit_type]);
} else if (m->type == LABEL)
fprintf(f, " %s:", m->ref);
else if (m->offset)
fputc(' ', f);
if (emit_type == TYPE_NONE) {
assert(chunk_len == 0);
continue; continue;
chunk_len = type_marker_length(m);
if (!chunk_len)
chunk_len = len - m->offset;
if (emit_type != TYPE_NONE)
fprintf(f, "%s, ", delim_end[emit_type]);
emit_type = m->type;
for_each_marker_of_type(m_label, LABEL) {
if (m_label->offset > m->offset)
break;
fprintf(f, "%s: ", m_label->ref);
} }
fprintf(f, "%s", delim_start[emit_type]);
if (chunk_len <= 0)
continue;
switch(emit_type) { switch(emit_type) {
case TYPE_UINT16: case TYPE_UINT16:
write_propval_int(f, p, chunk_len, 2); write_propval_int(f, p, chunk_len, 2);
@ -268,15 +274,23 @@ static void write_propval(FILE *f, struct property *prop)
default: default:
write_propval_int(f, p, chunk_len, 1); write_propval_int(f, p, chunk_len, 1);
} }
}
/* Wrap up any labels at the end of the value */ if (chunk_len == data_len) {
for_each_marker_of_type(m_label, LABEL) { size_t pos = m->offset + chunk_len;
assert (m_label->offset == len); fprintf(f, pos == len ? "%s" : "%s,",
fprintf(f, " %s:", m_label->ref); delim_end[emit_type] ? : "");
emit_type = TYPE_NONE;
}
} }
fprintf(f, ";");
fprintf(f, "%s;\n", delim_end[emit_type] ? : ""); if (annotate) {
srcstr = srcpos_string_first(prop->srcpos, annotate);
if (srcstr) {
fprintf(f, " /* %s */", srcstr);
free(srcstr);
}
}
fprintf(f, "\n");
} }
static void write_tree_source_node(FILE *f, struct node *tree, int level) static void write_tree_source_node(FILE *f, struct node *tree, int level)
@ -284,14 +298,24 @@ static void write_tree_source_node(FILE *f, struct node *tree, int level)
struct property *prop; struct property *prop;
struct node *child; struct node *child;
struct label *l; struct label *l;
char *srcstr;
write_prefix(f, level); write_prefix(f, level);
for_each_label(tree->labels, l) for_each_label(tree->labels, l)
fprintf(f, "%s: ", l->label); fprintf(f, "%s: ", l->label);
if (tree->name && (*tree->name)) if (tree->name && (*tree->name))
fprintf(f, "%s {\n", tree->name); fprintf(f, "%s {", tree->name);
else else
fprintf(f, "/ {\n"); fprintf(f, "/ {");
if (annotate) {
srcstr = srcpos_string_first(tree->srcpos, annotate);
if (srcstr) {
fprintf(f, " /* %s */", srcstr);
free(srcstr);
}
}
fprintf(f, "\n");
for_each_property(tree, prop) { for_each_property(tree, prop) {
write_prefix(f, level+1); write_prefix(f, level+1);
@ -305,10 +329,17 @@ static void write_tree_source_node(FILE *f, struct node *tree, int level)
write_tree_source_node(f, child, level+1); write_tree_source_node(f, child, level+1);
} }
write_prefix(f, level); write_prefix(f, level);
fprintf(f, "};\n"); fprintf(f, "};");
if (annotate) {
srcstr = srcpos_string_last(tree->srcpos, annotate);
if (srcstr) {
fprintf(f, " /* %s */", srcstr);
free(srcstr);
}
}
fprintf(f, "\n");
} }
void dt_to_source(FILE *f, struct dt_info *dti) void dt_to_source(FILE *f, struct dt_info *dti)
{ {
struct reserve_info *re; struct reserve_info *re;

View File

@ -46,36 +46,54 @@ char *xstrdup(const char *s)
return d; return d;
} }
/* based in part from (3) vsnprintf */ int xavsprintf_append(char **strp, const char *fmt, va_list ap)
int xasprintf(char **strp, const char *fmt, ...)
{ {
int n, size = 128; /* start with 128 bytes */ int n, size = 0; /* start with 128 bytes */
char *p; char *p;
va_list ap; va_list ap_copy;
/* initial pointer is NULL making the fist realloc to be malloc */ p = *strp;
p = NULL; if (p)
while (1) { size = strlen(p);
p = xrealloc(p, size);
/* Try to print in the allocated space. */ va_copy(ap_copy, ap);
va_start(ap, fmt); n = vsnprintf(NULL, 0, fmt, ap_copy) + 1;
n = vsnprintf(p, size, fmt, ap); va_end(ap_copy);
va_end(ap);
p = xrealloc(p, size + n);
n = vsnprintf(p + size, n, fmt, ap);
/* If that worked, return the string. */
if (n > -1 && n < size)
break;
/* Else try again with more space. */
if (n > -1) /* glibc 2.1 */
size = n + 1; /* precisely what is needed */
else /* glibc 2.0 */
size *= 2; /* twice the old size */
}
*strp = p; *strp = p;
return strlen(p); return strlen(p);
} }
int xasprintf_append(char **strp, const char *fmt, ...)
{
int n;
va_list ap;
va_start(ap, fmt);
n = xavsprintf_append(strp, fmt, ap);
va_end(ap);
return n;
}
int xasprintf(char **strp, const char *fmt, ...)
{
int n;
va_list ap;
*strp = NULL;
va_start(ap, fmt);
n = xavsprintf_append(strp, fmt, ap);
va_end(ap);
return n;
}
char *join_path(const char *path, const char *name) char *join_path(const char *path, const char *name)
{ {
int lenp = strlen(path); int lenp = strlen(path);

View File

@ -72,6 +72,8 @@ static inline void *xrealloc(void *p, size_t len)
extern char *xstrdup(const char *s); extern char *xstrdup(const char *s);
extern int PRINTF(2, 3) xasprintf(char **strp, const char *fmt, ...); extern int PRINTF(2, 3) xasprintf(char **strp, const char *fmt, ...);
extern int PRINTF(2, 3) xasprintf_append(char **strp, const char *fmt, ...);
extern int xavsprintf_append(char **strp, const char *fmt, va_list ap);
extern char *join_path(const char *path, const char *name); extern char *join_path(const char *path, const char *name);
/** /**

View File

@ -1 +1 @@
#define DTC_VERSION "DTC 1.4.7-gc86da84d" #define DTC_VERSION "DTC 1.4.7-gf267e674"