Skip to content

Commit 08df52c

Browse files
authored
Implement a Vector-based TLVBackingStore (project-chip#32508)
* Implement a Vector-based TLVBackingStore for converting from stream-based input to TLV format * :Address comments * Fix for esp32 qemu build
1 parent a0b2fe2 commit 08df52c

File tree

7 files changed

+375
-0
lines changed

7 files changed

+375
-0
lines changed

scripts/tools/check_includes_config.py

+2
Original file line numberDiff line numberDiff line change
@@ -167,6 +167,8 @@
167167
'src/app/PendingResponseTrackerImpl.h': {'unordered_set'},
168168

169169
# Not intended for embedded clients
170+
'src/lib/core/TLVVectorWriter.cpp': {'vector'},
171+
'src/lib/core/TLVVectorWriter.h': {'vector'},
170172
'src/lib/support/jsontlv/JsonToTlv.cpp': {'sstream', 'string', 'vector'},
171173
'src/lib/support/jsontlv/JsonToTlv.h': {'string'},
172174
'src/lib/support/jsontlv/TlvToJson.h': {'string'},

src/lib/core/BUILD.gn

+14
Original file line numberDiff line numberDiff line change
@@ -187,3 +187,17 @@ static_library("core") {
187187
"${chip_root}/src/system",
188188
]
189189
}
190+
191+
static_library("vectortlv") {
192+
output_name = "libVectorTlv"
193+
output_dir = "${root_out_dir}/lib"
194+
195+
sources = [
196+
"TLVVectorWriter.cpp",
197+
"TLVVectorWriter.h",
198+
]
199+
200+
cflags = [ "-Wconversion" ]
201+
202+
public_deps = [ ":core" ]
203+
}

src/lib/core/TLVVectorWriter.cpp

+93
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
/*
2+
*
3+
* Copyright (c) 2024 Project CHIP Authors
4+
*
5+
* Licensed under the Apache License, Version 2.0 (the "License");
6+
* you may not use this file except in compliance with the License.
7+
* You may obtain a copy of the License at
8+
*
9+
* http://www.apache.org/licenses/LICENSE-2.0
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS,
13+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
* See the License for the specific language governing permissions and
15+
* limitations under the License.
16+
*/
17+
18+
#include <lib/core/TLVVectorWriter.h>
19+
20+
#include <cstdint>
21+
#include <vector>
22+
23+
#include <lib/core/CHIPError.h>
24+
#include <lib/core/TLVCommon.h>
25+
26+
namespace chip {
27+
namespace TLV {
28+
29+
namespace {
30+
31+
constexpr uint32_t kIpv6MtuSize = 1280;
32+
33+
} // namespace
34+
35+
TlvVectorWriter::TlvVectorWriter(std::vector<uint8_t> & buffer) : mVectorBuffer(buffer)
36+
{
37+
Init(mVectorBuffer);
38+
}
39+
40+
TlvVectorWriter::~TlvVectorWriter() = default;
41+
42+
TlvVectorWriter::TlvVectorBuffer::TlvVectorBuffer(std::vector<uint8_t> & buffer) : mFinalBuffer(buffer) {}
43+
44+
TlvVectorWriter::TlvVectorBuffer::~TlvVectorBuffer() {}
45+
46+
CHIP_ERROR TlvVectorWriter::TlvVectorBuffer::OnInit(TLVWriter & /*writer*/, uint8_t *& bufStart, uint32_t & bufLen)
47+
{
48+
VerifyOrReturnError(mFinalBuffer.empty(), CHIP_ERROR_INCORRECT_STATE);
49+
50+
ResizeWriteBuffer(bufStart, bufLen);
51+
52+
return CHIP_NO_ERROR;
53+
}
54+
55+
CHIP_ERROR TlvVectorWriter::TlvVectorBuffer::GetNewBuffer(TLVWriter & /*writer*/, uint8_t *& bufStart, uint32_t & bufLen)
56+
{
57+
VerifyOrReturnError(!mFinalBuffer.empty(), CHIP_ERROR_INCORRECT_STATE);
58+
VerifyOrReturnError(mWritingBuffer.data() == bufStart, CHIP_ERROR_INCORRECT_STATE);
59+
60+
ResizeWriteBuffer(bufStart, bufLen);
61+
62+
return CHIP_NO_ERROR;
63+
}
64+
65+
CHIP_ERROR TlvVectorWriter::TlvVectorBuffer::FinalizeBuffer(TLVWriter & /*writer*/, uint8_t * bufStart, uint32_t bufLen)
66+
{
67+
VerifyOrReturnError(mWritingBuffer.data() == bufStart, CHIP_ERROR_INCORRECT_STATE);
68+
VerifyOrReturnError(bufLen <= mWritingBuffer.size(), CHIP_ERROR_BUFFER_TOO_SMALL);
69+
70+
mWritingBuffer.resize(bufLen);
71+
72+
mFinalBuffer.insert(mFinalBuffer.end(), mWritingBuffer.begin(), mWritingBuffer.end());
73+
mWritingBuffer.resize(0);
74+
75+
mFinalBuffer.shrink_to_fit();
76+
77+
return CHIP_NO_ERROR;
78+
}
79+
80+
void TlvVectorWriter::TlvVectorBuffer::ResizeWriteBuffer(uint8_t *& bufStart, uint32_t & bufLen)
81+
{
82+
VerifyOrReturn(mWritingBuffer.empty());
83+
84+
mWritingBuffer.resize(kIpv6MtuSize);
85+
bufStart = mWritingBuffer.data();
86+
87+
auto size = mWritingBuffer.size();
88+
VerifyOrReturn(size <= std::numeric_limits<uint32_t>::max());
89+
bufLen = static_cast<uint32_t>(size);
90+
}
91+
92+
} // namespace TLV
93+
} // namespace chip

