dtc: Add support for binary includes.

On Wed, Jun 04, 2008 at 09:26:23AM -0500, Jon Loeliger wrote:
> David Gibson wrote:
>
>> But as I said that can be dealt with in the future without breaking
>> compatibility.  Objection withdrawn.
>>
>
> And on that note, I officially implore Scott to
> re-submit his binary include patch!

Scott's original patch does still have some implementation details I
didn't like.  So in the interests of saving time, I've addressed some
of those, added a testcase, and and now resubmitting my revised
version of Scott's patch.

dtc: Add support for binary includes.

A property's data can be populated with a file's contents
as follows:

node {
	prop = /incbin/("path/to/data");
};

A subset of a file can be included by passing start and size parameters.
For example, to include bytes 8 through 23:

node {
	prop = /incbin/("path/to/data", 8, 16);
};

As with /include/, non-absolute paths are looked for in the directory
of the source file that includes them.

Implementation revised, and a testcase added by David Gibson

Signed-off-by: Scott Wood <scottwood@freescale.com>
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Acked-by: Scott Wood <scottwood@freescale.com>
This commit is contained in:
David Gibson 2008-06-11 11:58:39 +10:00 committed by Jon Loeliger
parent 050e6f0cff
commit e37ec7d588
8 changed files with 145 additions and 6 deletions

25
data.c
View file

@ -167,14 +167,29 @@ struct data data_copy_escape_string(const char *s, int len)
return d; return d;
} }
struct data data_copy_file(FILE *f, size_t len) struct data data_copy_file(FILE *f, size_t maxlen)
{ {
struct data d; struct data d = empty_data;
d = data_grow_for(empty_data, len); while (!feof(f) && (d.len < maxlen)) {
size_t chunksize, ret;
d.len = len; if (maxlen == -1)
fread(d.val, len, 1, f); chunksize = 4096;
else
chunksize = maxlen - d.len;
d = data_grow_for(d, chunksize);
ret = fread(d.val + d.len, 1, chunksize, f);
if (ferror(f))
die("Error reading file into data: %s", strerror(errno));
if (d.len + ret < d.len)
die("Overflow reading file into data\n");
d.len += ret;
}
return d; return d;
} }

View file

