Skip to content

Commit 3100cbe

Browse files
authored
Look for more flash space savings in cluster-objects.cpp (project-chip#29561)
* Updates to cluster objects: use direct context tag comparisons * Also optimize code size usage for structure decoding * Completely split struct iterations and use std::variant based returns on an iterator * Another space save: using a common err for all decodes and single macro invocation seems to make code even smaller * One more cleanup return and proper use of first/last to remove unused code on empty structs (mainly command inputs) * Added a comment and a missed auto keyword * Missed some first/last iterations
1 parent ae1fdb1 commit 3100cbe

File tree

3 files changed

+10973
-13399
lines changed

3 files changed

+10973
-13399
lines changed

src/app/zap-templates/partials/cluster-objects-struct.zapt

+22-21
Original file line numberDiff line numberDiff line change
@@ -113,32 +113,33 @@ CHIP_ERROR Type::Encode(TLV::TLVWriter & aWriter, TLV::Tag aTag) const
113113
{{/if}}
114114

115115
CHIP_ERROR DecodableType::Decode(TLV::TLVReader &reader) {
116-
CHIP_ERROR err = CHIP_NO_ERROR;
117-
TLV::TLVType outer;
118-
VerifyOrReturnError(TLV::kTLVType_Structure == reader.GetType(), CHIP_ERROR_WRONG_TLV_TYPE);
119-
err = reader.EnterContainer(outer);
120-
ReturnErrorOnFailure(err);
121-
while ((err = reader.Next()) == CHIP_NO_ERROR) {
122-
if (!TLV::IsContextTag(reader.GetTag()))
116+
detail::StructDecodeIterator __iterator(reader);
117+
while (true) {
118+
auto __element = __iterator.Next();
119+
if (std::holds_alternative<CHIP_ERROR>(__element)) {
120+
return std::get<CHIP_ERROR>(__element);
121+
}
122+
123+
{{#zcl_struct_items}}
124+
{{#first}}
125+
CHIP_ERROR err = CHIP_NO_ERROR;
126+
const uint8_t __context_tag = std::get<uint8_t>(__element);
127+
128+
{{/first~}}
129+
{{! NOTE: using if/else instead of switch because it seems to generate smaller code. ~}}
130+
if (__context_tag == to_underlying(Fields::k{{asUpperCamelCase label}}))
123131
{
124-
continue;
132+
err = DataModel::Decode(reader, {{asLowerCamelCase label}});
125133
}
126-
switch (TLV::TagNumFromTag(reader.GetTag()))
134+
else
135+
{{#last}}
127136
{
128-
{{#zcl_struct_items}}
129-
case to_underlying(Fields::k{{asUpperCamelCase label}}):
130-
ReturnErrorOnFailure(DataModel::Decode(reader, {{asLowerCamelCase label}}));
131-
break;
132-
{{/zcl_struct_items}}
133-
default:
134-
break;
135137
}
136-
}
137-
138-
VerifyOrReturnError(err == CHIP_END_OF_TLV, err);
139-
ReturnErrorOnFailure(reader.ExitContainer(outer));
140138

141-
return CHIP_NO_ERROR;
139+
ReturnErrorOnFailure(err);
140+
{{/last}}
141+
{{/zcl_struct_items}}
142+
}
142143
}
143144

144145
} // namespace {{asUpperCamelCase name}}

src/app/zap-templates/templates/app/cluster-objects-src.zapt

+85-72
Original file line numberDiff line numberDiff line change
@@ -1,46 +1,56 @@
11
{{> header}}
22

33
#include <app-common/zap-generated/cluster-objects.h>
4+
#include <variant>
45

56
namespace chip {
67
namespace app {
78
namespace Clusters {
89

910
namespace detail {
1011

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+
}
1736

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)) {
2639
continue;
27-
}
28-
}
29-
VerifyOrReturnError(err == CHIP_END_OF_TLV, err);
30-
ReturnErrorOnFailure(temp_reader.ExitContainer(outer));
40+
}
3141

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+
}
3645

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+
};
4454

4555
// Structs shared across multiple clusters.
4656
namespace Structs {
@@ -75,34 +85,37 @@ CHIP_ERROR Type::Encode(TLV::TLVWriter & aWriter, TLV::Tag aTag) const{
7585
{{#zcl_command_arguments}}
7686
ReturnErrorOnFailure(DataModel::Encode(aWriter, TLV::ContextTag(Fields::k{{asUpperCamelCase label}}), {{asLowerCamelCase label}}));
7787
{{/zcl_command_arguments}}
78-
ReturnErrorOnFailure(aWriter.EndContainer(outer));
79-
return CHIP_NO_ERROR;
88+
return aWriter.EndContainer(outer);
8089
}
8190

8291
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);
85103

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}}))
89107
{
90-
continue;
108+
err = DataModel::Decode(reader, {{asLowerCamelCase label}});
91109
}
92-
switch (TLV::TagNumFromTag(reader.GetTag()))
110+
else
111+
{{#last}}
93112
{
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;
101113
}
102-
}
103114

104-
chip::app::Clusters::detail::ExitStructAfterDecode(reader, outer);
105-
return CHIP_NO_ERROR;
115+
ReturnErrorOnFailure(err);
116+
{{/last}}
117+
{{/zcl_command_arguments}}
118+
}
106119
}
107120
} // namespace {{asUpperCamelCase name}}.
108121
{{/zcl_commands}}
@@ -114,14 +127,11 @@ CHIP_ERROR TypeInfo::DecodableType::Decode(TLV::TLVReader &reader, const Concret
114127
{
115128
{{#zcl_attributes_server}}
116129
case Attributes::{{asUpperCamelCase label}}::TypeInfo::GetAttributeId():
117-
ReturnErrorOnFailure(DataModel::Decode(reader, {{asLowerCamelCase label}}));
118-
break;
130+
return DataModel::Decode(reader, {{asLowerCamelCase label}});
119131
{{/zcl_attributes_server}}
120132
default:
121-
break;
133+
return CHIP_NO_ERROR;
122134
}
123-
124-
return CHIP_NO_ERROR;
125135
}
126136
} // namespace Attributes
127137

@@ -138,34 +148,37 @@ CHIP_ERROR Type::Encode(TLV::TLVWriter & aWriter, TLV::Tag aTag) const{
138148
ReturnErrorOnFailure(DataModel::Encode(aWriter, TLV::ContextTag(Fields::k{{asUpperCamelCase name}}), {{asLowerCamelCase name}}));
139149
{{/if_is_fabric_scoped_struct}}
140150
{{/zcl_event_fields}}
141-
ReturnErrorOnFailure(aWriter.EndContainer(outer));
142-
return CHIP_NO_ERROR;
151+
return aWriter.EndContainer(outer);
143152
}
144153

145154
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);
148166

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}}))
152170
{
153-
continue;
171+
err = DataModel::Decode(reader, {{asLowerCamelCase name}});
154172
}
155-
switch (TLV::TagNumFromTag(reader.GetTag()))
173+
else
174+
{{#last}}
156175
{
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;
164176
}
165-
}
166177

167-
chip::app::Clusters::detail::ExitStructAfterDecode(reader, outer);
168-
return CHIP_NO_ERROR;
178+
ReturnErrorOnFailure(err);
179+
{{/last}}
180+
{{/zcl_event_fields}}
181+
}
169182
}
170183
} // namespace {{asUpperCamelCase name}}.
171184
{{/zcl_events}}

0 commit comments

Comments
 (0)