Skip to content

Commit 21bbc36

Browse files
Decode TLV payload data and present it in a human readable format (#27638)
* Start with a flat tree library for a human TLV format * Temp change for test * Switch to a flat list and more flexible finding ... expect I want to find by id AND name eventually * Clang-format * Add tests for searching by name in a flat tree * Provide non-array find-entry * Have a good tree position that works for navigating and descend/ascend * Added more documentation * Add more unit tests * Fix naming * Support current path for flat tree positions * Restyle * Add IM message encoding, to have pretty-print of data once available * Added secure channel message formats * Add UDC defintions * Rename things * make the matter file parseable * Attempt to start a codegen for tlv meta mapping * Restyle * Add missing files * Restyle * Have some codegen working, start defining types and names * Start implementing a bit of a table generation. not done, but tables start to exist * Support events for tables * Add support for commands (untested though) * More work, all except lists and constants are code-generated * Restyle * List support and better tag support including anonymous support * Make tags specific: many tags are NOT context tags currently * Restyle * Add some test data for development tests * Start adding some test support ... to be removed later * Code compiles * Add a unit test that compiles and runs * Starting some decoding support. Still very much broken * A bit more decoding, this time we handle lists. TLV interface is VERY bad * Better decoding, we now show data * Add some item information, to prepare for enum and bitmap decoding * Restyle * Add error messages on usage of command line * remove a non yes/no argument * Update error syntax for 2 more arguments * update the help. using both true/false and yes/no is a mess * Update logic for decoding * Do not allow restyle * Better StringBuilder formatting * Test adjustment * Naming update * Restyle * Add Format option for buffer writers and string builders * Update comments * Updated to only have printf inside stringbuilder and NOT bufferwriter * Restyle * Add missing file * remove cpp file comments * Fix cast to make clang happy * Much better formatting and make the compile clang-friendly * minor const correctness change. TLVReader has non-const getters * Add special tags for payloads of things * Added logic for binary data and payloads, to process things * Start adding clusters metadata, make everything const-correct * Minor update * Start updating formats * Never pass null pointer in vsnprintf, since our size available is never 0 * Restyle * Restyle * Better decoding * Clean up some printfs * Iterator decoding seems to work, including getting sub-data types * Better organization of code ... protocols decoder is actually a class now * Allow passing in decode trees for protocol understanding * Better arg parsing - was able to test for invalid data * Restyle * Add back reset call to stringbuilder * Restyle and make protocol decoding actually work * Unformatted protocols/cluster meta * Add more trace data for testing, fix SEGFAULT in decoder * Support non-struct list entries * Switch list decoding logic to be inside generated metadata * Restyle * Fix compilation and generation * Start having codegen support for protocols metadata * Move clusters meta to compile time codegen as well * Restyle * Cleanup dependencies a bit * Start making TestDecoding be actual unit tests * more unit tests, without protocol decoding * Slightly better formatting * More unit tests ... although invalid data looks odd * Better formatting of unknown attributes * Updated tests * Restyle * Better messaging, test overflows * Removed unused file * Undo submodule update * Add some tests for invoke. Command list is NOT complete * Restyle * Yield commands that have no request structure * Fix comment * Start adding some unit tests for cpp-tlvmeta codegen * Add tests for real * Allow both hex and json at the same time for output * Rename log_json to just json * Add file output option for json tracing * make the output look like a json array when outputing to file * Restyle * Fix support of "json:log" * Fix support of "json:log" * make things compile * Rename open/close to openfile/closefile to avoid override errors * Restyle * StartsWith should be available now globally as it is always used * StartsWith should be available now globally as it is always used * Forward declare json to make arm cross compile pass * Forward declare json to make arm cross compile pass * Restyle * Add some support for formatting enums and bitmaps * Restyle * Add json_tracing exceptions for includes checks * Proper bitmap support with tests, status codes are bitmaps now * Restyle * Update test for overflow to have more unique values * Restyle * Fix decoding of command inputs and names * Add a fuzz test for payload decoder * Handle invalid TLV * Add more error handling. Fuzzing runs longer now * Restyle * Make clang happy * Fix efr32 unit test compilation * Fix python lint * Restyle * Fix typo and restyle * Restyle * Add dependencies to flat-tree for generated code: they are needed * make tests use uppercase for unknown tags as this is the code update I made recently * Undo submodule update * Make clang-tidy happy * Fix subscribe response message indexing to match spec --------- Co-authored-by: Andrei Litvin <andreilitvin@google.com>
1 parent 8c288b4 commit 21bbc36

38 files changed

+3953
-13
lines changed

BUILD.gn

+1
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,7 @@ if (current_toolchain != "${dir_pw_toolchain}/default:default") {
5454
"${chip_root}/src/credentials/tests:fuzz-chip-cert",
5555
"${chip_root}/src/lib/core/tests:fuzz-tlv-reader",
5656
"${chip_root}/src/lib/dnssd/minimal_mdns/tests:fuzz-minmdns-packet-parsing",
57+
"${chip_root}/src/lib/format/tests:fuzz-payload-decoder",
5758
]
5859
}
5960
}

