From mboxrd@z Thu Jan 1 00:00:00 1970 Received: from smtpout-03.galae.net (smtpout-03.galae.net [185.246.85.4]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 590863C6A41; Thu, 9 Apr 2026 11:55:00 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=185.246.85.4 ARC-Seal:i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1775735701; cv=none; b=W7uODTeSq9K6e0T+NSzxJTfrx4iFfhmMZafhfonlN1AQxsUMYT/emmWkvXqQa3SmvjDWN8iv3Gm5Kih0yfISEVha3BmT7N22JZNiWZ6lfy5RXIXcTkgdcAdSj5irqMbEdQUljVcEb7xk3Q4u6lw6UjeD4lTvsZTeN+oBFVJ+q4A= ARC-Message-Signature:i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1775735701; c=relaxed/simple; bh=PbtZqqPlzrM54W5R1+7Q1xReO+adybswjyLlbjgZMBY=; h=From:To:Cc:Subject:Date:Message-ID:In-Reply-To:References: MIME-Version; b=S2jW1Y1LqEnibV3KcWmvHgAweI8d+Gcx+hNgPOlJva4bONsyHD6unuv2NYPF5blFMZ3s6EAt5jQ5XD5+HozGMye2W43J4gWf8vHP9aX6be+yAL9mzbmZXbPzSgw1dkSEs57Lj61AQiW/UK0TEm0ZU620uJ9rJDc38/aR8DFiZxM= ARC-Authentication-Results:i=1; smtp.subspace.kernel.org; dmarc=pass (p=reject dis=none) header.from=bootlin.com; spf=pass smtp.mailfrom=bootlin.com; dkim=pass (2048-bit key) header.d=bootlin.com header.i=@bootlin.com header.b=e3gtGKSs; arc=none smtp.client-ip=185.246.85.4 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=reject dis=none) header.from=bootlin.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=bootlin.com Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=bootlin.com header.i=@bootlin.com header.b="e3gtGKSs" Received: from smtpout-01.galae.net (smtpout-01.galae.net [212.83.139.233]) by smtpout-03.galae.net (Postfix) with ESMTPS id 1CD514E42985; Thu, 9 Apr 2026 11:54:59 +0000 (UTC) Received: from mail.galae.net (mail.galae.net [212.83.136.155]) by smtpout-01.galae.net (Postfix) with ESMTPS id E6A315FDEB; Thu, 9 Apr 2026 11:54:58 +0000 (UTC) Received: from [127.0.0.1] (localhost [127.0.0.1]) by localhost (Mailerdaemon) with ESMTPSA id 8F33E104500AD; Thu, 9 Apr 2026 13:54:56 +0200 (CEST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=bootlin.com; s=dkim; t=1775735698; h=from:subject:date:message-id:to:cc:mime-version: content-transfer-encoding:in-reply-to:references; bh=RRV5ZWnwqddzg4PjMP2N9rwyYeJhVv9d53AJqgJJWzs=; b=e3gtGKSsjB+Z3fSH1ypzNlkUKGSI+I89hXzcK69XzcshvyT2MB0NGpW05QiyeCrPWLzAk0 MeMhmJOQaW7cX80nzFmkZG7yfwhM413DcdmvdLKw4YsvpBIPoNjIdI7uZ2r2Gehan51CWe h4S5yyd978ommu1LUowSbZseR0I8RiSSejaZMGBw7R3yYeJ8nEDWJFsEE45TmqYDB045KV pFGM/NHx8m7diWT2q8pjDj65nCYrb4ZaWRAxoSlevR/m9z/YYCXshrx4+rss3borPWv+ny H1DERqtZUKqiXokFxMEdB7UwH+BWu9lVTFUC2y5Pl2k3g3KzRqKSs8NDRg/yzQ== From: Herve Codina To: David Gibson , Rob Herring , Krzysztof Kozlowski , Conor Dooley Cc: Ayush Singh , Geert Uytterhoeven , devicetree-compiler@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, devicetree-spec@vger.kernel.org, Hui Pu , Ian Ray , Luca Ceresoli , Thomas Petazzoni , Herve Codina Subject: [PATCH v2 05/10] fdtdump: Handle unknown tags Date: Thu, 9 Apr 2026 13:54:21 +0200 Message-ID: <20260409115426.352214-6-herve.codina@bootlin.com> X-Mailer: git-send-email 2.53.0 In-Reply-To: <20260409115426.352214-1-herve.codina@bootlin.com> References: <20260409115426.352214-1-herve.codina@bootlin.com> Precedence: bulk X-Mailing-List: devicetree@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-Last-TLS-Session-Version: TLSv1.3 The structured tag value definition introduced recently gives the ability to ignore unknown tags without any error when they are read. Add support for those structured tags in fdtdump and introduce a command line option to dump unknown tags that should be ignored. Signed-off-by: Herve Codina --- fdtdump.c | 45 +++++++++- tests/dumptrees.c | 4 +- tests/run_tests.sh | 41 +++++++++ tests/testdata.h | 2 + tests/trees.S | 116 +++++++++++++++++++++++++ tests/unknown_tags_can_skip.dtb.expect | 29 +++++++ 6 files changed, 233 insertions(+), 4 deletions(-) create mode 100644 tests/unknown_tags_can_skip.dtb.expect diff --git a/fdtdump.c b/fdtdump.c index 0e7a265..7a8b278 100644 --- a/fdtdump.c +++ b/fdtdump.c @@ -44,7 +44,7 @@ static const char *tagname(uint32_t tag) #define dumpf(fmt, args...) \ do { if (debug) printf("// " fmt, ## args); } while (0) -static void dump_blob(void *blob, bool debug) +static void dump_blob(void *blob, bool debug, int dump_unknown) { uintptr_t blob_off = (uintptr_t)blob; struct fdt_header *bph = blob; @@ -146,20 +146,55 @@ static void dump_blob(void *blob, bool debug) continue; } + if ((tag & FDT_TAG_STRUCTURED) && (tag & FDT_TAG_SKIP_SAFE)) { + sz = 0; + switch (tag & FDT_TAG_DATA_MASK) { + case FDT_TAG_DATA_NONE: + break; + case FDT_TAG_DATA_1CELL: + sz = FDT_CELLSIZE; + break; + case FDT_TAG_DATA_2CELLS: + sz = 2 * FDT_CELLSIZE; + break; + case FDT_TAG_DATA_VARLEN: + /* Get the length */ + sz = fdt32_to_cpu(GET_CELL(p)); + break; + } + + if (dump_unknown) { + printf("%*s// Unknown tag ignored: 0x%08"PRIx32", data len %d", + depth * shift, "", tag, sz); + if (dump_unknown > 1 && sz != 0) { + printf(" "); + for (i = 0; i < sz; i++) + printf("%02hhx", *(p + i)); + } + printf("\n"); + } + + /* Skip the data bytes */ + p = PALIGN(p + sz, 4); + continue; + } + die("** Unknown tag 0x%08"PRIx32"\n", tag); } } /* Usage related data. */ static const char usage_synopsis[] = "fdtdump [options] "; -static const char usage_short_opts[] = "ds" USAGE_COMMON_SHORT_OPTS; +static const char usage_short_opts[] = "dus" USAGE_COMMON_SHORT_OPTS; static struct option const usage_long_opts[] = { {"debug", no_argument, NULL, 'd'}, + {"unknown", no_argument, NULL, 'u'}, {"scan", no_argument, NULL, 's'}, USAGE_COMMON_LONG_OPTS }; static const char * const usage_opts_help[] = { "Dump debug information while decoding the file", + "Dump unknown tags information while decoding the file (-uu to dump data)", "Scan for an embedded fdt in file", USAGE_COMMON_OPTS_HELP }; @@ -183,6 +218,7 @@ int main(int argc, char *argv[]) const char *file; char *buf; bool debug = false; + int dump_unknown = 0; bool scan = false; size_t len; @@ -198,6 +234,9 @@ int main(int argc, char *argv[]) case 'd': debug = true; break; + case 'u': + dump_unknown++; + break; case 's': scan = true; break; @@ -242,7 +281,7 @@ int main(int argc, char *argv[]) } else if (!valid_header(buf, len)) die("%s: header is not valid\n", file); - dump_blob(buf, debug); + dump_blob(buf, debug, dump_unknown); return 0; } diff --git a/tests/dumptrees.c b/tests/dumptrees.c index 08967b3..4732fff 100644 --- a/tests/dumptrees.c +++ b/tests/dumptrees.c @@ -25,7 +25,9 @@ static struct { TREE(truncated_property), TREE(truncated_string), TREE(truncated_memrsv), TREE(two_roots), - TREE(named_root) + TREE(named_root), + TREE(unknown_tags_can_skip), + TREE(unknown_tags_no_skip) }; #define NUM_TREES (sizeof(trees) / sizeof(trees[0])) diff --git a/tests/run_tests.sh b/tests/run_tests.sh index f07092b..f2855dd 100755 --- a/tests/run_tests.sh +++ b/tests/run_tests.sh @@ -196,6 +196,40 @@ check_align () { ) } +# $1: f1 file +# $2: f2 file +check_diff () { + printf "diff $1 $2: " + local f1="$1" + local f2="$2" + ( + if diff $f1 $f2 >/dev/null; then + PASS + else + if [ -z "$QUIET_TEST" ]; then + echo "DIFF :-:" + diff -u $f1 $f2 + fi + FAIL "Results differ from expected" + fi + ) +} + +# $1: dtb file +# $2: out file +wrap_fdtdump () { + printf "wrap_fdtdump -uu $1: " + local dtb="$1" + local out="$2" + ( + if $FDTDUMP -uu ${dtb} 2>/dev/null >${out}; then + PASS + else + FAIL + fi + ) +} + run_dtc_test () { printf "dtc $*: " base_run_test wrap_test $VALGRIND $DTC "$@" @@ -1007,6 +1041,13 @@ utilfdt_tests () { fdtdump_tests () { run_fdtdump_test "$SRCDIR/fdtdump.dts" + + base_run_test wrap_fdtdump unknown_tags_can_skip.dtb unknown_tags_can_skip.dtb.out + # Remove unneeded comments + sed -i '/^\/\/ [^U]/d' unknown_tags_can_skip.dtb.out + base_run_test check_diff unknown_tags_can_skip.dtb.out "$SRCDIR/unknown_tags_can_skip.dtb.expect" + + run_wrap_error_test $FDTDUMP unknown_tags_no_skip.dtb } fdtoverlay_tests() { diff --git a/tests/testdata.h b/tests/testdata.h index fcebc2c..aef04ab 100644 --- a/tests/testdata.h +++ b/tests/testdata.h @@ -57,4 +57,6 @@ extern struct fdt_header truncated_string; extern struct fdt_header truncated_memrsv; extern struct fdt_header two_roots; extern struct fdt_header named_root; +extern struct fdt_header unknown_tags_can_skip; +extern struct fdt_header unknown_tags_no_skip; #endif /* ! __ASSEMBLER__ */ diff --git a/tests/trees.S b/tests/trees.S index 4db2b9b..ef9a175 100644 --- a/tests/trees.S +++ b/tests/trees.S @@ -328,3 +328,119 @@ named_root_strings: named_root_strings_end: named_root_end: + + + /* Tree with "unknown" tags that can be skipped + * Use a really future dtb version to check version downgrade on + * modification. + */ + treehdr_vers unknown_tags_can_skip 0xffffffff 0x10 + empty_rsvmap unknown_tags_can_skip + +unknown_tags_can_skip_struct: + fdtlong FDT_TEST_1CELL_CAN_SKIP + fdtlong 0x1 + + beginn "" + fdtlong FDT_TEST_NONE_CAN_SKIP + + propu32 unknown_tags_can_skip, prop_int, 0x3201 + + fdtlong FDT_TEST_1CELL_CAN_SKIP + fdtlong 0x110 + + propstr unknown_tags_can_skip, prop_str, "abcd" + + fdtlong FDT_TEST_2CELLS_CAN_SKIP + fdtlong 0x120 + fdtlong 0x121 + + fdtlong FDT_TEST_VARLEN_CAN_SKIP + fdtlong 3 + .byte 0x10 + .byte 0x11 + .byte 0x12 + .byte 0 /* padding */ + + beginn "subnode1" + propu64 unknown_tags_can_skip, prop_int, 0x6401, 0x6402 + fdtlong FDT_TEST_NONE_CAN_SKIP + endn + + beginn "subnode2" + fdtlong FDT_TEST_1CELL_CAN_SKIP + fdtlong 0x121 + propu64 unknown_tags_can_skip, prop_int1, 0x64020, 0x64021 + fdtlong FDT_TEST_2CELLS_CAN_SKIP + fdtlong 0x1220 + fdtlong 0x1221 + propu32 unknown_tags_can_skip, prop_int2, 0x32022 + beginn "subsubnode" + fdtlong FDT_TEST_2CELLS_CAN_SKIP + fdtlong 0x1230 + fdtlong 0x1231 + propu64 unknown_tags_can_skip, prop_int, 0x64023, 0x64024 + endn + fdtlong FDT_TEST_VARLEN_CAN_SKIP + fdtlong 3 + .byte 0x21 + .byte 0x22 + .byte 0x23 + .byte 0x0 + endn + + fdtlong FDT_TEST_VARLEN_CAN_SKIP + fdtlong 5 + .byte 0x31 + .byte 0x32 + .byte 0x33 + .byte 0x34 + .byte 0x35 + .byte 0 /* padding */ + .byte 0 /* padding */ + .byte 0 /* padding */ + endn + + fdtlong FDT_TEST_1CELL_CAN_SKIP + fdtlong 0x2 + + fdtlong FDT_TEST_VARLEN_CAN_SKIP + fdtlong 2 + .byte 0x40 + .byte 0x41 + .byte 0 /* padding */ + .byte 0 /* padding */ + + fdtlong FDT_END + +unknown_tags_can_skip_struct_end: + +unknown_tags_can_skip_strings: + string unknown_tags_can_skip, prop_int, "prop-int" + string unknown_tags_can_skip, prop_int1, "prop-int1" + string unknown_tags_can_skip, prop_int2, "prop-int2" + string unknown_tags_can_skip, prop_str, "prop-str" +unknown_tags_can_skip_strings_end: + +unknown_tags_can_skip_end: + + + /* Tree with "unknown" tags that cannot be skipped */ + treehdr unknown_tags_no_skip + empty_rsvmap unknown_tags_no_skip + +unknown_tags_no_skip_struct: + beginn "" + fdtlong FDT_TEST_NONE_NO_SKIP + beginn "subnode1" + propu64 unknown_tags_no_skip, prop_int, 1, 2 + endn + endn + fdtlong FDT_END +unknown_tags_no_skip_struct_end: + +unknown_tags_no_skip_strings: + string unknown_tags_no_skip, prop_int, "prop-int" +unknown_tags_no_skip_strings_end: + +unknown_tags_no_skip_end: diff --git a/tests/unknown_tags_can_skip.dtb.expect b/tests/unknown_tags_can_skip.dtb.expect new file mode 100644 index 0000000..bf52e05 --- /dev/null +++ b/tests/unknown_tags_can_skip.dtb.expect @@ -0,0 +1,29 @@ +/dts-v1/; + +// Unknown tag ignored: 0xd0000000, data len 4 00000001 +/ { + // Unknown tag ignored: 0xc0000000, data len 0 + prop-int = <0x00003201>; + // Unknown tag ignored: 0xd0000000, data len 4 00000110 + prop-str = "abcd"; + // Unknown tag ignored: 0xe0000000, data len 8 0000012000000121 + // Unknown tag ignored: 0xf0000000, data len 3 101112 + subnode1 { + prop-int = <0x00006401 0x00006402>; + // Unknown tag ignored: 0xc0000000, data len 0 + }; + subnode2 { + // Unknown tag ignored: 0xd0000000, data len 4 00000121 + prop-int1 = <0x00064020 0x00064021>; + // Unknown tag ignored: 0xe0000000, data len 8 0000122000001221 + prop-int2 = <0x00032022>; + subsubnode { + // Unknown tag ignored: 0xe0000000, data len 8 0000123000001231 + prop-int = <0x00064023 0x00064024>; + }; + // Unknown tag ignored: 0xf0000000, data len 3 212223 + }; + // Unknown tag ignored: 0xf0000000, data len 5 3132333435 +}; +// Unknown tag ignored: 0xd0000000, data len 4 00000002 +// Unknown tag ignored: 0xf0000000, data len 2 4041 -- 2.53.0