af0dd31fc4
be used on the host system (and not installed on the device, if required). The GPL'd one is still available if there are any devices that need it (make universe passes with it, including kernels that use fdt, but there may be some out-of-tree ones). WITH_GPL_DTC can be used to select the old one, for now. Probably won't be MFC'd, but we'll remove the GPL'd version in head after the new one has had a lot more testing and ship it in 10.0.
1358 lines
29 KiB
C++
1358 lines
29 KiB
C++
/*-
|
|
* Copyright (c) 2013 David Chisnall
|
|
* All rights reserved.
|
|
*
|
|
* This software was developed by SRI International and the University of
|
|
* Cambridge Computer Laboratory under DARPA/AFRL contract (FA8750-10-C-0237)
|
|
* ("CTSRD"), as part of the DARPA CRASH research programme.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*
|
|
* $FreeBSD$
|
|
*/
|
|
|
|
#include "fdt.hh"
|
|
|
|
#include <algorithm>
|
|
#include <inttypes.h>
|
|
#include <fcntl.h>
|
|
#include <libgen.h>
|
|
#include "dtb.hh"
|
|
|
|
namespace dtc
|
|
{
|
|
|
|
namespace fdt
|
|
{
|
|
|
|
uint32_t
|
|
property_value::get_as_uint32()
|
|
{
|
|
if (byte_data.size() != 4)
|
|
{
|
|
return 0;
|
|
}
|
|
uint32_t v = 0;
|
|
v &= byte_data[0] << 24;
|
|
v &= byte_data[1] << 16;
|
|
v &= byte_data[2] << 8;
|
|
v &= byte_data[3] << 0;
|
|
return v;
|
|
}
|
|
|
|
void
|
|
property_value::push_to_buffer(byte_buffer &buffer)
|
|
{
|
|
if (!byte_data.empty())
|
|
{
|
|
buffer.insert(buffer.end(), byte_data.begin(), byte_data.end());
|
|
}
|
|
else
|
|
{
|
|
string_data.push_to_buffer(buffer, true);
|
|
// Trailing nul
|
|
buffer.push_back(0);
|
|
}
|
|
}
|
|
|
|
void
|
|
property_value::write_dts(FILE *file)
|
|
{
|
|
resolve_type();
|
|
switch (type)
|
|
{
|
|
default:
|
|
assert(0 && "Invalid type");
|
|
case STRING:
|
|
case STRING_LIST:
|
|
case CROSS_REFERENCE:
|
|
write_as_string(file);
|
|
break;
|
|
case PHANDLE:
|
|
write_as_cells(file);
|
|
break;
|
|
case BINARY:
|
|
if (byte_data.size() % 4 == 0)
|
|
{
|
|
write_as_cells(file);
|
|
break;
|
|
}
|
|
write_as_bytes(file);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
property_value::resolve_type()
|
|
{
|
|
if (type != UNKNOWN)
|
|
{
|
|
return;
|
|
}
|
|
if (byte_data.empty())
|
|
{
|
|
type = STRING;
|
|
return;
|
|
}
|
|
if (byte_data.back() == 0)
|
|
{
|
|
bool is_all_printable = true;
|
|
int nuls = 0;
|
|
int bytes = 0;
|
|
for (byte_buffer::iterator i=byte_data.begin(), e=byte_data.end()-1; i<e ; i++)
|
|
{
|
|
bytes++;
|
|
is_all_printable &= (*i == '\0') || isprint(*i);
|
|
if (*i == '\0')
|
|
{
|
|
nuls++;
|
|
}
|
|
if (!is_all_printable)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
if (is_all_printable && (bytes > nuls))
|
|
{
|
|
type = STRING;
|
|
if (nuls > 0)
|
|
{
|
|
type = STRING_LIST;
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
type = BINARY;
|
|
}
|
|
|
|
void
|
|
property_value::write_as_string(FILE *file)
|
|
{
|
|
putc('"', file);
|
|
if (byte_data.empty())
|
|
{
|
|
string_data.print(file);
|
|
}
|
|
else
|
|
{
|
|
for (byte_buffer::iterator i=byte_data.begin(), e=byte_data.end()-1; i!=e ; ++i)
|
|
{
|
|
// FIXME Escape tabs, newlines, and so on.
|
|
if (*i == '\0')
|
|
{
|
|
fputs("\", \"", file);
|
|
continue;
|
|
}
|
|
putc(*i, file);
|
|
}
|
|
}
|
|
putc('"', file);
|
|
}
|
|
|
|
void
|
|
property_value::write_as_cells(FILE *file)
|
|
{
|
|
putc('<', file);
|
|
assert((byte_data.size() % 4) == 0);
|
|
for (byte_buffer::iterator i=byte_data.begin(), e=byte_data.end(); i!=e ; ++i)
|
|
{
|
|
uint32_t v = 0;
|
|
v = (v << 8) | *i;
|
|
++i;
|
|
v = (v << 8) | *i;
|
|
++i;
|
|
v = (v << 8) | *i;
|
|
++i;
|
|
v = (v << 8) | *i;
|
|
fprintf(file, "0x%" PRIx32, v);
|
|
if (i+1 != e)
|
|
{
|
|
putc(' ', file);
|
|
}
|
|
}
|
|
putc('>', file);
|
|
}
|
|
|
|
void
|
|
property_value::write_as_bytes(FILE *file)
|
|
{
|
|
putc('[', file);
|
|
for (byte_buffer::iterator i=byte_data.begin(), e=byte_data.end(); i!=e ; i++)
|
|
{
|
|
fprintf(file, "%hhx", *i);
|
|
if (i+1 != e)
|
|
{
|
|
putc(' ', file);
|
|
}
|
|
}
|
|
putc(']', file);
|
|
}
|
|
|
|
void
|
|
property::parse_string(input_buffer &input)
|
|
{
|
|
property_value v;
|
|
assert(input[0] == '"');
|
|
++input;
|
|
const char *start = (const char*)input;
|
|
int length = 0;
|
|
while (char c = input[0])
|
|
{
|
|
if (c == '"' && input[-1] != '\\')
|
|
{
|
|
input.consume('"');
|
|
break;
|
|
}
|
|
++input;
|
|
++length;
|
|
}
|
|
v.string_data = string(start, length);
|
|
values.push_back(v);
|
|
}
|
|
|
|
void
|
|
property::parse_cells(input_buffer &input)
|
|
{
|
|
assert(input[0] == '<');
|
|
++input;
|
|
property_value v;
|
|
input.next_token();
|
|
while (!input.consume('>'))
|
|
{
|
|
input.next_token();
|
|
// If this is a phandle then we need to get the name of the
|
|
// referenced node
|
|
if (input.consume('&'))
|
|
{
|
|
input.next_token();
|
|
// FIXME: We should support full paths here, but we
|
|
// don't.
|
|
string referenced = string::parse_node_name(input);
|
|
if (referenced.empty())
|
|
{
|
|
input.parse_error("Expected node name");
|
|
valid = false;
|
|
return;
|
|
}
|
|
input.next_token();
|
|
// If we already have some bytes, make the phandle a
|
|
// separate component.
|
|
if (!v.byte_data.empty())
|
|
{
|
|
values.push_back(v);
|
|
v = property_value();
|
|
}
|
|
v.string_data = referenced;
|
|
v.type = property_value::PHANDLE;
|
|
values.push_back(v);
|
|
v = property_value();
|
|
}
|
|
else
|
|
{
|
|
//FIXME: We should support labels in the middle
|
|
//of these, but we don't.
|
|
long long val;
|
|
if (!input.consume_integer(val))
|
|
{
|
|
input.parse_error("Expected numbers in array of cells");
|
|
valid = false;
|
|
return;
|
|
}
|
|
push_big_endian(v.byte_data, (uint32_t)val);
|
|
input.next_token();
|
|
}
|
|
}
|
|
// Don't store an empty string value here.
|
|
if (v.byte_data.size() > 0)
|
|
{
|
|
values.push_back(v);
|
|
}
|
|
}
|
|
|
|
void
|
|
property::parse_bytes(input_buffer &input)
|
|
{
|
|
assert(input[0] == '[');
|
|
++input;
|
|
property_value v;
|
|
input.next_token();
|
|
while (!input.consume(']'))
|
|
{
|
|
{
|
|
//FIXME: We should support
|
|
//labels in the middle of
|
|
//these, but we don't.
|
|
uint8_t val;
|
|
if (!input.consume_hex_byte(val))
|
|
{
|
|
input.parse_error("Expected hex bytes in array of bytes");
|
|
valid = false;
|
|
return;
|
|
}
|
|
v.byte_data.push_back(val);
|
|
input.next_token();
|
|
}
|
|
}
|
|
values.push_back(v);
|
|
}
|
|
|
|
void
|
|
property::parse_reference(input_buffer &input)
|
|
{
|
|
assert(input[0] == '&');
|
|
++input;
|
|
input.next_token();
|
|
property_value v;
|
|
v.string_data = string::parse_node_name(input);
|
|
if (v.string_data.empty())
|
|
{
|
|
input.parse_error("Expected node name");
|
|
valid = false;
|
|
return;
|
|
}
|
|
v.type = property_value::CROSS_REFERENCE;
|
|
values.push_back(v);
|
|
}
|
|
|
|
property::property(input_buffer &structs, input_buffer &strings)
|
|
{
|
|
uint32_t name_offset;
|
|
uint32_t length;
|
|
valid = structs.consume_binary(length) &&
|
|
structs.consume_binary(name_offset);
|
|
if (!valid)
|
|
{
|
|
fprintf(stderr, "Failed to read property\n");
|
|
return;
|
|
}
|
|
// Find the name
|
|
input_buffer name_buffer = strings.buffer_from_offset(name_offset);
|
|
if (name_buffer.empty())
|
|
{
|
|
fprintf(stderr, "Property name offset %" PRIu32
|
|
" is past the end of the strings table\n",
|
|
name_offset);
|
|
valid = false;
|
|
return;
|
|
}
|
|
key = string(name_buffer);
|
|
// Read the value
|
|
uint8_t byte;
|
|
property_value v;
|
|
for (uint32_t i=0 ; i<length ; i++)
|
|
{
|
|
if (!(valid = structs.consume_binary(byte)))
|
|
{
|
|
fprintf(stderr, "Failed to read property value\n");
|
|
return;
|
|
}
|
|
v.byte_data.push_back(byte);
|
|
}
|
|
values.push_back(v);
|
|
}
|
|
|
|
property::property(input_buffer &input, string k, string l) : key(k), label(l),
|
|
valid(true)
|
|
{
|
|
do {
|
|
input.next_token();
|
|
switch (input[0])
|
|
{
|
|
default:
|
|
input.parse_error("Invalid property value.");
|
|
valid = false;
|
|
return;
|
|
case '"':
|
|
parse_string(input);
|
|
break;
|
|
case '<':
|
|
parse_cells(input);
|
|
break;
|
|
case '[':
|
|
parse_bytes(input);
|
|
break;
|
|
case '&':
|
|
parse_reference(input);
|
|
break;
|
|
case ';':
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
input.next_token();
|
|
} while (input.consume(','));
|
|
if (!input.consume(';'))
|
|
{
|
|
input.parse_error("Expected ; at end of property");
|
|
valid = false;
|
|
}
|
|
}
|
|
|
|
property*
|
|
property::parse_dtb(input_buffer &structs, input_buffer &strings)
|
|
{
|
|
property *p = new property(structs, strings);
|
|
if (!p->valid)
|
|
{
|
|
delete p;
|
|
p = 0;
|
|
}
|
|
return p;
|
|
}
|
|
|
|
property*
|
|
property::parse(input_buffer &input, string key, string label)
|
|
{
|
|
property *p = new property(input, key, label);
|
|
if (!p->valid)
|
|
{
|
|
delete p;
|
|
p = 0;
|
|
}
|
|
return p;
|
|
}
|
|
|
|
void
|
|
property::write(dtb::output_writer &writer, dtb::string_table &strings)
|
|
{
|
|
writer.write_token(dtb::FDT_PROP);
|
|
byte_buffer value_buffer;
|
|
for (value_iterator i=begin(), e=end() ; i!=e ; ++i)
|
|
{
|
|
i->push_to_buffer(value_buffer);
|
|
}
|
|
writer.write_data((uint32_t)value_buffer.size());
|
|
writer.write_comment(key);
|
|
writer.write_data(strings.add_string(key));
|
|
writer.write_data(value_buffer);
|
|
}
|
|
|
|
void
|
|
property::write_dts(FILE *file, int indent)
|
|
{
|
|
for (int i=0 ; i<indent ; i++)
|
|
{
|
|
putc('\t', file);
|
|
}
|
|
if (label != string())
|
|
{
|
|
label.print(file);
|
|
fputs(": ", file);
|
|
}
|
|
if (key != string())
|
|
{
|
|
key.print(file);
|
|
}
|
|
if (!values.empty())
|
|
{
|
|
fputs(" = ", file);
|
|
for (value_iterator i=begin(), e=end() ; i!=e ; ++i)
|
|
{
|
|
i->write_dts(file);
|
|
if (i+1 != e)
|
|
{
|
|
putc(',', file);
|
|
putc(' ', file);
|
|
}
|
|
}
|
|
}
|
|
fputs(";\n", file);
|
|
}
|
|
|
|
string
|
|
node::parse_name(input_buffer &input, bool &is_property, const char *error)
|
|
{
|
|
if (!valid)
|
|
{
|
|
return string();
|
|
}
|
|
input.next_token();
|
|
if (is_property)
|
|
{
|
|
return string::parse_property_name(input);
|
|
}
|
|
string n = string::parse_node_or_property_name(input, is_property);
|
|
if (n.empty())
|
|
{
|
|
if (n.empty())
|
|
{
|
|
input.parse_error(error);
|
|
valid = false;
|
|
}
|
|
}
|
|
return n;
|
|
}
|
|
|
|
node::node(input_buffer &structs, input_buffer &strings) : valid(true)
|
|
{
|
|
const char *name_start = (const char*)structs;
|
|
int name_length = 0;
|
|
while (structs[0] != '\0' && structs[0] != '@')
|
|
{
|
|
name_length++;
|
|
++structs;
|
|
}
|
|
name = string(name_start, name_length);
|
|
if (structs[0] == '@')
|
|
{
|
|
++structs;
|
|
name_start = (const char*)structs;
|
|
name_length = 0;
|
|
while (structs[0] != '\0')
|
|
{
|
|
name_length++;
|
|
++structs;
|
|
}
|
|
unit_address = string(name_start, name_length);
|
|
}
|
|
++structs;
|
|
uint32_t token;
|
|
while (structs.consume_binary(token))
|
|
{
|
|
switch (token)
|
|
{
|
|
default:
|
|
fprintf(stderr, "Unexpected token 0x%" PRIx32
|
|
" while parsing node.\n", token);
|
|
valid = false;
|
|
return;
|
|
// Child node, parse it.
|
|
case dtb::FDT_BEGIN_NODE:
|
|
{
|
|
node *child = node::parse_dtb(structs, strings);
|
|
if (child == 0)
|
|
{
|
|
valid = false;
|
|
return;
|
|
}
|
|
children.push_back(child);
|
|
break;
|
|
}
|
|
// End of this node, no errors.
|
|
case dtb::FDT_END_NODE:
|
|
return;
|
|
// Property, parse it.
|
|
case dtb::FDT_PROP:
|
|
{
|
|
property *prop = property::parse_dtb(structs, strings);
|
|
if (prop == 0)
|
|
{
|
|
valid = false;
|
|
return;
|
|
}
|
|
properties.push_back(prop);
|
|
break;
|
|
}
|
|
break;
|
|
// End of structs table. Should appear after
|
|
// the end of the last node.
|
|
case dtb::FDT_END:
|
|
fprintf(stderr, "Unexpected FDT_END token while parsing node.\n");
|
|
valid = false;
|
|
return;
|
|
// NOPs are padding. Ignore them.
|
|
case dtb::FDT_NOP:
|
|
break;
|
|
}
|
|
}
|
|
fprintf(stderr, "Failed to read token from structs table while parsing node.\n");
|
|
valid = false;
|
|
return;
|
|
}
|
|
|
|
node::node(input_buffer &input, string n, string l, string a) :
|
|
label(l), name(n), unit_address(a), valid(true)
|
|
{
|
|
if (!input.consume('{'))
|
|
{
|
|
input.parse_error("Expected { to start new device tree node.\n");
|
|
}
|
|
input.next_token();
|
|
while (valid && !input.consume('}'))
|
|
{
|
|
// flag set if we find any characters that are only in
|
|
// the property name character set, not the node
|
|
bool is_property = false;
|
|
string child_name, child_label, child_address;
|
|
child_name = parse_name(input, is_property,
|
|
"Expected property or node name");
|
|
if (input.consume(':'))
|
|
{
|
|
// Node labels can contain any characters? The
|
|
// spec doesn't say, so we guess so...
|
|
is_property = false;
|
|
child_label = child_name;
|
|
child_name = parse_name(input, is_property, "Expected property or node name");
|
|
}
|
|
if (input.consume('@'))
|
|
{
|
|
child_address = parse_name(input, is_property, "Expected unit address");
|
|
}
|
|
if (!valid)
|
|
{
|
|
return;
|
|
}
|
|
input.next_token();
|
|
// If we're parsing a property, then we must actually do that.
|
|
if (input.consume('='))
|
|
{
|
|
property *p= property::parse(input, child_name,
|
|
child_label);
|
|
if (p == 0)
|
|
{
|
|
valid = false;
|
|
}
|
|
else
|
|
{
|
|
properties.push_back(p);
|
|
}
|
|
}
|
|
else if (!is_property && input[0] == ('{'))
|
|
{
|
|
node *child = node::parse(input, child_name,
|
|
child_label, child_address);
|
|
if (child)
|
|
{
|
|
children.push_back(child);
|
|
}
|
|
else
|
|
{
|
|
valid = false;
|
|
}
|
|
}
|
|
else if (input.consume(';'))
|
|
{
|
|
properties.push_back(new property(child_name, child_label));
|
|
}
|
|
else
|
|
{
|
|
input.parse_error("Error parsing property.");
|
|
valid = false;
|
|
}
|
|
input.next_token();
|
|
}
|
|
input.consume(';');
|
|
}
|
|
|
|
bool
|
|
node::cmp_properties(property *p1, property *p2)
|
|
{
|
|
return p1->get_key() < p2->get_key();
|
|
}
|
|
|
|
bool
|
|
node::cmp_children(node *c1, node *c2)
|
|
{
|
|
if (c1->name == c2->name)
|
|
{
|
|
return c1->unit_address < c2->unit_address;
|
|
}
|
|
return c1->name < c2->name;
|
|
}
|
|
|
|
void
|
|
node::sort()
|
|
{
|
|
std::sort(property_begin(), property_end(), cmp_properties);
|
|
std::sort(child_begin(), child_end(), cmp_children);
|
|
for (child_iterator i=child_begin(), e=child_end() ; i!=e ; ++i)
|
|
{
|
|
(*i)->sort();
|
|
}
|
|
}
|
|
|
|
node*
|
|
node::parse(input_buffer &input, string name, string label, string address)
|
|
{
|
|
node *n = new node(input, name, label, address);
|
|
if (!n->valid)
|
|
{
|
|
delete n;
|
|
n = 0;
|
|
}
|
|
return n;
|
|
}
|
|
|
|
node*
|
|
node::parse_dtb(input_buffer &structs, input_buffer &strings)
|
|
{
|
|
node *n = new node(structs, strings);
|
|
if (!n->valid)
|
|
{
|
|
delete n;
|
|
n = 0;
|
|
}
|
|
return n;
|
|
}
|
|
|
|
node::~node()
|
|
{
|
|
while (!children.empty())
|
|
{
|
|
delete children.back();
|
|
children.pop_back();
|
|
}
|
|
while (!properties.empty())
|
|
{
|
|
delete properties.back();
|
|
properties.pop_back();
|
|
}
|
|
}
|
|
|
|
property*
|
|
node::get_property(string key)
|
|
{
|
|
for (property_iterator i=property_begin(), e=property_end() ; i!=e ; ++i)
|
|
{
|
|
if ((*i)->get_key() == key)
|
|
{
|
|
return *i;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
node::merge_node(node *other)
|
|
{
|
|
if (!other->label.empty())
|
|
{
|
|
label = other->label;
|
|
}
|
|
// Note: this is an O(n*m) operation. It might be sensible to
|
|
// optimise this if we find that there are nodes with very
|
|
// large numbers of properties, but for typical usage the
|
|
// entire vector will fit (easily) into cache, so iterating
|
|
// over it repeatedly isn't that expensive.
|
|
while (!other->properties.empty())
|
|
{
|
|
property *p = other->properties.front();
|
|
for (property_iterator i=property_begin(), e=property_end() ; i!=e ; ++i)
|
|
{
|
|
if ((*i)->get_key() == p->get_key())
|
|
{
|
|
delete *i;
|
|
properties.erase(i);
|
|
break;
|
|
}
|
|
}
|
|
add_property(p);
|
|
other->properties.erase(other->properties.begin());
|
|
}
|
|
while (!other->children.empty())
|
|
{
|
|
node *c = other->children.front();
|
|
bool found = false;
|
|
for (child_iterator i=child_begin(), e=child_end() ; i!=e ; ++i)
|
|
{
|
|
if ((*i)->name == c->name && (*i)->unit_address == c->unit_address)
|
|
{
|
|
(*i)->merge_node(c);
|
|
delete c;
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!found)
|
|
{
|
|
children.push_back(c);
|
|
}
|
|
other->children.erase(other->children.begin());
|
|
}
|
|
}
|
|
|
|
void
|
|
node::write(dtb::output_writer &writer, dtb::string_table &strings)
|
|
{
|
|
writer.write_token(dtb::FDT_BEGIN_NODE);
|
|
byte_buffer name_buffer;
|
|
name.push_to_buffer(name_buffer);
|
|
if (unit_address != string())
|
|
{
|
|
name_buffer.push_back('@');
|
|
unit_address.push_to_buffer(name_buffer);
|
|
}
|
|
writer.write_comment(name);
|
|
writer.write_data(name_buffer);
|
|
writer.write_data((uint8_t)0);
|
|
for (property_iterator i=property_begin(), e=property_end() ; i!=e ; ++i)
|
|
{
|
|
(*i)->write(writer, strings);
|
|
}
|
|
for (child_iterator i=child_begin(), e=child_end() ; i!=e ; ++i)
|
|
{
|
|
(*i)->write(writer, strings);
|
|
}
|
|
writer.write_token(dtb::FDT_END_NODE);
|
|
}
|
|
|
|
void
|
|
node::write_dts(FILE *file, int indent)
|
|
{
|
|
for (int i=0 ; i<indent ; i++)
|
|
{
|
|
putc('\t', file);
|
|
}
|
|
if (label != string())
|
|
{
|
|
label.print(file);
|
|
fputs(": ", file);
|
|
}
|
|
if (name != string())
|
|
{
|
|
name.print(file);
|
|
}
|
|
if (unit_address != string())
|
|
{
|
|
putc('@', file);
|
|
unit_address.print(file);
|
|
}
|
|
fputs(" {\n\n", file);
|
|
for (property_iterator i=property_begin(), e=property_end() ; i!=e ; ++i)
|
|
{
|
|
(*i)->write_dts(file, indent+1);
|
|
}
|
|
for (child_iterator i=child_begin(), e=child_end() ; i!=e ; ++i)
|
|
{
|
|
(*i)->write_dts(file, indent+1);
|
|
}
|
|
for (int i=0 ; i<indent ; i++)
|
|
{
|
|
putc('\t', file);
|
|
}
|
|
fputs("};\n", file);
|
|
}
|
|
|
|
void
|
|
device_tree::collect_names_recursive(node* n, node_path &path)
|
|
{
|
|
string name = n->label;
|
|
path.push_back(std::make_pair(n->name, n->unit_address));
|
|
if (name != string())
|
|
{
|
|
if (node_names.find(name) == node_names.end())
|
|
{
|
|
node_names.insert(std::make_pair(name, n));
|
|
node_paths.insert(std::make_pair(name, path));
|
|
}
|
|
else
|
|
{
|
|
node_names[name] = (node*)-1;
|
|
std::map<string, node_path>::iterator i = node_paths.find(name);
|
|
if (i != node_paths.end())
|
|
{
|
|
node_paths.erase(name);
|
|
}
|
|
fprintf(stderr, "Label not unique: ");
|
|
name.dump();
|
|
fprintf(stderr, ". References to this label will not be resolved.");
|
|
}
|
|
}
|
|
for (node::child_iterator i=n->child_begin(), e=n->child_end() ; i!=e ; ++i)
|
|
{
|
|
collect_names_recursive(*i, path);
|
|
}
|
|
path.pop_back();
|
|
// Now we collect the phandles and properties that reference
|
|
// other nodes.
|
|
for (node::property_iterator i=n->property_begin(), e=n->property_end() ; i!=e ; ++i)
|
|
{
|
|
for (property::value_iterator p=(*i)->begin(),pe=(*i)->end() ; p!=pe ; ++p)
|
|
{
|
|
if (p->is_phandle())
|
|
{
|
|
phandles.push_back(&*p);
|
|
}
|
|
if (p->is_cross_reference())
|
|
{
|
|
cross_references.push_back(&*p);
|
|
}
|
|
}
|
|
if ((*i)->get_key() == string("phandle") ||
|
|
(*i)->get_key() == string("linux,phandle"))
|
|
{
|
|
if ((*i)->begin()->byte_data.size() != 4)
|
|
{
|
|
fprintf(stderr, "Invalid phandle value for node ");
|
|
n->name.dump();
|
|
fprintf(stderr, ". Should be a 4-byte value.\n");
|
|
valid = false;
|
|
}
|
|
else
|
|
{
|
|
uint32_t phandle = (*i)->begin()->get_as_uint32();
|
|
used_phandles.insert(std::make_pair(phandle, n));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
device_tree::collect_names()
|
|
{
|
|
node_path p;
|
|
collect_names_recursive(root, p);
|
|
}
|
|
|
|
void
|
|
device_tree::resolve_cross_references()
|
|
{
|
|
for (std::vector<property_value*>::iterator i=cross_references.begin(), e=cross_references.end() ; i!=e ; ++i)
|
|
{
|
|
property_value* pv = *i;
|
|
node_path path = node_paths[pv->string_data];
|
|
// Skip the first name in the path. It's always "", and implicitly /
|
|
for (node_path::iterator p=path.begin()+1, pe=path.end() ; p!=pe ; ++p)
|
|
{
|
|
pv->byte_data.push_back('/');
|
|
p->first.push_to_buffer(pv->byte_data);
|
|
if (!(p->second.empty()))
|
|
{
|
|
pv->byte_data.push_back('@');
|
|
p->second.push_to_buffer(pv->byte_data);
|
|
}
|
|
}
|
|
pv->byte_data.push_back(0);
|
|
}
|
|
uint32_t phandle = 1;
|
|
for (std::vector<property_value*>::iterator i=phandles.begin(), e=phandles.end() ; i!=e ; ++i)
|
|
{
|
|
string target_name = (*i)->string_data;
|
|
node *target = node_names[target_name];
|
|
if (target == 0)
|
|
{
|
|
fprintf(stderr, "Failed to find node with label:");
|
|
target_name.dump();
|
|
fprintf(stderr, "\n");
|
|
valid = 0;
|
|
return;
|
|
}
|
|
// If there is an existing phandle, use it
|
|
property *p = target->get_property("phandle");
|
|
if (p == 0)
|
|
{
|
|
p = target->get_property("linux,phandle");
|
|
}
|
|
if (p == 0)
|
|
{
|
|
// Otherwise insert a new phandle node
|
|
property_value v;
|
|
while (used_phandles.find(phandle) != used_phandles.end())
|
|
{
|
|
// Note that we only don't need to
|
|
// store this phandle in the set,
|
|
// because we are monotonically
|
|
// increasing the value of phandle and
|
|
// so will only ever revisit this value
|
|
// if we have used 2^32 phandles, at
|
|
// which point our blob won't fit in
|
|
// any 32-bit system and we've done
|
|
// something badly wrong elsewhere
|
|
// already.
|
|
phandle++;
|
|
}
|
|
push_big_endian(v.byte_data, phandle++);
|
|
if (phandle_node_name == BOTH || phandle_node_name == LINUX)
|
|
{
|
|
p = new property(string("linux,phandle"));
|
|
p->add_value(v);
|
|
target->add_property(p);
|
|
}
|
|
if (phandle_node_name == BOTH || phandle_node_name == EPAPR)
|
|
{
|
|
p = new property(string("phandle"));
|
|
p->add_value(v);
|
|
target->add_property(p);
|
|
}
|
|
}
|
|
p->begin()->push_to_buffer((*i)->byte_data);
|
|
assert((*i)->byte_data.size() == 4);
|
|
}
|
|
}
|
|
|
|
void
|
|
device_tree::parse_roots(input_buffer &input, std::vector<node*> &roots)
|
|
{
|
|
input.next_token();
|
|
while (valid && input.consume('/'))
|
|
{
|
|
input.next_token();
|
|
node *n = node::parse(input, string("", 1));
|
|
if (n)
|
|
{
|
|
roots.push_back(n);
|
|
}
|
|
else
|
|
{
|
|
valid = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
input_buffer*
|
|
device_tree::buffer_for_file(const char *path)
|
|
{
|
|
if (string(path) == string("-"))
|
|
{
|
|
input_buffer *b = new stream_input_buffer();
|
|
buffers.push_back(b);
|
|
return b;
|
|
}
|
|
int source = open(path, O_RDONLY);
|
|
if (source == -1)
|
|
{
|
|
fprintf(stderr, "Unable to open file %s\n", path);
|
|
return 0;
|
|
}
|
|
input_buffer *b = new mmap_input_buffer(source);
|
|
// Keep the buffer that owns the memory around for the lifetime
|
|
// of this FDT. Ones simply referring to it may have shorter
|
|
// lifetimes.
|
|
buffers.push_back(b);
|
|
close(source);
|
|
return b;
|
|
}
|
|
|
|
template<class writer> void
|
|
device_tree::write(int fd)
|
|
{
|
|
dtb::string_table st;
|
|
dtb::header head;
|
|
writer head_writer;
|
|
writer reservation_writer;
|
|
writer struct_writer;
|
|
writer strings_writer;
|
|
|
|
// Build the reservation table
|
|
reservation_writer.write_comment(string("Memory reservations"));
|
|
reservation_writer.write_label(string("dt_reserve_map"));
|
|
for (std::vector<reservation>::iterator i=reservations.begin(),
|
|
e=reservations.end() ; i!=e ; ++i)
|
|
{
|
|
reservation_writer.write_comment(string("Reservation start"));
|
|
reservation_writer.write_data(i->first);
|
|
reservation_writer.write_comment(string("Reservation length"));
|
|
reservation_writer.write_data(i->first);
|
|
}
|
|
// Write n spare reserve map entries, plus the trailing 0.
|
|
for (uint32_t i=0 ; i<=spare_reserve_map_entries ; i++)
|
|
{
|
|
reservation_writer.write_data((uint64_t)0);
|
|
reservation_writer.write_data((uint64_t)0);
|
|
}
|
|
|
|
|
|
struct_writer.write_comment(string("Device tree"));
|
|
struct_writer.write_label(string("dt_struct_start"));
|
|
root->write(struct_writer, st);
|
|
struct_writer.write_token(dtb::FDT_END);
|
|
struct_writer.write_label(string("dt_struct_end"));
|
|
|
|
st.write(strings_writer);
|
|
// Find the strings size before we stick padding on the end.
|
|
// Note: We should possibly use a new writer for the padding.
|
|
head.size_dt_strings = strings_writer.size();
|
|
|
|
// Stick the padding in the strings writer, but after the
|
|
// marker indicating that it's the end.
|
|
// Note: We probably should add a padding call to the writer so
|
|
// that the asm back end can write padding directives instead
|
|
// of a load of 0 bytes.
|
|
for (uint32_t i=0 ; i<blob_padding ; i++)
|
|
{
|
|
strings_writer.write_data((uint8_t)0);
|
|
}
|
|
head.totalsize = sizeof(head) + strings_writer.size() +
|
|
struct_writer.size() + reservation_writer.size();
|
|
while (head.totalsize < minimum_blob_size)
|
|
{
|
|
head.totalsize++;
|
|
strings_writer.write_data((uint8_t)0);
|
|
}
|
|
head.off_dt_struct = sizeof(head) + reservation_writer.size();;
|
|
head.off_dt_strings = head.off_dt_struct + struct_writer.size();
|
|
head.off_mem_rsvmap = sizeof(head);
|
|
head.boot_cpuid_phys = boot_cpu;
|
|
head.size_dt_struct = struct_writer.size();
|
|
head.write(head_writer);
|
|
|
|
head_writer.write_to_file(fd);
|
|
reservation_writer.write_to_file(fd);
|
|
struct_writer.write_to_file(fd);
|
|
strings_writer.write_label(string("dt_blob_end"));
|
|
strings_writer.write_to_file(fd);
|
|
}
|
|
|
|
node*
|
|
device_tree::referenced_node(property_value &v)
|
|
{
|
|
if (v.is_phandle())
|
|
{
|
|
return node_names[v.string_data];
|
|
}
|
|
if (v.is_binary())
|
|
{
|
|
return used_phandles[v.get_as_uint32()];
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
device_tree::write_binary(int fd)
|
|
{
|
|
write<dtb::binary_writer>(fd);
|
|
}
|
|
|
|
void
|
|
device_tree::write_asm(int fd)
|
|
{
|
|
write<dtb::asm_writer>(fd);
|
|
}
|
|
|
|
void
|
|
device_tree::write_dts(int fd)
|
|
{
|
|
FILE *file = fdopen(fd, "w");
|
|
fputs("/dtc-v1/;\n\n", file);
|
|
|
|
if (!reservations.empty())
|
|
{
|
|
const char msg[] = "/memreserve/";
|
|
fwrite(msg, sizeof(msg), 1, file);
|
|
for (std::vector<reservation>::iterator i=reservations.begin(),
|
|
e=reservations.end() ; i!=e ; ++i)
|
|
{
|
|
fprintf(stderr, " %" PRIx64 " %" PRIx64, i->first, i->second);
|
|
}
|
|
fputs(";\n\n", file);
|
|
}
|
|
putc('/', file);
|
|
putc(' ', file);
|
|
root->write_dts(file, 0);
|
|
fclose(file);
|
|
}
|
|
|
|
void
|
|
device_tree::parse_dtb(const char *fn, FILE *depfile)
|
|
{
|
|
input_buffer *in = buffer_for_file(fn);
|
|
if (in == 0)
|
|
{
|
|
valid = false;
|
|
return;
|
|
}
|
|
input_buffer &input = *in;
|
|
dtb::header h;
|
|
valid = h.read_dtb(input);
|
|
boot_cpu = h.boot_cpuid_phys;
|
|
if (h.last_comp_version > 17)
|
|
{
|
|
fprintf(stderr, "Don't know how to read this version of the device tree blob");
|
|
valid = false;
|
|
}
|
|
if (!valid)
|
|
{
|
|
return;
|
|
}
|
|
input_buffer reservation_map =
|
|
input.buffer_from_offset(h.off_mem_rsvmap, 0);
|
|
uint64_t start, length;
|
|
do
|
|
{
|
|
if (!(reservation_map.consume_binary(start) &&
|
|
reservation_map.consume_binary(length)))
|
|
{
|
|
fprintf(stderr, "Failed to read memory reservation table\n");
|
|
valid = false;
|
|
return;
|
|
}
|
|
} while (!((start == 0) && (length == 0)));
|
|
input_buffer struct_table =
|
|
input.buffer_from_offset(h.off_dt_struct, h.size_dt_struct);
|
|
input_buffer strings_table =
|
|
input.buffer_from_offset(h.off_dt_strings, h.size_dt_strings);
|
|
uint32_t token;
|
|
if (!(struct_table.consume_binary(token) &&
|
|
(token == dtb::FDT_BEGIN_NODE)))
|
|
{
|
|
fprintf(stderr, "Expected FDT_BEGIN_NODE token.\n");
|
|
valid = false;
|
|
return;
|
|
}
|
|
root = node::parse_dtb(struct_table, strings_table);
|
|
if (!(struct_table.consume_binary(token) && (token == dtb::FDT_END)))
|
|
{
|
|
fprintf(stderr, "Expected FDT_END token after parsing root node.\n");
|
|
valid = false;
|
|
return;
|
|
}
|
|
valid = (root != 0);
|
|
}
|
|
|
|
void
|
|
device_tree::parse_dts(const char *fn, FILE *depfile)
|
|
{
|
|
input_buffer *in = buffer_for_file(fn);
|
|
if (in == 0)
|
|
{
|
|
valid = false;
|
|
return;
|
|
}
|
|
std::vector<node*> roots;
|
|
input_buffer &input = *in;
|
|
input.next_token();
|
|
bool read_header = false;
|
|
// Read the header
|
|
if (input.consume("/dts-v1/;"))
|
|
{
|
|
read_header = true;
|
|
}
|
|
input.next_token();
|
|
while(input.consume("/include/"))
|
|
{
|
|
input.next_token();
|
|
if (!input.consume('"'))
|
|
{
|
|
input.parse_error("Expected quoted filename");
|
|
valid = false;
|
|
return;
|
|
}
|
|
int length = 0;
|
|
while (input[length] != '"') length++;
|
|
|
|
const char *file = (const char*)input;
|
|
const char *dir = dirname(fn);
|
|
int dir_length = strlen(dir);
|
|
char *include_file = (char*)malloc(strlen(dir) + length + 2);
|
|
memcpy(include_file, dir, dir_length);
|
|
include_file[dir_length] = '/';
|
|
memcpy(include_file+dir_length+1, file, length);
|
|
include_file[dir_length+length+1] = 0;
|
|
input_buffer *include_buffer = buffer_for_file(include_file);
|
|
|
|
if (include_buffer == 0)
|
|
{
|
|
for (std::vector<const char*>::iterator i=include_paths.begin(), e=include_paths.end() ; e!=i ; ++i)
|
|
{
|
|
free(include_file);
|
|
dir = *i;
|
|
dir_length = strlen(dir);
|
|
include_file = (char*)malloc(strlen(dir) +
|
|
length + 2);
|
|
memcpy(include_file, dir, dir_length);
|
|
include_file[dir_length] = '/';
|
|
memcpy(include_file+dir_length+1, file, length);
|
|
include_file[dir_length+length+1] = 0;
|
|
include_buffer = buffer_for_file(include_file);
|
|
if (include_buffer != 0)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (depfile != 0)
|
|
{
|
|
putc(' ', depfile);
|
|
fputs(include_file, depfile);
|
|
}
|
|
if (include_buffer == 0)
|
|
{
|
|
valid = false;
|
|
return;
|
|
}
|
|
input_buffer &include = *include_buffer;
|
|
input.consume(include_file+dir_length+1);
|
|
input.consume('"');
|
|
free((void*)include_file);
|
|
|
|
if (!read_header)
|
|
{
|
|
include.next_token();
|
|
read_header = include.consume("/dts-v1/;");
|
|
}
|
|
parse_roots(include, roots);
|
|
}
|
|
input.next_token();
|
|
if (!read_header)
|
|
{
|
|
input.parse_error("Expected /dts-v1/; version string");
|
|
}
|
|
// Read any memory reservations
|
|
while(input.consume("/memreserve/"))
|
|
{
|
|
long long start, len;
|
|
input.next_token();
|
|
// Read the start and length.
|
|
if (!(input.consume_integer(start) &&
|
|
(input.next_token(),
|
|
input.consume_integer(len))))
|
|
{
|
|
input.parse_error("Expected /dts-v1/; version string");
|
|
}
|
|
input.next_token();
|
|
input.consume(';');
|
|
reservations.push_back(reservation(start, len));
|
|
}
|
|
parse_roots(input, roots);
|
|
switch (roots.size())
|
|
{
|
|
case 0:
|
|
valid = false;
|
|
input.parse_error("Failed to find root node /.");
|
|
return;
|
|
case 1:
|
|
root = roots[0];
|
|
break;
|
|
default:
|
|
{
|
|
root = roots[0];
|
|
for (std::vector<node*>::iterator i=roots.begin()+1,
|
|
e=roots.end() ; i!=e ; ++i)
|
|
{
|
|
root->merge_node(*i);
|
|
delete *i;
|
|
}
|
|
roots.resize(1);
|
|
}
|
|
}
|
|
collect_names();
|
|
resolve_cross_references();
|
|
}
|
|
|
|
device_tree::~device_tree()
|
|
{
|
|
if (root != 0)
|
|
{
|
|
delete root;
|
|
}
|
|
while (!buffers.empty())
|
|
{
|
|
delete buffers.back();
|
|
buffers.pop_back();
|
|
}
|
|
}
|
|
|
|
} // namespace fdt
|
|
|
|
} // namespace dtc
|
|
|