2019-06-21 08:18:32 -06:00
/* SPDX-License-Identifier: GPL-2.0-or-later */
2018-02-27 17:40:38 -06:00
# ifndef DTC_H
# define DTC_H
2007-12-18 15:06:42 +11:00
/*
* ( C ) Copyright David Gibson < dwg @ au1 . ibm . com > , IBM Corporation . 2005.
*/
# include <stdio.h>
# include <string.h>
# include <stdlib.h>
# include <stdint.h>
2012-09-28 21:25:59 +00:00
# include <stdbool.h>
2007-12-18 15:06:42 +11:00
# include <stdarg.h>
# include <assert.h>
# include <ctype.h>
# include <errno.h>
# include <unistd.h>
2017-10-03 11:37:04 -05:00
# include <inttypes.h>
2007-12-18 15:06:42 +11:00
2008-08-07 12:24:17 +10:00
# include <libfdt_env.h>
2007-12-18 15:06:42 +11:00
# include <fdt.h>
2010-11-17 15:28:20 -08:00
# include "util.h"
# ifdef DEBUG
2015-04-29 16:00:05 -05:00
# define debug(...) printf(__VA_ARGS__)
2010-11-17 15:28:20 -08:00
# else
2015-04-29 16:00:05 -05:00
# define debug(...)
2010-11-17 15:28:20 -08:00
# endif
2007-12-18 15:06:42 +11:00
# define DEFAULT_FDT_VERSION 17
2010-11-17 15:28:20 -08:00
2007-12-18 15:06:42 +11:00
/*
* Command line options
*/
extern int quiet ; /* Level of quietness */
2021-10-25 11:05:45 -05:00
extern unsigned int reservenum ; /* Number of memory reservation slots */
2007-12-18 15:06:42 +11:00
extern int minsize ; /* Minimum blob size */
extern int padsize ; /* Additional padding to blob */
2017-01-04 10:45:20 -06:00
extern int alignsize ; /* Additional padding to blob accroding to the alignsize */
2010-11-17 15:28:20 -08:00
extern int phandle_format ; /* Use linux,phandle or phandle properties */
2017-01-04 10:45:20 -06:00
extern int generate_symbols ; /* generate symbols for nodes with labels */
extern int generate_fixups ; /* generate fixups */
extern int auto_label_aliases ; /* auto generate labels -> aliases */
2018-11-28 18:37:35 -06:00
extern int annotate ; /* annotate .dts with input source location */
2007-12-18 15:06:42 +11:00
2010-11-17 15:28:20 -08:00
# define PHANDLE_LEGACY 0x1
# define PHANDLE_EPAPR 0x2
# define PHANDLE_BOTH 0x3
2007-12-18 15:06:42 +11:00
2008-08-07 12:24:17 +10:00
typedef uint32_t cell_t ;
2007-12-18 15:06:42 +11:00
2021-10-25 11:05:45 -05:00
static inline bool phandle_is_valid ( cell_t phandle )
{
return phandle ! = 0 & & phandle ! = ~ 0U ;
}
2020-06-29 12:15:13 -06:00
static inline uint16_t dtb_ld16 ( const void * p )
{
const uint8_t * bp = ( const uint8_t * ) p ;
return ( ( uint16_t ) bp [ 0 ] < < 8 )
| bp [ 1 ] ;
}
static inline uint32_t dtb_ld32 ( const void * p )
{
const uint8_t * bp = ( const uint8_t * ) p ;
return ( ( uint32_t ) bp [ 0 ] < < 24 )
| ( ( uint32_t ) bp [ 1 ] < < 16 )
| ( ( uint32_t ) bp [ 2 ] < < 8 )
| bp [ 3 ] ;
}
static inline uint64_t dtb_ld64 ( const void * p )
{
const uint8_t * bp = ( const uint8_t * ) p ;
return ( ( uint64_t ) bp [ 0 ] < < 56 )
| ( ( uint64_t ) bp [ 1 ] < < 48 )
| ( ( uint64_t ) bp [ 2 ] < < 40 )
| ( ( uint64_t ) bp [ 3 ] < < 32 )
| ( ( uint64_t ) bp [ 4 ] < < 24 )
| ( ( uint64_t ) bp [ 5 ] < < 16 )
| ( ( uint64_t ) bp [ 6 ] < < 8 )
| bp [ 7 ] ;
}
2007-12-18 15:06:42 +11:00
# define streq(a, b) (strcmp((a), (b)) == 0)
2018-02-27 17:40:38 -06:00
# define strstarts(s, prefix) (strncmp((s), (prefix), strlen(prefix)) == 0)
# define strprefixeq(a, n, b) (strlen(b) == (n) && (memcmp(a, b, n) == 0))
2021-10-25 11:05:45 -05:00
static inline bool strends ( const char * str , const char * suffix )
{
unsigned int len , suffix_len ;
len = strlen ( str ) ;
suffix_len = strlen ( suffix ) ;
if ( len < suffix_len )
return false ;
return streq ( str + len - suffix_len , suffix ) ;
}
2007-12-18 15:06:42 +11:00
# define ALIGN(x, a) (((x) + (a) - 1) & ~((a) - 1))
/* Data blobs */
enum markertype {
2018-09-13 08:59:25 -05:00
TYPE_NONE ,
2007-12-18 15:06:42 +11:00
REF_PHANDLE ,
REF_PATH ,
LABEL ,
2018-09-13 08:59:25 -05:00
TYPE_UINT8 ,
TYPE_UINT16 ,
TYPE_UINT32 ,
TYPE_UINT64 ,
TYPE_STRING ,
2007-12-18 15:06:42 +11:00
} ;
2021-10-25 11:05:45 -05:00
static inline bool is_type_marker ( enum markertype type )
{
return type > = TYPE_UINT8 ;
}
2018-09-13 08:59:25 -05:00
extern const char * markername ( enum markertype markertype ) ;
2007-12-18 15:06:42 +11:00
struct marker {
enum markertype type ;
2021-02-03 15:26:03 -06:00
unsigned int offset ;
2007-12-18 15:06:42 +11:00
char * ref ;
struct marker * next ;
} ;
struct data {
2021-02-03 15:26:03 -06:00
unsigned int len ;
2007-12-18 15:06:42 +11:00
char * val ;
struct marker * markers ;
} ;
2015-04-29 16:00:05 -05:00
# define empty_data ((struct data){ 0 /* all .members = 0 or NULL */ })
2007-12-18 15:06:42 +11:00
# define for_each_marker(m) \
for ( ; ( m ) ; ( m ) = ( m ) - > next )
# define for_each_marker_of_type(m, t) \
for_each_marker ( m ) \
if ( ( m ) - > type = = ( t ) )
2021-10-25 11:05:45 -05:00
static inline struct marker * next_type_marker ( struct marker * m )
{
for_each_marker ( m )
if ( is_type_marker ( m - > type ) )
break ;
return m ;
}
static inline size_t type_marker_length ( struct marker * m )
{
struct marker * next = next_type_marker ( m - > next ) ;
if ( next )
return next - > offset - m - > offset ;
return 0 ;
}
2018-09-13 08:59:25 -05:00
2007-12-18 15:06:42 +11:00
void data_free ( struct data d ) ;
2021-02-03 15:26:03 -06:00
struct data data_grow_for ( struct data d , unsigned int xlen ) ;
2007-12-18 15:06:42 +11:00
struct data data_copy_mem ( const char * mem , int len ) ;
struct data data_copy_escape_string ( const char * s , int len ) ;
struct data data_copy_file ( FILE * f , size_t len ) ;
struct data data_append_data ( struct data d , const void * p , int len ) ;
struct data data_insert_at_marker ( struct data d , struct marker * m ,
const void * p , int len ) ;
struct data data_merge ( struct data d1 , struct data d2 ) ;
struct data data_append_cell ( struct data d , cell_t word ) ;
2012-09-28 21:25:59 +00:00
struct data data_append_integer ( struct data d , uint64_t word , int bits ) ;
2017-03-21 09:01:08 -05:00
struct data data_append_re ( struct data d , uint64_t address , uint64_t size ) ;
2008-08-07 12:24:17 +10:00
struct data data_append_addr ( struct data d , uint64_t addr ) ;
2007-12-18 15:06:42 +11:00
struct data data_append_byte ( struct data d , uint8_t byte ) ;
struct data data_append_zeroes ( struct data d , int len ) ;
struct data data_append_align ( struct data d , int align ) ;
struct data data_add_marker ( struct data d , enum markertype type , char * ref ) ;
2015-04-29 16:00:05 -05:00
bool data_is_one_string ( struct data d ) ;
2007-12-18 15:06:42 +11:00
/* DT constraints */
# define MAX_PROPNAME_LEN 31
# define MAX_NODENAME_LEN 31
/* Live trees */
2010-11-17 15:28:20 -08:00
struct label {
2015-04-29 16:00:05 -05:00
bool deleted ;
2010-11-17 15:28:20 -08:00
char * label ;
struct label * next ;
} ;
2017-03-21 09:01:08 -05:00
struct bus_type {
const char * name ;
} ;
2007-12-18 15:06:42 +11:00
struct property {
2015-04-29 16:00:05 -05:00
bool deleted ;
2007-12-18 15:06:42 +11:00
char * name ;
struct data val ;
struct property * next ;
2010-11-17 15:28:20 -08:00
struct label * labels ;
2018-11-28 18:37:35 -06:00
struct srcpos * srcpos ;
2007-12-18 15:06:42 +11:00
} ;
struct node {
2015-04-29 16:00:05 -05:00
bool deleted ;
2007-12-18 15:06:42 +11:00
char * name ;
struct property * proplist ;
struct node * children ;
struct node * parent ;
struct node * next_sibling ;
char * fullpath ;
int basenamelen ;
cell_t phandle ;
int addr_cells , size_cells ;
2010-11-17 15:28:20 -08:00
struct label * labels ;
2017-03-21 09:01:08 -05:00
const struct bus_type * bus ;
2018-11-28 18:37:35 -06:00
struct srcpos * srcpos ;
2018-05-08 13:07:49 -05:00
bool omit_if_unused , is_referenced ;
2007-12-18 15:06:42 +11:00
} ;
2012-09-28 21:25:59 +00:00
# define for_each_label_withdel(l0, l) \
2010-11-17 15:28:20 -08:00
for ( ( l ) = ( l0 ) ; ( l ) ; ( l ) = ( l ) - > next )
2012-10-08 16:15:26 -06:00
# define for_each_label(l0, l) \
for_each_label_withdel ( l0 , l ) \
if ( ! ( l ) - > deleted )
2012-09-28 21:25:59 +00:00
# define for_each_property_withdel(n, p) \
2007-12-18 15:06:42 +11:00
for ( ( p ) = ( n ) - > proplist ; ( p ) ; ( p ) = ( p ) - > next )
2012-10-08 16:15:26 -06:00
# define for_each_property(n, p) \
for_each_property_withdel ( n , p ) \
if ( ! ( p ) - > deleted )
2012-09-28 21:25:59 +00:00
# define for_each_child_withdel(n, c) \
2007-12-18 15:06:42 +11:00
for ( ( c ) = ( n ) - > children ; ( c ) ; ( c ) = ( c ) - > next_sibling )
2012-10-08 16:15:26 -06:00
# define for_each_child(n, c) \
for_each_child_withdel ( n , c ) \
if ( ! ( c ) - > deleted )
2010-11-17 15:28:20 -08:00
void add_label ( struct label * * labels , char * label ) ;
2012-09-28 21:25:59 +00:00
void delete_labels ( struct label * * labels ) ;
2010-11-17 15:28:20 -08:00
2018-11-28 18:37:35 -06:00
struct property * build_property ( char * name , struct data val ,
struct srcpos * srcpos ) ;
2012-09-28 21:25:59 +00:00
struct property * build_property_delete ( char * name ) ;
2007-12-18 15:06:42 +11:00
struct property * chain_property ( struct property * first , struct property * list ) ;
struct property * reverse_properties ( struct property * first ) ;
2018-11-28 18:37:35 -06:00
struct node * build_node ( struct property * proplist , struct node * children ,
struct srcpos * srcpos ) ;
struct node * build_node_delete ( struct srcpos * srcpos ) ;
2010-11-17 15:28:20 -08:00
struct node * name_node ( struct node * node , char * name ) ;
2018-05-08 13:07:49 -05:00
struct node * omit_node_if_unused ( struct node * node ) ;
struct node * reference_node ( struct node * node ) ;
2007-12-18 15:06:42 +11:00
struct node * chain_node ( struct node * first , struct node * list ) ;
2010-11-17 15:28:20 -08:00
struct node * merge_nodes ( struct node * old_node , struct node * new_node ) ;
2018-02-27 17:40:38 -06:00
struct node * add_orphan_node ( struct node * old_node , struct node * new_node , char * ref ) ;
2007-12-18 15:06:42 +11:00
void add_property ( struct node * node , struct property * prop ) ;
2012-09-28 21:25:59 +00:00
void delete_property_by_name ( struct node * node , char * name ) ;
void delete_property ( struct property * prop ) ;
2007-12-18 15:06:42 +11:00
void add_child ( struct node * parent , struct node * child ) ;
2012-09-28 21:25:59 +00:00
void delete_node_by_name ( struct node * parent , char * name ) ;
void delete_node ( struct node * node ) ;
2017-01-04 10:45:20 -06:00
void append_to_property ( struct node * node ,
2019-06-12 07:05:52 -06:00
char * name , const void * data , int len ,
enum markertype type ) ;
2007-12-18 15:06:42 +11:00
const char * get_unitname ( struct node * node ) ;
struct property * get_property ( struct node * node , const char * propname ) ;
cell_t propval_cell ( struct property * prop ) ;
2021-02-03 15:26:03 -06:00
cell_t propval_cell_n ( struct property * prop , unsigned int n ) ;
2010-11-17 15:28:20 -08:00
struct property * get_property_by_label ( struct node * tree , const char * label ,
struct node * * node ) ;
struct marker * get_marker_label ( struct node * tree , const char * label ,
struct node * * node , struct property * * prop ) ;
2007-12-18 15:06:42 +11:00
struct node * get_subnode ( struct node * node , const char * nodename ) ;
struct node * get_node_by_path ( struct node * tree , const char * path ) ;
struct node * get_node_by_label ( struct node * tree , const char * label ) ;
struct node * get_node_by_phandle ( struct node * tree , cell_t phandle ) ;
struct node * get_node_by_ref ( struct node * tree , const char * ref ) ;
cell_t get_node_phandle ( struct node * root , struct node * node ) ;
2010-11-17 15:28:20 -08:00
uint32_t guess_boot_cpuid ( struct node * tree ) ;
2007-12-18 15:06:42 +11:00
/* Boot info (tree plus memreserve information */
struct reserve_info {
2017-03-21 09:01:08 -05:00
uint64_t address , size ;
2007-12-18 15:06:42 +11:00
struct reserve_info * next ;
2010-11-17 15:28:20 -08:00
struct label * labels ;
2007-12-18 15:06:42 +11:00
} ;
2010-11-17 15:28:20 -08:00
struct reserve_info * build_reserve_entry ( uint64_t start , uint64_t len ) ;
2007-12-18 15:06:42 +11:00
struct reserve_info * chain_reserve_entry ( struct reserve_info * first ,
struct reserve_info * list ) ;
struct reserve_info * add_reserve_entry ( struct reserve_info * list ,
struct reserve_info * new ) ;
2017-01-04 10:45:20 -06:00
struct dt_info {
unsigned int dtsflags ;
2007-12-18 15:06:42 +11:00
struct reserve_info * reservelist ;
2008-08-07 12:24:17 +10:00
uint32_t boot_cpuid_phys ;
2017-01-04 10:45:20 -06:00
struct node * dt ; /* the device tree */
2017-03-21 09:01:08 -05:00
const char * outname ; /* filename being written to, "-" for stdout */
2007-12-18 15:06:42 +11:00
} ;
2017-01-04 10:45:20 -06:00
/* DTS version flags definitions */
# define DTSF_V1 0x0001 /* /dts-v1/ */
# define DTSF_PLUGIN 0x0002 /* /plugin/ */
struct dt_info * build_dt_info ( unsigned int dtsflags ,
struct reserve_info * reservelist ,
struct node * tree , uint32_t boot_cpuid_phys ) ;
void sort_tree ( struct dt_info * dti ) ;
void generate_label_tree ( struct dt_info * dti , char * name , bool allocph ) ;
void generate_fixups_tree ( struct dt_info * dti , char * name ) ;
void generate_local_fixups_tree ( struct dt_info * dti , char * name ) ;
2007-12-18 15:06:42 +11:00
/* Checks */
2015-04-29 16:00:05 -05:00
void parse_checks_option ( bool warn , bool error , const char * arg ) ;
2017-01-04 10:45:20 -06:00
void process_checks ( bool force , struct dt_info * dti ) ;
2007-12-18 15:06:42 +11:00
/* Flattened trees */
2017-01-04 10:45:20 -06:00
void dt_to_blob ( FILE * f , struct dt_info * dti , int version ) ;
void dt_to_asm ( FILE * f , struct dt_info * dti , int version ) ;
2007-12-18 15:06:42 +11:00
2017-01-04 10:45:20 -06:00
struct dt_info * dt_from_blob ( const char * fname ) ;
2007-12-18 15:06:42 +11:00
/* Tree source */
2017-01-04 10:45:20 -06:00
void dt_to_source ( FILE * f , struct dt_info * dti ) ;
struct dt_info * dt_from_source ( const char * f ) ;
2007-12-18 15:06:42 +11:00
2018-09-13 08:59:25 -05:00
/* YAML source */
void dt_to_yaml ( FILE * f , struct dt_info * dti ) ;
2007-12-18 15:06:42 +11:00
/* FS trees */
2017-01-04 10:45:20 -06:00
struct dt_info * dt_from_fs ( const char * dirname ) ;
2007-12-18 15:06:42 +11:00
2018-02-27 17:40:38 -06:00
# endif /* DTC_H */