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:
parent
050e6f0cff
commit
e37ec7d588
8 changed files with 145 additions and 6 deletions
25
data.c
25
data.c
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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 */
|
||||||
|
|
||||||
<*>"/*"([^*]|\*+[^*/])*\*+"/" {
|
<*>"/*"([^*]|\*+[^*/])*\*+"/" {
|
||||||
|
|
31
dtc-parser.y
31
dtc-parser.y
|
@ -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);
|
||||||
|
|
|
@ -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
1
tests/incbin.bin
Normal file
|
@ -0,0 +1 @@
|
||||||
|
abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
|
75
tests/incbin.c
Normal file
75
tests/incbin.c
Normal 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
6
tests/incbin.dts
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
/dts-v1/;
|
||||||
|
|
||||||
|
/ {
|
||||||
|
incbin = /incbin/("incbin.bin");
|
||||||
|
incbin-partial = /incbin/("incbin.bin", 13, 17);
|
||||||
|
};
|
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue