2019-06-21 17:18:32 +03:00
/* SPDX-License-Identifier: GPL-2.0-or-later */
2018-02-28 02:40:38 +03:00
# ifndef UTIL_H
# define UTIL_H
2010-11-18 02:28:20 +03:00
2020-10-12 17:58:15 +03:00
# include <stdlib.h>
2012-09-29 01:25:59 +04:00
# include <stdarg.h>
2015-04-30 00:00:05 +03:00
# include <stdbool.h>
2014-01-21 16:54:49 +04:00
# include <getopt.h>
2012-09-29 01:25:59 +04:00
2010-11-18 02:28:20 +03:00
/*
2012-09-29 01:25:59 +04:00
* Copyright 2011 The Chromium Authors , All Rights Reserved .
2010-11-18 02:28:20 +03:00
* Copyright 2008 Jon Loeliger , Freescale Semiconductor , Inc .
*/
2017-03-21 17:01:08 +03:00
# ifdef __GNUC__
2021-10-25 19:05:45 +03:00
# if __GNUC__ >= 5 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 4)
2019-12-27 01:36:47 +03:00
# define PRINTF(i, j) __attribute__((format (gnu_printf, i, j)))
2021-10-25 19:05:45 +03:00
# else
# define PRINTF(i, j) __attribute__((format (printf, i, j)))
2019-12-27 01:36:47 +03:00
# endif
2017-03-21 17:01:08 +03:00
# define NORETURN __attribute__((noreturn))
# else
# define PRINTF(i, j)
# define NORETURN
# endif
2014-01-21 16:54:49 +04:00
# define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))
2018-02-28 02:40:38 +03:00
# define stringify(s) stringify_(s)
# define stringify_(s) #s
2017-03-21 17:01:08 +03:00
static inline void NORETURN PRINTF ( 1 , 2 ) die ( const char * str , . . . )
2010-11-18 02:28:20 +03:00
{
va_list ap ;
va_start ( ap , str ) ;
fprintf ( stderr , " FATAL ERROR: " ) ;
vfprintf ( stderr , str , ap ) ;
2015-04-30 00:00:05 +03:00
va_end ( ap ) ;
2010-11-18 02:28:20 +03:00
exit ( 1 ) ;
}
static inline void * xmalloc ( size_t len )
{
void * new = malloc ( len ) ;
if ( ! new )
die ( " malloc() failed \n " ) ;
return new ;
}
static inline void * xrealloc ( void * p , size_t len )
{
void * new = realloc ( p , len ) ;
if ( ! new )
2017-03-21 17:01:08 +03:00
die ( " realloc() failed (len=%zd) \n " , len ) ;
2010-11-18 02:28:20 +03:00
return new ;
}
extern char * xstrdup ( const char * s ) ;
2022-11-01 20:54:44 +03:00
extern char * xstrndup ( const char * s , size_t len ) ;
2017-03-21 17:01:08 +03:00
extern int PRINTF ( 2 , 3 ) xasprintf ( char * * strp , const char * fmt , . . . ) ;
2018-11-29 03:37:35 +03:00
extern int PRINTF ( 2 , 3 ) xasprintf_append ( char * * strp , const char * fmt , . . . ) ;
extern int xavsprintf_append ( char * * strp , const char * fmt , va_list ap ) ;
2010-11-18 02:28:20 +03:00
extern char * join_path ( const char * path , const char * name ) ;
2012-09-29 01:25:59 +04:00
/**
2014-01-21 16:54:49 +04:00
* Check a property of a given length to see if it is all printable and
* has a valid terminator . The property can contain either a single string ,
* or multiple strings each of non - zero length .
2012-09-29 01:25:59 +04:00
*
* @ param data The string to check
* @ param len The string length including terminator
2014-01-21 16:54:49 +04:00
* @ return 1 if a valid printable string , 0 if not
*/
2015-04-30 00:00:05 +03:00
bool util_is_printable_string ( const void * data , int len ) ;
2012-09-29 01:25:59 +04:00
/*
* Parse an escaped character starting at index i in string s . The resulting
* character will be returned and the index i will be updated to point at the
* character directly after the end of the encoding , this may be the ' \0 '
* terminator of the string .
*/
char get_escape_char ( const char * s , int * i ) ;
/**
* Read a device tree file into a buffer . This will report any errors on
* stderr .
*
* @ param filename The filename to read , or - for stdin
2014-01-21 16:54:49 +04:00
* @ param len If non - NULL , the amount of data we managed to read
2018-09-13 16:59:25 +03:00
* @ return Pointer to allocated buffer containing fdt , or NULL on error
2014-01-21 16:54:49 +04:00
*/
2018-09-13 16:59:25 +03:00
char * utilfdt_read ( const char * filename , size_t * len ) ;
2014-01-21 16:54:49 +04:00
2012-09-29 01:25:59 +04:00
/**
* Read a device tree file into a buffer . Does not report errors , but only
* returns them . The value returned can be passed to strerror ( ) to obtain
* an error message for the user .
*
* @ param filename The filename to read , or - for stdin
* @ param buffp Returns pointer to buffer containing fdt
2014-01-21 16:54:49 +04:00
* @ param len If non - NULL , the amount of data we managed to read
2018-09-13 16:59:25 +03:00
* @ return 0 if ok , else an errno value representing the error
2014-01-21 16:54:49 +04:00
*/
2018-09-13 16:59:25 +03:00
int utilfdt_read_err ( const char * filename , char * * buffp , size_t * len ) ;
2012-09-29 01:25:59 +04:00
/**
* Write a device tree buffer to a file . This will report any errors on
* stderr .
*
* @ param filename The filename to write , or - for stdout
2019-06-12 16:05:52 +03:00
* @ param blob Pointer to buffer containing fdt
2012-09-29 01:25:59 +04:00
* @ return 0 if ok , - 1 on error
*/
int utilfdt_write ( const char * filename , const void * blob ) ;
/**
* Write a device tree buffer to a file . Does not report errors , but only
* returns them . The value returned can be passed to strerror ( ) to obtain
* an error message for the user .
*
* @ param filename The filename to write , or - for stdout
2019-06-12 16:05:52 +03:00
* @ param blob Pointer to buffer containing fdt
2012-09-29 01:25:59 +04:00
* @ return 0 if ok , else an errno value representing the error
*/
int utilfdt_write_err ( const char * filename , const void * blob ) ;
/**
* Decode a data type string . The purpose of this string
*
* The string consists of an optional character followed by the type :
* Modifier characters :
* hh or b 1 byte
* h 2 byte
* l 4 byte , default
*
* Type character :
* s string
* i signed integer
* u unsigned integer
* x hex
2022-11-01 20:54:44 +03:00
* r raw
2012-09-29 01:25:59 +04:00
*
* TODO : Implement ll modifier ( 8 bytes )
* TODO : Implement o type ( octal )
*
* @ param fmt Format string to process
* @ param type Returns type found ( s / d / u / x ) , or 0 if none
* @ param size Returns size found ( 1 , 2 , 4 , 8 ) or 4 if none
* @ return 0 if ok , - 1 on error ( no type given , or other invalid format )
*/
int utilfdt_decode_type ( const char * fmt , int * type , int * size ) ;
/*
* This is a usage message fragment for the - t option . It is the format
* supported by utilfdt_decode_type .
*/
# define USAGE_TYPE_MSG \
2022-11-01 20:54:44 +03:00
" <type> \t s=string, i=int, u=unsigned, x=hex, r=raw \n " \
2012-09-29 01:25:59 +04:00
" \t Optional modifier prefix: \n " \
2014-01-21 16:54:49 +04:00
" \t \t hh or b=byte, h=2 byte, l=4 byte (default) " ;
/**
* Print property data in a readable format to stdout
*
* Properties that look like strings will be printed as strings . Otherwise
* the data will be displayed either as cells ( if len is a multiple of 4
* bytes ) or bytes .
*
* If len is 0 then this function does nothing .
*
* @ param data Pointers to property data
* @ param len Length of property data
*/
void utilfdt_print_data ( const char * data , int len ) ;
/**
* Show source version and exit
*/
2017-03-21 17:01:08 +03:00
void NORETURN util_version ( void ) ;
2014-01-21 16:54:49 +04:00
/**
* Show usage and exit
*
* This helps standardize the output of various utils . You most likely want
* to use the usage ( ) helper below rather than call this .
*
* @ param errmsg If non - NULL , an error message to display
* @ param synopsis The initial example usage text ( and possible examples )
* @ param short_opts The string of short options
* @ param long_opts The structure of long options
* @ param opts_help An array of help strings ( should align with long_opts )
*/
2017-03-21 17:01:08 +03:00
void NORETURN util_usage ( const char * errmsg , const char * synopsis ,
const char * short_opts ,
struct option const long_opts [ ] ,
const char * const opts_help [ ] ) ;
2014-01-21 16:54:49 +04:00
/**
* Show usage and exit
*
* If you name all your usage variables with usage_xxx , then you can call this
* help macro rather than expanding all arguments yourself .
*
* @ param errmsg If non - NULL , an error message to display
*/
# define usage(errmsg) \
util_usage ( errmsg , usage_synopsis , usage_short_opts , \
usage_long_opts , usage_opts_help )
/**
* Call getopt_long ( ) with standard options
*
* Since all util code runs getopt in the same way , provide a helper .
*/
# define util_getopt_long() getopt_long(argc, argv, usage_short_opts, \
usage_long_opts , NULL )
/* Helper for aligning long_opts array */
# define a_argument required_argument
/* Helper for usage_short_opts string constant */
# define USAGE_COMMON_SHORT_OPTS "hV"
/* Helper for usage_long_opts option array */
# define USAGE_COMMON_LONG_OPTS \
{ " help " , no_argument , NULL , ' h ' } , \
{ " version " , no_argument , NULL , ' V ' } , \
{ NULL , no_argument , NULL , 0x0 }
/* Helper for usage_opts_help array */
# define USAGE_COMMON_OPTS_HELP \
" Print this help and exit " , \
" Print version and exit " , \
NULL
/* Helper for getopt case statements */
# define case_USAGE_COMMON_FLAGS \
case ' h ' : usage ( NULL ) ; \
case ' V ' : util_version ( ) ; \
case ' ? ' : usage ( " unknown option " ) ;
2012-09-29 01:25:59 +04:00
2018-02-28 02:40:38 +03:00
# endif /* UTIL_H */