Add fdtget utility to read property values from a device tree
This simply utility makes it easy for scripts to read values from the device tree. It is written in C and uses the same libfdt as the rest of the dtc package. What is it for: - Reading fdt values from scripts - Extracting fdt information within build systems - Looking at particular values without having to dump the entire tree To use it, specify the fdt binary file on command line followed by a list of node, property pairs. The utility then looks up each node, finds the property and displays the value. Each value is printed on a new line. fdtget tries to guess the type of each property based on its contents. This is not always reliable, so you can use the -t option to force fdtget to decode the value as a string, or byte, etc. To read from stdin, use - as the file. Usage: fdtget <options> <dt file> [<node> <property>]... Options: -t <type> Type of data -h Print this help <type> s=string, i=int, u=unsigned, x=hex Optional modifier prefix: hh or b=byte, h=2 byte, l=4 byte (default) Signed-off-by: Simon Glass <sjg@chromium.org>
This commit is contained in:
parent
69df9f0de2
commit
68d057f20d
8 changed files with 326 additions and 1 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -10,3 +10,4 @@ lex.yy.c
|
||||||
/fdtdump
|
/fdtdump
|
||||||
/convert-dtsv0
|
/convert-dtsv0
|
||||||
/version_gen.h
|
/version_gen.h
|
||||||
|
/fdtget
|
||||||
|
|
4
Makefile
4
Makefile
|
@ -110,6 +110,7 @@ include Makefile.utils
|
||||||
BIN += convert-dtsv0
|
BIN += convert-dtsv0
|
||||||
BIN += dtc
|
BIN += dtc
|
||||||
BIN += fdtdump
|
BIN += fdtdump
|
||||||
|
BIN += fdtget
|
||||||
|
|
||||||
SCRIPTS = dtdiff
|
SCRIPTS = dtdiff
|
||||||
|
|
||||||
|
@ -120,6 +121,7 @@ ifneq ($(DEPTARGETS),)
|
||||||
-include $(DTC_OBJS:%.o=%.d)
|
-include $(DTC_OBJS:%.o=%.d)
|
||||||
-include $(CONVERT_OBJS:%.o=%.d)
|
-include $(CONVERT_OBJS:%.o=%.d)
|
||||||
-include $(FDTDUMP_OBJS:%.o=%.d)
|
-include $(FDTDUMP_OBJS:%.o=%.d)
|
||||||
|
-include $(FDTGET_OBJS:%.o=%.d)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
|
||||||
|
@ -180,6 +182,8 @@ convert-dtsv0: $(CONVERT_OBJS)
|
||||||
|
|
||||||
fdtdump: $(FDTDUMP_OBJS)
|
fdtdump: $(FDTDUMP_OBJS)
|
||||||
|
|
||||||
|
fdtget: $(FDTGET_OBJS) $(LIBFDT_archive)
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
# Testsuite rules
|
# Testsuite rules
|
||||||
|
|
|
@ -8,3 +8,10 @@ FDTDUMP_SRCS = \
|
||||||
util.c
|
util.c
|
||||||
|
|
||||||
FDTDUMP_OBJS = $(FDTDUMP_SRCS:%.c=%.o)
|
FDTDUMP_OBJS = $(FDTDUMP_SRCS:%.c=%.o)
|
||||||
|
|
||||||
|
|
||||||
|
FDTGET_SRCS = \
|
||||||
|
fdtget.c \
|
||||||
|
util.c
|
||||||
|
|
||||||
|
FDTGET_OBJS = $(FDTGET_SRCS:%.c=%.o)
|
||||||
|
|
226
fdtget.c
Normal file
226
fdtget.c
Normal file
|
@ -0,0 +1,226 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <ctype.h>
|
||||||
|
#include <getopt.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include <libfdt.h>
|
||||||
|
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
/* Holds information which controls our output and options */
|
||||||
|
struct display_info {
|
||||||
|
int type; /* data type (s/i/u/x or 0 for default) */
|
||||||
|
int size; /* data size (1/2/4) */
|
||||||
|
};
|
||||||
|
|
||||||
|
static void report_error(const char *where, int err)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Error at '%s': %s\n", where, fdt_strerror(err));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Displays data of a given length according to selected options
|
||||||
|
*
|
||||||
|
* If a specific data type is provided in disp, then this is used. Otherwise
|
||||||
|
* we try to guess the data type / size from the contents.
|
||||||
|
*
|
||||||
|
* @param disp Display information / options
|
||||||
|
* @param data Data to display
|
||||||
|
* @param len Maximum length of buffer
|
||||||
|
* @return 0 if ok, -1 if data does not match format
|
||||||
|
*/
|
||||||
|
static int show_data(struct display_info *disp, const char *data, int len)
|
||||||
|
{
|
||||||
|
int i, size;
|
||||||
|
const uint8_t *p = (const uint8_t *)data;
|
||||||
|
const char *s;
|
||||||
|
int value;
|
||||||
|
int is_string;
|
||||||
|
char fmt[3];
|
||||||
|
|
||||||
|
/* no data, don't print */
|
||||||
|
if (len == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
is_string = (disp->type) == 's' ||
|
||||||
|
(!disp->type && util_is_printable_string(data, len));
|
||||||
|
if (is_string) {
|
||||||
|
if (data[len - 1] != '\0') {
|
||||||
|
fprintf(stderr, "Unterminated string\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
for (s = data; s - data < len; s += strlen(s) + 1) {
|
||||||
|
if (s != data)
|
||||||
|
printf(" ");
|
||||||
|
printf("%s", (const char *)s);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
size = disp->size;
|
||||||
|
if (size == -1)
|
||||||
|
size = (len % 4) == 0 ? 4 : 1;
|
||||||
|
else if (len % size) {
|
||||||
|
fprintf(stderr, "Property length must be a multiple of "
|
||||||
|
"selected data size\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
fmt[0] = '%';
|
||||||
|
fmt[1] = disp->type ? disp->type : 'd';
|
||||||
|
fmt[2] = '\0';
|
||||||
|
for (i = 0; i < len; i += size, p += size) {
|
||||||
|
if (i)
|
||||||
|
printf(" ");
|
||||||
|
value = size == 4 ? fdt32_to_cpu(*(const uint32_t *)p) :
|
||||||
|
size == 2 ? (*p << 8) | p[1] : *p;
|
||||||
|
printf(fmt, value);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Show the data for a given node (and perhaps property) according to the
|
||||||
|
* display option provided.
|
||||||
|
*
|
||||||
|
* @param blob FDT blob
|
||||||
|
* @param disp Display information / options
|
||||||
|
* @param node Node to display
|
||||||
|
* @param property Name of property to display, or NULL if none
|
||||||
|
* @return 0 if ok, -ve on error
|
||||||
|
*/
|
||||||
|
static int show_data_for_item(const void *blob, struct display_info *disp,
|
||||||
|
int node, const char *property)
|
||||||
|
{
|
||||||
|
const void *value = NULL;
|
||||||
|
int len, err = 0;
|
||||||
|
|
||||||
|
value = fdt_getprop(blob, node, property, &len);
|
||||||
|
if (value) {
|
||||||
|
if (show_data(disp, value, len))
|
||||||
|
err = -1;
|
||||||
|
else
|
||||||
|
printf("\n");
|
||||||
|
} else {
|
||||||
|
report_error(property, len);
|
||||||
|
err = -1;
|
||||||
|
}
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Run the main fdtget operation, given a filename and valid arguments
|
||||||
|
*
|
||||||
|
* @param disp Display information / options
|
||||||
|
* @param filename Filename of blob file
|
||||||
|
* @param arg List of arguments to process
|
||||||
|
* @param arg_count Number of arguments
|
||||||
|
* @param return 0 if ok, -ve on error
|
||||||
|
*/
|
||||||
|
static int do_fdtget(struct display_info *disp, const char *filename,
|
||||||
|
char **arg, int arg_count)
|
||||||
|
{
|
||||||
|
char *blob;
|
||||||
|
int i, node;
|
||||||
|
|
||||||
|
blob = utilfdt_read(filename);
|
||||||
|
if (!blob)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
for (i = 0; i + 2 <= arg_count; i += 2) {
|
||||||
|
node = fdt_path_offset(blob, arg[0]);
|
||||||
|
if (node < 0) {
|
||||||
|
report_error(arg[0], node);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (show_data_for_item(blob, disp, node, arg[1]))
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *usage_msg =
|
||||||
|
"fdtget - read values from device tree\n"
|
||||||
|
"\n"
|
||||||
|
"Each value is printed on a new line.\n\n"
|
||||||
|
"Usage:\n"
|
||||||
|
" fdtget <options> <dt file> [<node> <property>]...\n"
|
||||||
|
"Options:\n"
|
||||||
|
"\t-t <type>\tType of data\n"
|
||||||
|
"\t-h\t\tPrint this help\n\n"
|
||||||
|
USAGE_TYPE_MSG;
|
||||||
|
|
||||||
|
static void usage(const char *msg)
|
||||||
|
{
|
||||||
|
if (msg)
|
||||||
|
fprintf(stderr, "Error: %s\n\n", msg);
|
||||||
|
|
||||||
|
fprintf(stderr, "%s", usage_msg);
|
||||||
|
exit(2);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char *argv[])
|
||||||
|
{
|
||||||
|
char *filename = NULL;
|
||||||
|
struct display_info disp;
|
||||||
|
|
||||||
|
/* set defaults */
|
||||||
|
memset(&disp, '\0', sizeof(disp));
|
||||||
|
disp.size = -1;
|
||||||
|
for (;;) {
|
||||||
|
int c = getopt(argc, argv, "ht:");
|
||||||
|
if (c == -1)
|
||||||
|
break;
|
||||||
|
|
||||||
|
switch (c) {
|
||||||
|
case 'h':
|
||||||
|
case '?':
|
||||||
|
usage(NULL);
|
||||||
|
|
||||||
|
case 't':
|
||||||
|
if (utilfdt_decode_type(optarg, &disp.type,
|
||||||
|
&disp.size))
|
||||||
|
usage("Invalid type string");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (optind < argc)
|
||||||
|
filename = argv[optind++];
|
||||||
|
if (!filename)
|
||||||
|
usage("Missing filename");
|
||||||
|
|
||||||
|
argv += optind;
|
||||||
|
argc -= optind;
|
||||||
|
|
||||||
|
/* Allow no arguments, and silently succeed */
|
||||||
|
if (!argc)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Check for node, property arguments */
|
||||||
|
if (argc % 2)
|
||||||
|
usage("Must have an even number of arguments");
|
||||||
|
|
||||||
|
if (do_fdtget(&disp, filename, argv, argc))
|
||||||
|
return 1;
|
||||||
|
return 0;
|
||||||
|
}
|
35
tests/fdtget-runtest.sh
Executable file
35
tests/fdtget-runtest.sh
Executable file
|
@ -0,0 +1,35 @@
|
||||||
|
#! /bin/sh
|
||||||
|
|
||||||
|
. ./tests.sh
|
||||||
|
|
||||||
|
LOG="tmp.log.$$"
|
||||||
|
EXPECT="tmp.expect.$$"
|
||||||
|
|
||||||
|
rm -f $TMPFILE $LOG
|
||||||
|
|
||||||
|
expect="$1"
|
||||||
|
echo "$expect" >$EXPECT
|
||||||
|
shift
|
||||||
|
|
||||||
|
verbose_run_log "$LOG" $VALGRIND "$DTGET" "$@"
|
||||||
|
ret="$?"
|
||||||
|
|
||||||
|
if [ "$ret" -ne 0 -a "$expect" = "ERR" ]; then
|
||||||
|
PASS
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ "$ret" -gt 127 ]; then
|
||||||
|
signame=$(kill -l $[ret - 128])
|
||||||
|
FAIL "Killed by SIG$signame"
|
||||||
|
fi
|
||||||
|
|
||||||
|
diff $EXPECT $LOG
|
||||||
|
ret="$?"
|
||||||
|
|
||||||
|
rm -f $LOG $EXPECT
|
||||||
|
|
||||||
|
if [ "$ret" -eq 0 ]; then
|
||||||
|
PASS
|
||||||
|
else
|
||||||
|
FAIL
|
||||||
|
fi
|
|
@ -83,6 +83,13 @@ asm_to_so_test () {
|
||||||
run_wrap_test asm_to_so "$@"
|
run_wrap_test asm_to_so "$@"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
run_fdtget_test () {
|
||||||
|
# run_fdtget_test name expected_output dtb_file args...
|
||||||
|
echo -n "$1: "
|
||||||
|
shift
|
||||||
|
base_run_test sh fdtget-runtest.sh "$@"
|
||||||
|
}
|
||||||
|
|
||||||
tree1_tests () {
|
tree1_tests () {
|
||||||
TREE=$1
|
TREE=$1
|
||||||
|
|
||||||
|
@ -402,6 +409,37 @@ dtbs_equal_tests () {
|
||||||
cmp_tests test_tree1.dtb $WRONG_TREE1
|
cmp_tests test_tree1.dtb $WRONG_TREE1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fdtget_tests () {
|
||||||
|
file=label01.dtb
|
||||||
|
$DTC -O dtb -o $file ${file%.dtb}.dts 2>/dev/null
|
||||||
|
|
||||||
|
# run_fdtget_test <test-name> <expected-result> <args>...
|
||||||
|
run_fdtget_test "Simple string" "MyBoardName" $file / model
|
||||||
|
run_fdtget_test "Multiple string i" "77 121 66 111 \
|
||||||
|
97 114 100 78 97 109 101 0 77 121 66 111 97 114 100 70 97 109 105 \
|
||||||
|
108 121 78 97 109 101 0" $file / compatible
|
||||||
|
run_fdtget_test "Multiple string s" "MyBoardName MyBoardFamilyName" \
|
||||||
|
-t s $file / compatible
|
||||||
|
run_fdtget_test "Integer" "32768" $file /cpus/PowerPC,970@1 d-cache-size
|
||||||
|
run_fdtget_test "Integer hex" "8000" -tx $file \
|
||||||
|
/cpus/PowerPC,970@1 d-cache-size
|
||||||
|
run_fdtget_test "Integer list" "61 62 63 0" -tbx $file \
|
||||||
|
/randomnode tricky1
|
||||||
|
run_fdtget_test "Byte list short" "a b c d de ea ad be ef" -tbx \
|
||||||
|
$file /randomnode blob
|
||||||
|
|
||||||
|
# Here the property size is not a multiple of 4 bytes, so it should fail
|
||||||
|
run_fdtget_test "Integer list invalid" ERR -tlx \
|
||||||
|
$file /randomnode mixed
|
||||||
|
run_fdtget_test "Integer list halfword" "6162 6300 1234 0 a 0 b 0 c" -thx \
|
||||||
|
$file /randomnode mixed
|
||||||
|
run_fdtget_test "Integer list byte" \
|
||||||
|
"61 62 63 0 12 34 0 0 0 a 0 0 0 b 0 0 0 c" -thhx \
|
||||||
|
$file /randomnode mixed
|
||||||
|
run_fdtget_test "Missing property" ERR -ts \
|
||||||
|
$file /randomnode doctor-who
|
||||||
|
}
|
||||||
|
|
||||||
utilfdt_tests () {
|
utilfdt_tests () {
|
||||||
run_test utilfdt_test
|
run_test utilfdt_test
|
||||||
}
|
}
|
||||||
|
@ -421,7 +459,7 @@ while getopts "vt:m" ARG ; do
|
||||||
done
|
done
|
||||||
|
|
||||||
if [ -z "$TESTSETS" ]; then
|
if [ -z "$TESTSETS" ]; then
|
||||||
TESTSETS="libfdt utilfdt dtc dtbs_equal"
|
TESTSETS="libfdt utilfdt dtc dtbs_equal fdtget"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Make sure we don't have stale blobs lying around
|
# Make sure we don't have stale blobs lying around
|
||||||
|
@ -441,6 +479,9 @@ for set in $TESTSETS; do
|
||||||
"dtbs_equal")
|
"dtbs_equal")
|
||||||
dtbs_equal_tests
|
dtbs_equal_tests
|
||||||
;;
|
;;
|
||||||
|
"fdtget")
|
||||||
|
fdtget_tests
|
||||||
|
;;
|
||||||
esac
|
esac
|
||||||
done
|
done
|
||||||
|
|
||||||
|
|
|
@ -11,6 +11,7 @@ FAIL () {
|
||||||
}
|
}
|
||||||
|
|
||||||
DTC=../dtc
|
DTC=../dtc
|
||||||
|
DTGET=../fdtget
|
||||||
|
|
||||||
verbose_run () {
|
verbose_run () {
|
||||||
if [ -z "$QUIET_TEST" ]; then
|
if [ -z "$QUIET_TEST" ]; then
|
||||||
|
|
10
util.h
10
util.h
|
@ -140,4 +140,14 @@ int utilfdt_write_err(const char *filename, const void *blob);
|
||||||
*/
|
*/
|
||||||
int utilfdt_decode_type(const char *fmt, int *type, int *size);
|
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 \
|
||||||
|
"<type>\ts=string, i=int, u=unsigned, x=hex\n" \
|
||||||
|
"\tOptional modifier prefix:\n" \
|
||||||
|
"\t\thh or b=byte, h=2 byte, l=4 byte (default)\n";
|
||||||
|
|
||||||
#endif /* _UTIL_H */
|
#endif /* _UTIL_H */
|
||||||
|
|
Loading…
Reference in a new issue