QCborValue::fromCbor: Apply a recursion limit to decoding

A simple 16k file can produce deep enough recursion in Qt to cause stack
overflow. So prevent that.

I tested 4096 recursions just fine on my Linux system (8 MB stack), but
decided 1024 was sufficient, as this code will also be run on embedded
systems that could have smaller stacks.

[ChangeLog][QtCore][QCborValue] fromCbor() now limits decoding to at
most 1024 nested maps, arrays, and tags to prevent stack overflows. This
should be sufficient for most uses of CBOR. An API to limit further or
to relax the limit will be provided in 5.15. Meanwhile, if decoding more
is required, QCborStreamReader can be used (note that each level of map
and array allocates memory).

Change-Id: Iaa63461109844e978376fffd15fa0fbefbf607a2
Reviewed-by: Lars Knoll <lars.knoll@qt.io>
(cherry picked from commit 02d595946faa7a21f6aa4109227f7e90db20ae7a)
Reviewed-by: Ulf Hermann <ulf.hermann@qt.io>
This commit is contained in:
Thiago Macieira 2020-03-07 07:38:51 -08:00
parent be28eb44a8
commit 07e8390b45
3 changed files with 84 additions and 13 deletions

View File

@ -1362,23 +1362,33 @@ static Element decodeBasicValueFromCbor(QCborStreamReader &reader)
return e; return e;
} }
static inline QCborContainerPrivate *createContainerFromCbor(QCborStreamReader &reader) static inline QCborContainerPrivate *createContainerFromCbor(QCborStreamReader &reader, int remainingRecursionDepth)
{ {
if (Q_UNLIKELY(remainingRecursionDepth == 0)) {
QCborContainerPrivate::setErrorInReader(reader, { QCborError::NestingTooDeep });
return nullptr;
}
auto d = new QCborContainerPrivate; auto d = new QCborContainerPrivate;
d->ref.store(1); d->ref.store(1);
d->decodeFromCbor(reader); d->decodeContainerFromCbor(reader, remainingRecursionDepth - 1);
return d; return d;
} }
static QCborValue taggedValueFromCbor(QCborStreamReader &reader) static QCborValue taggedValueFromCbor(QCborStreamReader &reader, int remainingRecursionDepth)
{ {
if (Q_UNLIKELY(remainingRecursionDepth == 0)) {
QCborContainerPrivate::setErrorInReader(reader, { QCborError::NestingTooDeep });
return QCborValue::Invalid;
}
auto d = new QCborContainerPrivate; auto d = new QCborContainerPrivate;
d->append(reader.toTag()); d->append(reader.toTag());
reader.next(); reader.next();
if (reader.lastError() == QCborError::NoError) { if (reader.lastError() == QCborError::NoError) {
// decode tagged value // decode tagged value
d->decodeValueFromCbor(reader); d->decodeValueFromCbor(reader, remainingRecursionDepth - 1);
} }
QCborValue::Type type = QCborValue::Tag; QCborValue::Type type = QCborValue::Tag;
@ -1586,9 +1596,10 @@ void QCborContainerPrivate::decodeStringFromCbor(QCborStreamReader &reader)
elements.append(e); elements.append(e);
} }
void QCborContainerPrivate::decodeValueFromCbor(QCborStreamReader &reader) void QCborContainerPrivate::decodeValueFromCbor(QCborStreamReader &reader, int remainingRecursionDepth)
{ {
switch (reader.type()) { QCborStreamReader::Type t = reader.type();
switch (t) {
case QCborStreamReader::UnsignedInteger: case QCborStreamReader::UnsignedInteger:
case QCborStreamReader::NegativeInteger: case QCborStreamReader::NegativeInteger:
case QCborStreamReader::SimpleType: case QCborStreamReader::SimpleType:
@ -1605,15 +1616,19 @@ void QCborContainerPrivate::decodeValueFromCbor(QCborStreamReader &reader)
case QCborStreamReader::Array: case QCborStreamReader::Array:
case QCborStreamReader::Map: case QCborStreamReader::Map:
return append(makeValue(t == QCborStreamReader::Array ? QCborValue::Array : QCborValue::Map, -1,
createContainerFromCbor(reader, remainingRecursionDepth),
MoveContainer));
case QCborStreamReader::Tag: case QCborStreamReader::Tag:
return append(QCborValue::fromCbor(reader)); return append(taggedValueFromCbor(reader, remainingRecursionDepth));
case QCborStreamReader::Invalid: case QCborStreamReader::Invalid:
return; // probably a decode error return; // probably a decode error
} }
} }
void QCborContainerPrivate::decodeFromCbor(QCborStreamReader &reader) void QCborContainerPrivate::decodeContainerFromCbor(QCborStreamReader &reader, int remainingRecursionDepth)
{ {
int mapShift = reader.isMap() ? 1 : 0; int mapShift = reader.isMap() ? 1 : 0;
if (reader.isLengthKnown()) { if (reader.isLengthKnown()) {
@ -1631,7 +1646,7 @@ void QCborContainerPrivate::decodeFromCbor(QCborStreamReader &reader)
return; return;
while (reader.hasNext() && reader.lastError() == QCborError::NoError) while (reader.hasNext() && reader.lastError() == QCborError::NoError)
decodeValueFromCbor(reader); decodeValueFromCbor(reader, remainingRecursionDepth);
if (reader.lastError() == QCborError::NoError) if (reader.lastError() == QCborError::NoError)
reader.leaveContainer(); reader.leaveContainer();
@ -2100,6 +2115,8 @@ const QCborValue QCborValue::operator[](qint64 key) const
return QCborValue(); return QCborValue();
} }
enum { MaximumRecursionDepth = 1024 };
/*! /*!
Decodes one item from the CBOR stream found in \a reader and returns the Decodes one item from the CBOR stream found in \a reader and returns the
equivalent representation. This function is recursive: if the item is a map equivalent representation. This function is recursive: if the item is a map
@ -2160,12 +2177,12 @@ QCborValue QCborValue::fromCbor(QCborStreamReader &reader)
case QCborStreamReader::Map: case QCborStreamReader::Map:
result.n = -1; result.n = -1;
result.t = reader.isArray() ? Array : Map; result.t = reader.isArray() ? Array : Map;
result.container = createContainerFromCbor(reader); result.container = createContainerFromCbor(reader, MaximumRecursionDepth);
break; break;
// tag // tag
case QCborStreamReader::Tag: case QCborStreamReader::Tag:
result = taggedValueFromCbor(reader); result = taggedValueFromCbor(reader, MaximumRecursionDepth);
break; break;
} }

View File

@ -389,8 +389,8 @@ public:
elements.remove(idx); elements.remove(idx);
} }
void decodeValueFromCbor(QCborStreamReader &reader); void decodeValueFromCbor(QCborStreamReader &reader, int remainiingStackDepth);
void decodeFromCbor(QCborStreamReader &reader); void decodeContainerFromCbor(QCborStreamReader &reader, int remainingStackDepth);
void decodeStringFromCbor(QCborStreamReader &reader); void decodeStringFromCbor(QCborStreamReader &reader);
static inline void setErrorInReader(QCborStreamReader &reader, QCborError error); static inline void setErrorInReader(QCborStreamReader &reader, QCborError error);
}; };

View File

@ -101,6 +101,8 @@ private slots:
void fromCborStreamReaderIODevice(); void fromCborStreamReaderIODevice();
void validation_data(); void validation_data();
void validation(); void validation();
void recursionLimit_data();
void recursionLimit();
void toDiagnosticNotation_data(); void toDiagnosticNotation_data();
void toDiagnosticNotation(); void toDiagnosticNotation();
}; };
@ -1581,6 +1583,58 @@ void tst_QCborValue::validation()
} }
} }
void tst_QCborValue::recursionLimit_data()
{
constexpr int RecursionAttempts = 4096;
QTest::addColumn<QByteArray>("data");
QByteArray arrays(RecursionAttempts, char(0x81));
QByteArray _arrays(RecursionAttempts, char(0x9f));
QByteArray maps(RecursionAttempts, char(0xa1));
QByteArray _maps(RecursionAttempts, char(0xbf));
QByteArray tags(RecursionAttempts, char(0xc0));
QTest::newRow("array-nesting-too-deep") << arrays;
QTest::newRow("_array-nesting-too-deep") << _arrays;
QTest::newRow("map-nesting-too-deep") << maps;
QTest::newRow("_map-nesting-too-deep") << _maps;
QTest::newRow("tag-nesting-too-deep") << tags;
QByteArray mixed(5 * RecursionAttempts, Qt::Uninitialized);
char *ptr = mixed.data();
for (int i = 0; i < RecursionAttempts; ++i) {
quint8 type = qBound(quint8(QCborStreamReader::Array), quint8(i & 0x80), quint8(QCborStreamReader::Tag));
quint8 additional_info = i & 0x1f;
if (additional_info == 0x1f)
(void)additional_info; // leave it
else if (additional_info > 0x1a)
additional_info = 0x1a;
else if (additional_info < 1)
additional_info = 1;
*ptr++ = type | additional_info;
if (additional_info == 0x18) {
*ptr++ = uchar(i);
} else if (additional_info == 0x19) {
qToBigEndian(ushort(i), ptr);
ptr += 2;
} else if (additional_info == 0x1a) {
qToBigEndian(uint(i), ptr);
ptr += 4;
}
}
QTest::newRow("mixed-nesting-too-deep") << mixed;
}
void tst_QCborValue::recursionLimit()
{
QFETCH(QByteArray, data);
QCborParserError error;
QCborValue decoded = QCborValue::fromCbor(data, &error);
QCOMPARE(error.error, QCborError::NestingTooDeep);
}
void tst_QCborValue::toDiagnosticNotation_data() void tst_QCborValue::toDiagnosticNotation_data()
{ {
QTest::addColumn<QCborValue>("v"); QTest::addColumn<QCborValue>("v");