1
1
{{> header}}
2
2
3
3
#include <app-common/zap-generated/cluster-objects.h>
4
+ #include <variant>
4
5
5
6
namespace chip {
6
7
namespace app {
7
8
namespace Clusters {
8
9
9
10
namespace detail {
10
11
11
- CHIP_ERROR FlightCheckDecodeAndEnterStruct(TLV::TLVReader & reader, TLV::TLVType & outer)
12
- {
13
- TLV::TLVReader temp_reader;
14
-
15
- // Make a copy of the struct reader to do pre-checks.
16
- temp_reader.Init(reader);
12
+ class StructDecodeIterator {
13
+ public:
14
+ // may return a context tag, a CHIP_ERROR (end iteration)
15
+ using EntryElement = std::variant<uint8_t, CHIP_ERROR>;
16
+
17
+ StructDecodeIterator(TLV::TLVReader &reader) : mReader(reader){}
18
+
19
+ // Iterate through structure elements. Returns one of:
20
+ // - uint8_t CONTEXT TAG (keep iterating)
21
+ // - CHIP_ERROR (including CHIP_NO_ERROR) which should be a final
22
+ // return value (stop iterating)
23
+ EntryElement Next() {
24
+ if (!mEntered) {
25
+ VerifyOrReturnError(TLV::kTLVType_Structure == mReader.GetType(), CHIP_ERROR_WRONG_TLV_TYPE);
26
+ ReturnErrorOnFailure(mReader.EnterContainer(mOuter));
27
+ mEntered = true;
28
+ }
29
+
30
+ while (true) {
31
+ CHIP_ERROR err = mReader.Next();
32
+ if (err != CHIP_NO_ERROR) {
33
+ VerifyOrReturnError(err == CHIP_ERROR_END_OF_TLV, err);
34
+ break;
35
+ }
17
36
18
- // Ensure we have a single struct and that it's properly bounded.
19
- CHIP_ERROR err = CHIP_NO_ERROR;
20
- VerifyOrReturnError(TLV::kTLVType_Structure == temp_reader.GetType(), CHIP_ERROR_WRONG_TLV_TYPE);
21
- ReturnErrorOnFailure(temp_reader.EnterContainer(outer));
22
- while ((err = temp_reader.Next()) == CHIP_NO_ERROR)
23
- {
24
- if (!TLV::IsContextTag(temp_reader.GetTag()))
25
- {
37
+ const TLV::Tag tag = mReader.GetTag();
38
+ if (!TLV::IsContextTag(tag)) {
26
39
continue;
27
- }
28
- }
29
- VerifyOrReturnError(err == CHIP_END_OF_TLV, err);
30
- ReturnErrorOnFailure(temp_reader.ExitContainer(outer));
40
+ }
31
41
32
- // Guaranteed to work due to prior checks.
33
- VerifyOrDie(reader.EnterContainer(outer) == CHIP_NO_ERROR);
34
- return CHIP_NO_ERROR;
35
- }
42
+ // we know context tags are 8-bit
43
+ return static_cast<uint8_t>(TLV::TagNumFromTag(tag));
44
+ }
36
45
37
- void ExitStructAfterDecode(TLV::TLVReader & reader, TLV::TLVType & outer)
38
- {
39
- // Ensure we exit the container. Will be OK since FlightCheckDecodeAndEnterStruct will have
40
- // already been called, and generated code properly iterates over entire container.
41
- VerifyOrDie(reader.Next() == CHIP_END_OF_TLV);
42
- VerifyOrDie(reader.ExitContainer(outer) == CHIP_NO_ERROR);
43
- }
46
+ return mReader.ExitContainer(mOuter);
47
+ }
48
+
49
+ private:
50
+ bool mEntered = false;
51
+ TLV::TLVType mOuter;
52
+ TLV::TLVReader &mReader;
53
+ };
44
54
45
55
// Structs shared across multiple clusters.
46
56
namespace Structs {
@@ -75,34 +85,37 @@ CHIP_ERROR Type::Encode(TLV::TLVWriter & aWriter, TLV::Tag aTag) const{
75
85
{{#zcl_command_arguments}}
76
86
ReturnErrorOnFailure(DataModel::Encode(aWriter, TLV::ContextTag(Fields::k{{asUpperCamelCase label}}), {{asLowerCamelCase label}}));
77
87
{{/zcl_command_arguments}}
78
- ReturnErrorOnFailure(aWriter.EndContainer(outer));
79
- return CHIP_NO_ERROR;
88
+ return aWriter.EndContainer(outer);
80
89
}
81
90
82
91
CHIP_ERROR DecodableType::Decode(TLV::TLVReader &reader) {
83
- TLV::TLVType outer;
84
- ReturnErrorOnFailure(chip::app::Clusters::detail::FlightCheckDecodeAndEnterStruct(reader, outer));
92
+ detail::StructDecodeIterator __iterator(reader);
93
+ while (true) {
94
+ auto __element = __iterator.Next();
95
+ if (std::holds_alternative<CHIP_ERROR>(__element)) {
96
+ return std::get<CHIP_ERROR>(__element);
97
+ }
98
+
99
+ {{#zcl_command_arguments~}}
100
+ {{#first}}
101
+ CHIP_ERROR err = CHIP_NO_ERROR;
102
+ const uint8_t __context_tag = std::get<uint8_t>(__element);
85
103
86
- CHIP_ERROR err = CHIP_NO_ERROR;
87
- while ((err = reader.Next()) == CHIP_NO_ERROR) {
88
- if (!TLV::IsContextTag(reader.GetTag() ))
104
+ {{/first~}}
105
+ {{! NOTE: using if/else instead of switch because it seems to generate smaller code. ~}}
106
+ if (__context_tag == to_underlying(Fields::k{{asUpperCamelCase label}} ))
89
107
{
90
- continue ;
108
+ err = DataModel::Decode(reader, {{asLowerCamelCase label}}) ;
91
109
}
92
- switch (TLV::TagNumFromTag(reader.GetTag()))
110
+ else
111
+ {{#last}}
93
112
{
94
- {{#zcl_command_arguments}}
95
- case to_underlying(Fields::k{{asUpperCamelCase label}}):
96
- ReturnErrorOnFailure(DataModel::Decode(reader, {{asLowerCamelCase label}}));
97
- break;
98
- {{/zcl_command_arguments}}
99
- default:
100
- break;
101
113
}
102
- }
103
114
104
- chip::app::Clusters::detail::ExitStructAfterDecode(reader, outer);
105
- return CHIP_NO_ERROR;
115
+ ReturnErrorOnFailure(err);
116
+ {{/last}}
117
+ {{/zcl_command_arguments}}
118
+ }
106
119
}
107
120
} // namespace {{asUpperCamelCase name}}.
108
121
{{/zcl_commands}}
@@ -114,14 +127,11 @@ CHIP_ERROR TypeInfo::DecodableType::Decode(TLV::TLVReader &reader, const Concret
114
127
{
115
128
{{#zcl_attributes_server}}
116
129
case Attributes::{{asUpperCamelCase label}}::TypeInfo::GetAttributeId():
117
- ReturnErrorOnFailure(DataModel::Decode(reader, {{asLowerCamelCase label}}));
118
- break;
130
+ return DataModel::Decode(reader, {{asLowerCamelCase label}});
119
131
{{/zcl_attributes_server}}
120
132
default:
121
- break ;
133
+ return CHIP_NO_ERROR ;
122
134
}
123
-
124
- return CHIP_NO_ERROR;
125
135
}
126
136
} // namespace Attributes
127
137
@@ -138,34 +148,37 @@ CHIP_ERROR Type::Encode(TLV::TLVWriter & aWriter, TLV::Tag aTag) const{
138
148
ReturnErrorOnFailure(DataModel::Encode(aWriter, TLV::ContextTag(Fields::k{{asUpperCamelCase name}}), {{asLowerCamelCase name}}));
139
149
{{/if_is_fabric_scoped_struct}}
140
150
{{/zcl_event_fields}}
141
- ReturnErrorOnFailure(aWriter.EndContainer(outer));
142
- return CHIP_NO_ERROR;
151
+ return aWriter.EndContainer(outer);
143
152
}
144
153
145
154
CHIP_ERROR DecodableType::Decode(TLV::TLVReader &reader) {
146
- TLV::TLVType outer;
147
- ReturnErrorOnFailure(chip::app::Clusters::detail::FlightCheckDecodeAndEnterStruct(reader, outer));
155
+ detail::StructDecodeIterator __iterator(reader);
156
+ while (true) {
157
+ auto __element = __iterator.Next();
158
+ if (std::holds_alternative<CHIP_ERROR>(__element)) {
159
+ return std::get<CHIP_ERROR>(__element);
160
+ }
161
+
162
+ {{#zcl_event_fields}}
163
+ {{#first}}
164
+ CHIP_ERROR err = CHIP_NO_ERROR;
165
+ const uint8_t __context_tag = std::get<uint8_t>(__element);
148
166
149
- CHIP_ERROR err = CHIP_NO_ERROR;
150
- while ((err = reader.Next()) == CHIP_NO_ERROR) {
151
- if (!TLV::IsContextTag(reader.GetTag() ))
167
+ {{/first~}}
168
+ {{! NOTE: using if/else instead of switch because it seems to generate smaller code. ~}}
169
+ if (__context_tag == to_underlying(Fields::k{{asUpperCamelCase name}} ))
152
170
{
153
- continue ;
171
+ err = DataModel::Decode(reader, {{asLowerCamelCase name}}) ;
154
172
}
155
- switch (TLV::TagNumFromTag(reader.GetTag()))
173
+ else
174
+ {{#last}}
156
175
{
157
- {{#zcl_event_fields}}
158
- case to_underlying(Fields::k{{asUpperCamelCase name}}):
159
- ReturnErrorOnFailure(DataModel::Decode(reader, {{asLowerCamelCase name}}));
160
- break;
161
- {{/zcl_event_fields}}
162
- default:
163
- break;
164
176
}
165
- }
166
177
167
- chip::app::Clusters::detail::ExitStructAfterDecode(reader, outer);
168
- return CHIP_NO_ERROR;
178
+ ReturnErrorOnFailure(err);
179
+ {{/last}}
180
+ {{/zcl_event_fields}}
181
+ }
169
182
}
170
183
} // namespace {{asUpperCamelCase name}}.
171
184
{{/zcl_events}}
0 commit comments