@ -190,6 +190,13 @@ static int pop_input_file(void);
return DT_PROPNODENAME; return DT_PROPNODENAME;
} }
"/incbin/" {
yylloc.file = srcpos_file;
yylloc.first_line = yylineno;
DPRINT("Binary Include\n");
return DT_INCBIN;
}
<*>[[:space:]]+ /* eat whitespace */ <*>[[:space:]]+ /* eat whitespace */
<*>"/*"([^*]|\*+[^*/])*\*+"/" { <*>"/*"([^*]|\*+[^*/])*\*+"/" {

View file

@ -21,6 +21,8 @@
%locations %locations
%{ %{
#include <stdio.h>
#include "dtc.h" #include "dtc.h"
#include "srcpos.h" #include "srcpos.h"
@ -59,6 +61,7 @@ static unsigned long long eval_literal(const char *s, int base, int bits);
%token <data> DT_STRING %token <data> DT_STRING
%token <labelref> DT_LABEL %token <labelref> DT_LABEL
%token <labelref> DT_REF %token <labelref> DT_REF
%token DT_INCBIN
%type <data> propdata %type <data> propdata
%type <data> propdataprefix %type <data> propdataprefix
@ -197,6 +200,34 @@ propdata:
{ {
$$ = data_add_marker($1, REF_PATH, $2); $$ = data_add_marker($1, REF_PATH, $2);
} }
| propdataprefix DT_INCBIN '(' DT_STRING ',' addr ',' addr ')'
{
struct search_path path = { srcpos_file->dir, NULL, NULL };
struct dtc_file *file = dtc_open_file($4.val, &path);
struct data d = empty_data;
if ($6 != 0)
if (fseek(file->file, $6, SEEK_SET) != 0)
yyerrorf("Couldn't seek to offset %llu in \"%s\": %s",
(unsigned long long)$6,
$4.val, strerror(errno));
d = data_copy_file(file->file, $8);
$$ = data_merge($1, d);
dtc_close_file(file);
}
| propdataprefix DT_INCBIN '(' DT_STRING ')'
{
struct search_path path = { srcpos_file->dir, NULL, NULL };
struct dtc_file *file = dtc_open_file($4.val, &path);
struct data d = empty_data;
d = data_copy_file(file->file, -1);
$$ = data_merge($1, d);
dtc_close_file(file);
}
| propdata DT_LABEL | propdata DT_LABEL
{ {
$$ = data_add_marker($1, LABEL, $2); $$ = data_add_marker($1, LABEL, $2);

View file

@ -9,7 +9,7 @@ LIB_TESTS_L = get_mem_rsv \
sw_tree1 \ sw_tree1 \
move_and_save mangle-layout nopulate \ move_and_save mangle-layout nopulate \
open_pack rw_tree1 set_name setprop del_property del_node \ open_pack rw_tree1 set_name setprop del_property del_node \
string_escapes references path-references boot-cpuid \ string_escapes references path-references boot-cpuid incbin \
dtbs_equal_ordered \ dtbs_equal_ordered \
add_subnode_with_nops add_subnode_with_nops
LIB_TESTS = $(LIB_TESTS_L:%=$(TESTS_PREFIX)%) LIB_TESTS = $(LIB_TESTS_L:%=$(TESTS_PREFIX)%)

1
tests/incbin.bin Normal file
View file

@ -0,0 +1 @@
abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ

75
tests/incbin.c Normal file
View file

@ -0,0 +1,75 @@
/*
* libfdt - Flat Device Tree manipulation
* Testcase for string escapes in dtc
* Copyright (C) 2006 David Gibson, IBM Corporation.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <stdint.h>
#include <errno.h>
#include <fdt.h>
#include <libfdt.h>
#include "tests.h"
#include "testdata.h"
#define CHUNKSIZE 1024
void *load_file(const char *name, int *len)
{
FILE *f;
void *buf = NULL;
int bufsize = 0, n;
*len = 0;
f = fopen(name, "r");
if (!f)
FAIL("Couldn't open \"%s\": %s", name, strerror(errno));
while (!feof(f)) {
if (bufsize < (*len + CHUNKSIZE)) {
buf = xrealloc(buf, *len + CHUNKSIZE);
bufsize = *len + CHUNKSIZE;
}
n = fread(buf + *len, 1, CHUNKSIZE, f);
if (ferror(f))
FAIL("Error reading \"%s\": %s", name, strerror(errno));
*len += n;
}
return buf;
}
int main(int argc, char *argv[])
{
void *fdt, *incbin;
int len;
test_init(argc, argv);
incbin = load_file("incbin.bin", &len);
fdt = load_blob_arg(argc, argv);
check_getprop(fdt, 0, "incbin", len, incbin);
check_getprop(fdt, 0, "incbin-partial", 17, incbin + 13);
PASS();
}

6
tests/incbin.dts Normal file
View file

@ -0,0 +1,6 @@
/dts-v1/;
/ {
incbin = /incbin/("incbin.bin");
incbin-partial = /incbin/("incbin.bin", 13, 17);
};

View file

@ -206,6 +206,10 @@ dtc_tests () {
run_dtc_test -I dts -O dtb -o dtc_comments-cmp.test.dtb comments-cmp.dts run_dtc_test -I dts -O dtb -o dtc_comments-cmp.test.dtb comments-cmp.dts
run_test dtbs_equal_ordered dtc_comments.test.dtb dtc_comments-cmp.test.dtb run_test dtbs_equal_ordered dtc_comments.test.dtb dtc_comments-cmp.test.dtb
# Check /incbin/ directive
run_dtc_test -I dts -O dtb -o incbin.test.dtb incbin.dts
run_test incbin incbin.test.dtb
# Check boot_cpuid_phys handling # Check boot_cpuid_phys handling
run_dtc_test -I dts -O dtb -b 17 -o boot_cpuid.test.dtb empty.dts run_dtc_test -I dts -O dtb -b 17 -o boot_cpuid.test.dtb empty.dts
run_test boot-cpuid boot_cpuid.test.dtb 17 run_test boot-cpuid boot_cpuid.test.dtb 17