Skip to content

Commit 9b58c27

Browse files
authored
Patch unsafe strlen in TLVWriter (project-chip#37065)
* Patch unsafe strlen in TLVWriter This is a vulnerability from Weave that was recently fixed. Apply the patch to Matter TLVWriter as well. This avoids reading bad pointers beyond stack-allocated memory. > One of the PutString function overloads makes a call to strlen > without safeguards. This has caused faults on several products when > passing in uninitialized memory. While these call sites have been > fixed with explicit initialization, we can also make the core > library more secure. Use the container to determine a maximum length > and avoid buffer overflow. * Fix comment and apply clang-format * Fixes for clang-tidy
1 parent de04883 commit 9b58c27

File tree

2 files changed

+34
-4
lines changed

2 files changed

+34
-4
lines changed

src/lib/core/TLVWriter.cpp

+18-4
Original file line numberDiff line numberDiff line change
@@ -290,12 +290,26 @@ CHIP_ERROR TLVWriter::PutBytes(Tag tag, const uint8_t * buf, uint32_t len)
290290

291291
CHIP_ERROR TLVWriter::PutString(Tag tag, const char * buf)
292292
{
293-
size_t len = strlen(buf);
293+
if (buf == nullptr)
294+
return CHIP_ERROR_INVALID_ARGUMENT;
295+
if (mMaxLen == 0)
296+
return CHIP_ERROR_INCORRECT_STATE;
297+
298+
// Calculate length with a hard limit to prevent unbounded reads.
299+
// Use mMaxLen instead of mRemainingLen to account for CircularTLVWriter.
300+
// Note: Overrun is still possible if buf is not null-terminated, and this
301+
// check cannot prevent all invalid memory reads.
302+
size_t len = strnlen(buf, mMaxLen);
303+
294304
if (!CanCastTo<uint32_t>(len))
295-
{
296305
return CHIP_ERROR_INVALID_ARGUMENT;
297-
}
298-
return PutString(tag, buf, static_cast<uint32_t>(len));
306+
307+
uint32_t stringLen = static_cast<uint32_t>(len);
308+
309+
// Null terminator was not found within the allocated space.
310+
if (stringLen == mMaxLen)
311+
return CHIP_ERROR_BUFFER_TOO_SMALL;
312+
return PutString(tag, buf, stringLen);
299313
}
300314

301315
CHIP_ERROR TLVWriter::PutString(Tag tag, const char * buf, uint32_t len)

src/lib/core/tests/TestTLV.cpp

+16
Original file line numberDiff line numberDiff line change
@@ -2605,6 +2605,22 @@ TEST_F(TestTLV, CheckCircularTLVBufferEdge)
26052605

26062606
TestEnd<TLVReader>(reader);
26072607
}
2608+
2609+
TEST_F(TestTLV, CheckTLVPutStringOverrun)
2610+
{
2611+
const size_t bufSize = 8;
2612+
uint8_t backingStore[bufSize] = {};
2613+
2614+
const char * badPointer = "Foo <segfault here>";
2615+
2616+
TLVWriter writer;
2617+
2618+
writer.Init(backingStore, bufSize);
2619+
2620+
CHIP_ERROR err = writer.PutString(ProfileTag(TestProfile_1, 1), badPointer);
2621+
EXPECT_EQ(err, CHIP_ERROR_BUFFER_TOO_SMALL);
2622+
}
2623+
26082624
TEST_F(TestTLV, CheckTLVPutStringF)
26092625
{
26102626
const size_t bufsize = 24;

0 commit comments

Comments
 (0)