src/lib/core/TLVVectorWriter.h

+83
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
/*
2+
*
3+
* Copyright (c) 2024 Project CHIP Authors
4+
*
5+
* Licensed under the Apache License, Version 2.0 (the "License");
6+
* you may not use this file except in compliance with the License.
7+
* You may obtain a copy of the License at
8+
*
9+
* http://www.apache.org/licenses/LICENSE-2.0
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS,
13+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
* See the License for the specific language governing permissions and
15+
* limitations under the License.
16+
*/
17+
#pragma once
18+
19+
#include <cstdint>
20+
#include <vector>
21+
22+
#include <lib/core/CHIPError.h>
23+
#include <lib/core/TLVBackingStore.h>
24+
#include <lib/core/TLVCommon.h>
25+
26+
namespace chip {
27+
namespace TLV {
28+
29+
// Implementation of TLVWriter that writes to a std::vector, automatically
30+
// resizing the vector as needed when writing.
31+
// Users of TlvVectorWriter may call any public API of TLVWriter, except for the
32+
// Init functions.
33+
// This class is not thread-safe, it must be constructed, used, and destroyed on
34+
// a single thread.
35+
class TlvVectorWriter : public TLVWriter
36+
{
37+
public:
38+
// All data will be written to and read from the provided buffer, which must
39+
// outlive this object.
40+
TlvVectorWriter(std::vector<uint8_t> & buffer);
41+
TlvVectorWriter(const TlvVectorWriter &) = delete;
42+
TlvVectorWriter & operator=(const TlvVectorWriter &) = delete;
43+
~TlvVectorWriter();
44+
45+
private:
46+
class TlvVectorBuffer : public TLVBackingStore
47+
{
48+
public:
49+
TlvVectorBuffer(std::vector<uint8_t> & buffer);
50+
TlvVectorBuffer(const TlvVectorBuffer &) = delete;
51+
TlvVectorBuffer & operator=(const TlvVectorBuffer &) = delete;
52+
~TlvVectorBuffer() override;
53+
54+
// TLVBackingStore implementation:
55+
CHIP_ERROR OnInit(TLVReader & reader, const uint8_t *& bufStart, uint32_t & bufLen) override
56+
{
57+
return CHIP_ERROR_NOT_IMPLEMENTED;
58+
}
59+
CHIP_ERROR GetNextBuffer(TLVReader & reader, const uint8_t *& bufStart, uint32_t & bufLen) override
60+
{
61+
return CHIP_ERROR_NOT_IMPLEMENTED;
62+
}
63+
CHIP_ERROR OnInit(TLVWriter & writer, uint8_t *& bufStart, uint32_t & bufLen) override;
64+
CHIP_ERROR GetNewBuffer(TLVWriter & writer, uint8_t *& bufStart, uint32_t & bufLen) override;
65+
CHIP_ERROR FinalizeBuffer(TLVWriter & writer, uint8_t * bufStart, uint32_t bufLen) override;
66+
67+
private:
68+
void ResizeWriteBuffer(uint8_t *& bufStart, uint32_t & bufLen);
69+
70+
// mWritingBuffer is the mutable buffer exposed via the TLVBackingStore
71+
// interface. When FinalizeBuffer is called the contents of mWritingBuffer
72+
// are appended to mFinalBuffer and mWritingBuffer is cleared. This allows
73+
// for reading all written data from a single, contiguous buffer
74+
// (mFinalBuffer).
75+
std::vector<uint8_t> mWritingBuffer;
76+
std::vector<uint8_t> & mFinalBuffer;
77+
};
78+
79+
TlvVectorBuffer mVectorBuffer;
80+
};
81+
82+
} // namespace TLV
83+
} // namespace chip

src/lib/core/tests/BUILD.gn

+2
Original file line numberDiff line numberDiff line change
@@ -30,12 +30,14 @@ chip_test_suite_using_nltest("tests") {
3030
"TestOptional.cpp",
3131
"TestReferenceCounted.cpp",
3232
"TestTLV.cpp",
33+
"TestTLVVectorWriter.cpp",
3334
]
3435

3536
cflags = [ "-Wconversion" ]
3637

3738
public_deps = [
3839
"${chip_root}/src/lib/core",
40+
"${chip_root}/src/lib/core:vectortlv",
3941
"${chip_root}/src/lib/support:test_utils",
4042
"${chip_root}/src/lib/support:testing_nlunit",
4143
"${chip_root}/src/platform",

0 commit comments

Comments
 (0)