build/chip/chip_codegen.gni

+14-1
Original file line numberDiff line numberDiff line change
@@ -53,9 +53,19 @@ template("_chip_build_time_codegen") {
5353
rebase_path(target_gen_dir, root_build_dir),
5454
"--expected-outputs",
5555
rebase_path(_expected_outputs, root_build_dir),
56-
rebase_path(_idl_file, root_build_dir),
5756
]
5857

58+
if (defined(invoker.options)) {
59+
foreach(option, invoker.options) {
60+
args += [
61+
"--option",
62+
option,
63+
]
64+
}
65+
}
66+
67+
args += [ rebase_path(_idl_file, root_build_dir) ]
68+
5969
inputs = [
6070
_idl_file,
6171
_expected_outputs,
@@ -313,12 +323,15 @@ template("chip_codegen") {
313323
"generator",
314324
"input",
315325
"outputs",
326+
"options",
316327
"public_configs",
317328
])
318329
}
319330
} else {
320331
_name = target_name
321332

333+
not_needed(invoker, [ "options" ])
334+
322335
# This constructs a path like:
323336
# FROM all-clusters-app.matter (inside examples/all-clusters-app/all-clusters-common/)
324337
# USING "cpp-app" for generator:

examples/chip-tool/args.gni

+3
Original file line numberDiff line numberDiff line change
@@ -27,3 +27,6 @@ matter_enable_tracing_support = true
2727

2828
# Perfetto requires C++17
2929
cpp_standard = "gnu++17"
30+
31+
matter_log_json_payload_hex = true
32+
matter_log_json_payload_decode_full = true

scripts/pregenerate/__init__.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,8 @@
2020
from typing import Iterator, List, Optional
2121

2222
from .types import IdlFileType, InputIdlFile
23-
from .using_codegen import CodegenCppAppPregenerator, CodegenJavaClassPregenerator, CodegenJavaJNIPregenerator
23+
from .using_codegen import (CodegenCppAppPregenerator, CodegenCppClustersTLVMetaPregenerator,
24+
CodegenCppProtocolsTLVMetaPregenerator, CodegenJavaClassPregenerator, CodegenJavaJNIPregenerator)
2425
from .using_zap import ZapApplicationPregenerator
2526

2627

@@ -94,6 +95,8 @@ def FindPregenerationTargets(sdk_root: str, external_roots: Optional[List[str]],
9495
CodegenJavaJNIPregenerator(sdk_root),
9596
CodegenJavaClassPregenerator(sdk_root),
9697
CodegenCppAppPregenerator(sdk_root),
98+
CodegenCppClustersTLVMetaPregenerator(sdk_root),
99+
CodegenCppProtocolsTLVMetaPregenerator(sdk_root),
97100

98101
# ZAP codegen
99102
ZapApplicationPregenerator(sdk_root),

scripts/pregenerate/using_codegen.py

+36-2
Original file line numberDiff line numberDiff line change
@@ -27,11 +27,12 @@
2727
class CodegenTarget:
2828
"""A target that uses `scripts/codegen.py` to generate files."""
2929

30-
def __init__(self, idl: InputIdlFile, generator: str, sdk_root: str, runner):
30+
def __init__(self, idl: InputIdlFile, generator: str, sdk_root: str, runner, options=[]):
3131
self.idl = idl
3232
self.generator = generator
3333
self.sdk_root = sdk_root
3434
self.runner = runner
35+
self.options = options
3536

3637
if idl.file_type != IdlFileType.MATTER:
3738
raise Exception(
@@ -51,8 +52,12 @@ def Generate(self, output_root: str):
5152
'--log-level', 'fatal',
5253
'--generator', self.generator,
5354
'--output-dir', output_dir,
54-
self.idl.full_path
5555
]
56+
for option in self.options:
57+
cmd.append("--option")
58+
cmd.append(option)
59+
60+
cmd.append(self.idl.full_path)
5661

5762
logging.debug(f"Executing {cmd}")
5863
self.runner.run(cmd)
@@ -97,6 +102,9 @@ def Accept(self, idl: InputIdlFile):
97102
if idl.file_type != IdlFileType.MATTER:
98103
return False
99104

105+
if '/lib/format/' in idl.relative_path:
106+
return False
107+
100108
# we should not be checked for these, but verify just in case
101109
if '/tests/' in idl.relative_path:
102110
return False
@@ -105,3 +113,29 @@ def Accept(self, idl: InputIdlFile):
105113

106114
def CreateTarget(self, idl: InputIdlFile, runner):
107115
return CodegenTarget(sdk_root=self.sdk_root, idl=idl, generator="cpp-app", runner=runner)
116+
117+
118+
class CodegenCppProtocolsTLVMetaPregenerator:
119+
"""Pregeneration logic for "cpp-app" codegen.py outputs"""
120+
121+
def __init__(self, sdk_root):
122+
self.sdk_root = sdk_root
123+
124+
def Accept(self, idl: InputIdlFile):
125+
return (idl.file_type == IdlFileType.MATTER) and idl.relative_path.endswith('/protocol_messages.matter')
126+
127+
def CreateTarget(self, idl: InputIdlFile, runner):
128+
return CodegenTarget(sdk_root=self.sdk_root, idl=idl, generator="cpp-tlvmeta", options=["table_name:protocols_meta"], runner=runner)
129+
130+
131+
class CodegenCppClustersTLVMetaPregenerator:
132+
"""Pregeneration logic for "cpp-app" codegen.py outputs"""
133+
134+
def __init__(self, sdk_root):
135+
self.sdk_root = sdk_root
136+
137+
def Accept(self, idl: InputIdlFile):
138+
return (idl.file_type == IdlFileType.MATTER) and idl.relative_path.endswith('/controller-clusters.matter')
139+
140+
def CreateTarget(self, idl: InputIdlFile, runner):
141+
return CodegenTarget(sdk_root=self.sdk_root, idl=idl, generator="cpp-tlvmeta", options=["table_name:clusters_meta"], runner=runner)

scripts/py_matter_idl/files.gni

+17-3
Original file line numberDiff line numberDiff line change
@@ -3,24 +3,38 @@ import("//build_overrides/chip.gni")
33

44
# Templates used for generation
55
matter_idl_generator_templates = [
6+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/cpp/application/CallbackStubSource.jinja",
7+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/cpp/application/PluginApplicationCallbacksHeader.jinja",
8+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/cpp/tlvmeta/TLVMetaData_cpp.jinja",
9+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/cpp/tlvmeta/TLVMetaData_h.jinja",
10+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/java/CHIPCallbackTypes.jinja",
611
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/java/ChipClustersCpp.jinja",
712
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/java/ChipClustersRead.jinja",
8-
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/java/ClusterWriteMapping.jinja",
13+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/java/CHIPGlobalCallbacks_cpp.jinja",
14+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/java/CHIPReadCallbacks_h.jinja",
915
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/java/ClusterIDMapping.jinja",
10-
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/cpp/application/CallbackStubSource.jinja",
11-
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/cpp/application/PluginApplicationCallbacksHeader.jinja",
16+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/java/ClusterReadMapping.jinja",
17+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/java/ClusterWriteMapping.jinja",
1218
]
1319

1420
matter_idl_generator_sources = [
1521
"${chip_root}/scripts/py_matter_idl/matter_idl/__init__.py",
1622
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/__init__.py",
1723
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/cpp/__init__.py",
1824
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/cpp/application/__init__.py",
25+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/cpp/tlvmeta/__init__.py",
1926
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/filters.py",
2027
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/java/__init__.py",
28+
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/registry.py",
2129
"${chip_root}/scripts/py_matter_idl/matter_idl/generators/types.py",
30+
"${chip_root}/scripts/py_matter_idl/matter_idl/lint/__init__.py",
31+
"${chip_root}/scripts/py_matter_idl/matter_idl/lint/lint_rules_parser.py",
32+
"${chip_root}/scripts/py_matter_idl/matter_idl/lint/types.py",
2233
"${chip_root}/scripts/py_matter_idl/matter_idl/matter_idl_parser.py",
2334
"${chip_root}/scripts/py_matter_idl/matter_idl/matter_idl_types.py",
35+
"${chip_root}/scripts/py_matter_idl/matter_idl/test_generators.py",
36+
"${chip_root}/scripts/py_matter_idl/matter_idl/test_matter_idl_parser.py",
37+
"${chip_root}/scripts/py_matter_idl/matter_idl/test_xml_parser.py",
2438
"${chip_root}/scripts/py_matter_idl/matter_idl/xml_parser.py",
2539
"${chip_root}/scripts/py_matter_idl/matter_idl/zapxml/__init__.py",
2640
"${chip_root}/scripts/py_matter_idl/matter_idl/zapxml/handlers/__init__.py",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
#include <tlv/meta/{{table_name}}.h>
2+
3+
namespace chip {
4+
namespace TLVMeta {
5+
namespace {
6+
7+
using namespace chip::FlatTree;
8+
using namespace chip::TLV;
9+
10+
{%- for table in sub_tables %}
11+
12+
const Entry<ItemInfo> _{{table.full_name}}[] = {
13+
{%- for entry in table.entries %}
14+
{ { {{entry.code}}, "{{entry.name}}", ItemType::{{entry.item_type}} }, {{entry.reference | indexInTable(sub_tables)}} }, // {{entry.real_type}}
15+
{%- endfor %}
16+
};
17+
{%- endfor %}
18+
19+
const Entry<ItemInfo> _all_clusters[] = {
20+
{%- for cluster in clusters | sort(attribute='code') %}
21+
{ { ClusterTag({{"0x%02X" | format(cluster.code)}}), "{{cluster.name}}", ItemType::kDefault }, {{cluster.name | indexInTable(sub_tables)}} },
22+
{%- endfor %}
23+
24+
};
25+
26+
// For any non-structure list like u64[] or similar.
27+
const Entry<ItemInfo> _primitive_type_list[] = {
28+
{ { AnonymousTag(), "[]", ItemType::kDefault }, kInvalidNodeIndex },
29+
};
30+
31+
} // namespace
32+
33+
#define _ENTRY(n) { sizeof(n) / sizeof(n[0]), n}
34+
35+
const std::array<const Node<ItemInfo>, {{ sub_tables | length }} + 2> {{table_name}} = { {
36+
_ENTRY(_all_clusters), // 0
37+
_ENTRY(_primitive_type_list), // 1
38+
{%- for table in sub_tables %}
39+
_ENTRY(_{{table.full_name}}), // {{loop.index + 1}}
40+
{%- endfor %}
41+
} };
42+
43+
} // namespace TLVMeta
44+
} // namespace chip
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
#include <lib/format/tlv_meta.h>
2+
#include <lib/format/FlatTree.h>
3+
4+
#include <array>
5+
6+
namespace chip {
7+
namespace TLVMeta {
8+
9+
extern const std::array<const FlatTree::Node<ItemInfo>, {{ sub_tables | length }} + 2> {{table_name}};
10+
11+
} // namespace TLVMeta
12+
} // namespace chip

0 commit comments

Comments
 (0)