2018-04-12 11:44:43 +02:00
|
|
|
/*
|
|
|
|
* Copyright 2013 Google Inc.
|
2018-04-03 09:57:33 +02:00
|
|
|
* Copyright 2018-present Facebook, Inc.
|
2018-04-12 11:44:43 +02:00
|
|
|
*
|
2018-04-03 09:57:33 +02:00
|
|
|
* Taken from depthcharge: src/base/device_tree.h
|
2018-04-12 11:44:43 +02:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
#ifndef __DEVICE_TREE_H__
|
|
|
|
#define __DEVICE_TREE_H__
|
2018-04-12 11:44:43 +02:00
|
|
|
|
|
|
|
#include <stdint.h>
|
2018-04-03 09:57:33 +02:00
|
|
|
#include <list.h>
|
2018-04-12 11:44:43 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Flattened device tree structures/constants.
|
|
|
|
*/
|
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct fdt_header {
|
2018-04-12 11:44:43 +02:00
|
|
|
uint32_t magic;
|
|
|
|
uint32_t totalsize;
|
|
|
|
uint32_t structure_offset;
|
|
|
|
uint32_t strings_offset;
|
|
|
|
uint32_t reserve_map_offset;
|
|
|
|
|
|
|
|
uint32_t version;
|
|
|
|
uint32_t last_compatible_version;
|
|
|
|
|
|
|
|
uint32_t boot_cpuid_phys;
|
|
|
|
|
|
|
|
uint32_t strings_size;
|
|
|
|
uint32_t structure_size;
|
2018-04-03 09:57:33 +02:00
|
|
|
};
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
#define FDT_HEADER_MAGIC 0xd00dfeed
|
|
|
|
#define FDT_TOKEN_BEGIN_NODE 1
|
|
|
|
#define FDT_TOKEN_END_NODE 2
|
|
|
|
#define FDT_TOKEN_PROPERTY 3
|
|
|
|
#define FDT_TOKEN_END 9
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct fdt_property
|
2018-04-12 11:44:43 +02:00
|
|
|
{
|
|
|
|
const char *name;
|
2018-08-22 09:55:15 +02:00
|
|
|
const void *data;
|
2018-04-12 11:44:43 +02:00
|
|
|
uint32_t size;
|
2018-04-03 09:57:33 +02:00
|
|
|
};
|
2018-04-12 11:44:43 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Unflattened device tree structures.
|
|
|
|
*/
|
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_property
|
2018-04-12 11:44:43 +02:00
|
|
|
{
|
2018-04-03 09:57:33 +02:00
|
|
|
struct fdt_property prop;
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct list_node list_node;
|
|
|
|
};
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_node
|
2018-04-12 11:44:43 +02:00
|
|
|
{
|
|
|
|
const char *name;
|
2018-04-03 09:57:33 +02:00
|
|
|
// List of struct device_tree_property-s.
|
|
|
|
struct list_node properties;
|
|
|
|
// List of struct device_tree_nodes.
|
|
|
|
struct list_node children;
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct list_node list_node;
|
|
|
|
};
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_reserve_map_entry
|
2018-04-12 11:44:43 +02:00
|
|
|
{
|
|
|
|
uint64_t start;
|
|
|
|
uint64_t size;
|
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct list_node list_node;
|
|
|
|
};
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree
|
2018-04-12 11:44:43 +02:00
|
|
|
{
|
2018-08-22 09:55:15 +02:00
|
|
|
const void *header;
|
2018-04-12 11:44:43 +02:00
|
|
|
uint32_t header_size;
|
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct list_node reserve_map;
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_node *root;
|
|
|
|
};
|
2018-04-12 11:44:43 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Flattened device tree functions. These generally return the number of bytes
|
|
|
|
* which were consumed reading the requested value.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Read the property, if any, at offset offset.
|
2018-08-22 09:55:15 +02:00
|
|
|
int fdt_next_property(const void *blob, uint32_t offset,
|
|
|
|
struct fdt_property *prop);
|
2018-04-12 11:44:43 +02:00
|
|
|
// Read the name of the node, if any, at offset offset.
|
2018-08-22 09:55:15 +02:00
|
|
|
int fdt_node_name(const void *blob, uint32_t offset, const char **name);
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-08-22 09:55:15 +02:00
|
|
|
void fdt_print_node(const void *blob, uint32_t offset);
|
|
|
|
int fdt_skip_node(const void *blob, uint32_t offset);
|
2018-04-12 11:44:43 +02:00
|
|
|
|
|
|
|
// Read a flattened device tree into a heirarchical structure which refers to
|
|
|
|
// the contents of the flattened tree in place. Modifying the flat tree
|
|
|
|
// invalidates the unflattened one.
|
2018-08-22 09:55:15 +02:00
|
|
|
struct device_tree *fdt_unflatten(const void *blob);
|
2018-04-12 11:44:43 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Unflattened device tree functions.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Figure out how big a device tree would be if it were flattened.
|
2018-08-22 09:55:15 +02:00
|
|
|
uint32_t dt_flat_size(const struct device_tree *tree);
|
2018-04-12 11:44:43 +02:00
|
|
|
// Flatten a device tree into the buffer pointed to by dest.
|
2018-08-22 09:55:15 +02:00
|
|
|
void dt_flatten(const struct device_tree *tree, void *dest);
|
|
|
|
void dt_print_node(const struct device_tree_node *node);
|
2018-04-12 11:44:43 +02:00
|
|
|
// Read #address-cells and #size-cells properties from a node.
|
2018-08-22 09:55:15 +02:00
|
|
|
void dt_read_cell_props(const struct device_tree_node *node, u32 *addrcp,
|
|
|
|
u32 *sizecp);
|
2018-04-12 11:44:43 +02:00
|
|
|
// Look up or create a node relative to a parent node, through its path
|
|
|
|
// represented as an array of strings.
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_node *dt_find_node(struct device_tree_node *parent, const char **path,
|
2018-04-12 11:44:43 +02:00
|
|
|
u32 *addrcp, u32 *sizecp, int create);
|
|
|
|
// Look up or create a node relative to a parent node, through its path
|
|
|
|
// represented as a string of '/' separated node names.
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_node *dt_find_node_by_path(struct device_tree_node *parent, const char *path,
|
2018-04-12 11:44:43 +02:00
|
|
|
u32 *addrcp, u32 *sizecp, int create);
|
|
|
|
// Look up a node relative to a parent node, through its compatible string.
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_node *dt_find_compat(struct device_tree_node *parent, const char *compatible);
|
2018-04-12 11:44:43 +02:00
|
|
|
// Look up the next child of a parent node, through its compatible string. It
|
|
|
|
// uses child pointer as the marker to find next.
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_node *dt_find_next_compat_child(struct device_tree_node *parent,
|
|
|
|
struct device_tree_node *child,
|
2018-04-12 11:44:43 +02:00
|
|
|
const char *compat);
|
|
|
|
// Look up a node relative to a parent node, through its property value.
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_node *dt_find_prop_value(struct device_tree_node *parent, const char *name,
|
2018-04-12 11:44:43 +02:00
|
|
|
void *data, size_t size);
|
2018-05-08 11:18:54 +02:00
|
|
|
// Return the phandle
|
2018-08-22 09:55:15 +02:00
|
|
|
uint32_t dt_get_phandle(const struct device_tree_node *node);
|
2018-04-12 11:44:43 +02:00
|
|
|
// Write src into *dest as a 'length'-byte big-endian integer.
|
|
|
|
void dt_write_int(u8 *dest, u64 src, size_t length);
|
2018-05-30 15:05:28 +02:00
|
|
|
// Delete a property
|
|
|
|
void dt_delete_prop(struct device_tree_node *node, const char *name);
|
2018-04-12 11:44:43 +02:00
|
|
|
// Add different kinds of properties to a node, or update existing ones.
|
2018-08-22 09:55:15 +02:00
|
|
|
void dt_add_bin_prop(struct device_tree_node *node, const char *name,
|
|
|
|
const void *data, size_t size);
|
|
|
|
void dt_add_string_prop(struct device_tree_node *node, const char *name,
|
|
|
|
const char *str);
|
2018-04-03 09:57:33 +02:00
|
|
|
void dt_add_u32_prop(struct device_tree_node *node, const char *name, u32 val);
|
2018-08-10 10:12:35 +02:00
|
|
|
void dt_add_u64_prop(struct device_tree_node *node, const char *name, u64 val);
|
2018-04-03 09:57:33 +02:00
|
|
|
void dt_add_reg_prop(struct device_tree_node *node, u64 *addrs, u64 *sizes,
|
2018-04-12 11:44:43 +02:00
|
|
|
int count, u32 addr_cells, u32 size_cells);
|
2018-04-03 09:57:33 +02:00
|
|
|
int dt_set_bin_prop_by_path(struct device_tree *tree, const char *path,
|
2018-04-12 11:44:43 +02:00
|
|
|
void *data, size_t size, int create);
|
|
|
|
|
2018-08-22 09:55:15 +02:00
|
|
|
void dt_find_bin_prop(const struct device_tree_node *node, const char *name,
|
|
|
|
const void **data, size_t *size);
|
|
|
|
const char *dt_find_string_prop(const struct device_tree_node *node,
|
|
|
|
const char *name);
|
2018-04-12 11:44:43 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Fixups to apply to a kernel's device tree before booting it.
|
|
|
|
*/
|
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_fixup
|
2018-04-12 11:44:43 +02:00
|
|
|
{
|
2018-04-03 09:57:33 +02:00
|
|
|
/**
|
|
|
|
* The function which does the fixing.
|
|
|
|
* 0 on success, non-zero on error.
|
|
|
|
*/
|
|
|
|
int (*fixup)(struct device_tree_fixup *fixup,
|
|
|
|
struct device_tree *tree);
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
struct list_node list_node;
|
|
|
|
};
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
extern struct list_node device_tree_fixups;
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
/**
|
|
|
|
* Function to apply fixups.
|
|
|
|
* 0 on success, non-zero on error.
|
2018-04-12 11:44:43 +02:00
|
|
|
*/
|
2018-04-03 09:57:33 +02:00
|
|
|
int dt_apply_fixups(struct device_tree *tree);
|
2018-04-12 11:44:43 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Init/retrieve the /reserved-memory/ node.
|
|
|
|
*/
|
2018-04-03 09:57:33 +02:00
|
|
|
struct device_tree_node *dt_init_reserved_memory_node(struct device_tree *tree);
|
2018-04-12 11:44:43 +02:00
|
|
|
|
2018-04-03 09:57:33 +02:00
|
|
|
#endif /* __DEVICE_TREE_H__ */
|