You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
379 lines
6.4 KiB
379 lines
6.4 KiB
/*
|
|
* (C) Copyright David Gibson <dwg@au1.ibm.com>, IBM Corporation. 2005.
|
|
*
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program 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
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
* USA
|
|
*/
|
|
|
|
%locations
|
|
|
|
%{
|
|
#include <stdio.h>
|
|
|
|
#include "dtc.h"
|
|
#include "srcpos.h"
|
|
|
|
extern int yylex(void);
|
|
|
|
extern struct boot_info *the_boot_info;
|
|
extern int treesource_error;
|
|
|
|
static unsigned long long eval_literal(const char *s, int base, int bits);
|
|
%}
|
|
|
|
%union {
|
|
char *propnodename;
|
|
char *literal;
|
|
char *labelref;
|
|
unsigned int cbase;
|
|
uint8_t byte;
|
|
struct data data;
|
|
|
|
uint64_t addr;
|
|
cell_t cell;
|
|
struct property *prop;
|
|
struct property *proplist;
|
|
struct node *node;
|
|
struct node *nodelist;
|
|
struct reserve_info *re;
|
|
}
|
|
|
|
%token DT_V1
|
|
%token DT_MEMRESERVE
|
|
%token <propnodename> DT_PROPNODENAME
|
|
%token <literal> DT_LITERAL
|
|
%token <literal> DT_LEGACYLITERAL
|
|
%token <cbase> DT_BASE
|
|
%token <byte> DT_BYTE
|
|
%token <data> DT_STRING
|
|
%token <labelref> DT_LABEL
|
|
%token <labelref> DT_REF
|
|
%token DT_INCBIN
|
|
|
|
%type <data> propdata
|
|
%type <data> propdataprefix
|
|
%type <re> memreserve
|
|
%type <re> memreserves
|
|
%type <re> v0_memreserve
|
|
%type <re> v0_memreserves
|
|
%type <addr> addr
|
|
%type <data> celllist
|
|
%type <cbase> cellbase
|
|
%type <cell> cellval
|
|
%type <data> bytestring
|
|
%type <prop> propdef
|
|
%type <proplist> proplist
|
|
|
|
%type <node> devicetree
|
|
%type <node> nodedef
|
|
%type <node> subnode
|
|
%type <nodelist> subnodes
|
|
%type <labelref> label
|
|
|
|
%%
|
|
|
|
sourcefile:
|
|
DT_V1 ';' memreserves devicetree
|
|
{
|
|
the_boot_info = build_boot_info($3, $4, 0);
|
|
}
|
|
| v0_memreserves devicetree
|
|
{
|
|
the_boot_info = build_boot_info($1, $2, 0);
|
|
}
|
|
;
|
|
|
|
memreserves:
|
|
/* empty */
|
|
{
|
|
$$ = NULL;
|
|
}
|
|
| memreserve memreserves
|
|
{
|
|
$$ = chain_reserve_entry($1, $2);
|
|
}
|
|
;
|
|
|
|
memreserve:
|
|
label DT_MEMRESERVE addr addr ';'
|
|
{
|
|
$$ = build_reserve_entry($3, $4, $1);
|
|
}
|
|
;
|
|
|
|
v0_memreserves:
|
|
/* empty */
|
|
{
|
|
$$ = NULL;
|
|
}
|
|
| v0_memreserve v0_memreserves
|
|
{
|
|
$$ = chain_reserve_entry($1, $2);
|
|
};
|
|
;
|
|
|
|
v0_memreserve:
|
|
memreserve
|
|
{
|
|
$$ = $1;
|
|
}
|
|
| label DT_MEMRESERVE addr '-' addr ';'
|
|
{
|
|
$$ = build_reserve_entry($3, $5 - $3 + 1, $1);
|
|
}
|
|
;
|
|
|
|
addr:
|
|
DT_LITERAL
|
|
{
|
|
$$ = eval_literal($1, 0, 64);
|
|
}
|
|
| DT_LEGACYLITERAL
|
|
{
|
|
$$ = eval_literal($1, 16, 64);
|
|
}
|
|
;
|
|
|
|
devicetree:
|
|
'/' nodedef
|
|
{
|
|
$$ = name_node($2, "", NULL);
|
|
}
|
|
;
|
|
|
|
nodedef:
|
|
'{' proplist subnodes '}' ';'
|
|
{
|
|
$$ = build_node($2, $3);
|
|
}
|
|
;
|
|
|
|
proplist:
|
|
/* empty */
|
|
{
|
|
$$ = NULL;
|
|
}
|
|
| proplist propdef
|
|
{
|
|
$$ = chain_property($2, $1);
|
|
}
|
|
;
|
|
|
|
propdef:
|
|
label DT_PROPNODENAME '=' propdata ';'
|
|
{
|
|
$$ = build_property($2, $4, $1);
|
|
}
|
|
| label DT_PROPNODENAME ';'
|
|
{
|
|
$$ = build_property($2, empty_data, $1);
|
|
}
|
|
;
|
|
|
|
propdata:
|
|
propdataprefix DT_STRING
|
|
{
|
|
$$ = data_merge($1, $2);
|
|
}
|
|
| propdataprefix '<' celllist '>'
|
|
{
|
|
$$ = data_merge($1, $3);
|
|
}
|
|
| propdataprefix '[' bytestring ']'
|
|
{
|
|
$$ = data_merge($1, $3);
|
|
}
|
|
| propdataprefix DT_REF
|
|
{
|
|
$$ = 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
|
|
{
|
|
$$ = data_add_marker($1, LABEL, $2);
|
|
}
|
|
;
|
|
|
|
propdataprefix:
|
|
/* empty */
|
|
{
|
|
$$ = empty_data;
|
|
}
|
|
| propdata ','
|
|
{
|
|
$$ = $1;
|
|
}
|
|
| propdataprefix DT_LABEL
|
|
{
|
|
$$ = data_add_marker($1, LABEL, $2);
|
|
}
|
|
;
|
|
|
|
celllist:
|
|
/* empty */
|
|
{
|
|
$$ = empty_data;
|
|
}
|
|
| celllist cellval
|
|
{
|
|
$$ = data_append_cell($1, $2);
|
|
}
|
|
| celllist DT_REF
|
|
{
|
|
$$ = data_append_cell(data_add_marker($1, REF_PHANDLE,
|
|
$2), -1);
|
|
}
|
|
| celllist DT_LABEL
|
|
{
|
|
$$ = data_add_marker($1, LABEL, $2);
|
|
}
|
|
;
|
|
|
|
cellbase:
|
|
/* empty */
|
|
{
|
|
$$ = 16;
|
|
}
|
|
| DT_BASE
|
|
;
|
|
|
|
cellval:
|
|
DT_LITERAL
|
|
{
|
|
$$ = eval_literal($1, 0, 32);
|
|
}
|
|
| cellbase DT_LEGACYLITERAL
|
|
{
|
|
$$ = eval_literal($2, $1, 32);
|
|
}
|
|
;
|
|
|
|
bytestring:
|
|
/* empty */
|
|
{
|
|
$$ = empty_data;
|
|
}
|
|
| bytestring DT_BYTE
|
|
{
|
|
$$ = data_append_byte($1, $2);
|
|
}
|
|
| bytestring DT_LABEL
|
|
{
|
|
$$ = data_add_marker($1, LABEL, $2);
|
|
}
|
|
;
|
|
|
|
subnodes:
|
|
/* empty */
|
|
{
|
|
$$ = NULL;
|
|
}
|
|
| subnode subnodes
|
|
{
|
|
$$ = chain_node($1, $2);
|
|
}
|
|
| subnode propdef
|
|
{
|
|
yyerror("syntax error: properties must precede subnodes");
|
|
YYERROR;
|
|
}
|
|
;
|
|
|
|
subnode:
|
|
label DT_PROPNODENAME nodedef
|
|
{
|
|
$$ = name_node($3, $2, $1);
|
|
}
|
|
;
|
|
|
|
label:
|
|
/* empty */
|
|
{
|
|
$$ = NULL;
|
|
}
|
|
| DT_LABEL
|
|
{
|
|
$$ = $1;
|
|
}
|
|
;
|
|
|
|
%%
|
|
|
|
void yyerrorf(char const *s, ...)
|
|
{
|
|
const char *fname = srcpos_file ? srcpos_file->name : "<no-file>";
|
|
va_list va;
|
|
va_start(va, s);
|
|
|
|
if (strcmp(fname, "-") == 0)
|
|
fname = "stdin";
|
|
|
|
fprintf(stderr, "%s:%d ", fname, yylloc.first_line);
|
|
vfprintf(stderr, s, va);
|
|
fprintf(stderr, "\n");
|
|
|
|
treesource_error = 1;
|
|
va_end(va);
|
|
}
|
|
|
|
void yyerror (char const *s)
|
|
{
|
|
yyerrorf("%s", s);
|
|
}
|
|
|
|
static unsigned long long eval_literal(const char *s, int base, int bits)
|
|
{
|
|
unsigned long long val;
|
|
char *e;
|
|
|
|
errno = 0;
|
|
val = strtoull(s, &e, base);
|
|
if (*e)
|
|
yyerror("bad characters in literal");
|
|
else if ((errno == ERANGE)
|
|
|| ((bits < 64) && (val >= (1ULL << bits))))
|
|
yyerror("literal out of range");
|
|
else if (errno != 0)
|
|
yyerror("bad literal");
|
|
return val;
|
|
}
|
|
|