Added JsonDocument::overflowed() (closes #1358)

This commit is contained in:
Benoit Blanchon
2020-09-05 10:54:46 +02:00
parent 6d2ad4539f
commit 8d37939086
13 changed files with 155 additions and 64 deletions

View File

@ -5,6 +5,7 @@ HEAD
---- ----
* Added a build failure when nullptr is defined as a macro (issue #1355) * Added a build failure when nullptr is defined as a macro (issue #1355)
* Added `JsonDocument::overflowed()` which tells if the memory pool was too small (issue #1358)
v6.16.1 (2020-08-04) v6.16.1 (2020-08-04)
------- -------

View File

@ -11,6 +11,7 @@ add_executable(JsonDocumentTests
DynamicJsonDocument.cpp DynamicJsonDocument.cpp
isNull.cpp isNull.cpp
nesting.cpp nesting.cpp
overflowed.cpp
remove.cpp remove.cpp
shrinkToFit.cpp shrinkToFit.cpp
size.cpp size.cpp

View File

@ -0,0 +1,79 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2020
// MIT License
#include <ArduinoJson.h>
#include <catch.hpp>
TEST_CASE("JsonDocument::overflowed()") {
SECTION("returns false on a fresh object") {
StaticJsonDocument<0> doc;
CHECK(doc.overflowed() == false);
}
SECTION("returns true after a failed insertion") {
StaticJsonDocument<0> doc;
doc.add(0);
CHECK(doc.overflowed() == true);
}
SECTION("returns false after successful insertion") {
StaticJsonDocument<JSON_ARRAY_SIZE(1)> doc;
doc.add(0);
CHECK(doc.overflowed() == false);
}
SECTION("returns true after a failed string copy") {
StaticJsonDocument<JSON_ARRAY_SIZE(1)> doc;
doc.add(std::string("example"));
CHECK(doc.overflowed() == true);
}
SECTION("returns false after a successful string copy") {
StaticJsonDocument<JSON_ARRAY_SIZE(1) + 8> doc;
doc.add(std::string("example"));
CHECK(doc.overflowed() == false);
}
SECTION("returns true after a failed deserialization") {
StaticJsonDocument<JSON_ARRAY_SIZE(1)> doc;
deserializeJson(doc, "[\"example\"]");
CHECK(doc.overflowed() == true);
}
SECTION("returns false after a successful deserialization") {
StaticJsonDocument<JSON_ARRAY_SIZE(1) + 8> doc;
deserializeJson(doc, "[\"example\"]");
CHECK(doc.overflowed() == false);
}
SECTION("returns false after clear()") {
StaticJsonDocument<0> doc;
doc.add(0);
doc.clear();
CHECK(doc.overflowed() == false);
}
SECTION("remains false after shrinkToFit()") {
DynamicJsonDocument doc(JSON_ARRAY_SIZE(1));
doc.add(0);
doc.shrinkToFit();
CHECK(doc.overflowed() == false);
}
SECTION("remains true after shrinkToFit()") {
DynamicJsonDocument doc(JSON_ARRAY_SIZE(1));
doc.add(0);
doc.add(0);
doc.shrinkToFit();
CHECK(doc.overflowed() == true);
}
SECTION("return false after garbageCollect()") {
DynamicJsonDocument doc(JSON_ARRAY_SIZE(1));
doc.add(0);
doc.add(0);
doc.garbageCollect();
CHECK(doc.overflowed() == false);
}
}

View File

@ -12,9 +12,9 @@ TEST_CASE("StringCopier") {
SECTION("Works when buffer is big enough") { SECTION("Works when buffer is big enough") {
MemoryPool pool(buffer, addPadding(JSON_STRING_SIZE(6))); MemoryPool pool(buffer, addPadding(JSON_STRING_SIZE(6)));
StringCopier str; StringCopier str(pool);
str.startString(&pool); str.startString();
str.append("hello"); str.append("hello");
str.append('\0'); str.append('\0');
@ -24,9 +24,9 @@ TEST_CASE("StringCopier") {
SECTION("Returns null when too small") { SECTION("Returns null when too small") {
MemoryPool pool(buffer, sizeof(void*)); MemoryPool pool(buffer, sizeof(void*));
StringCopier str; StringCopier str(pool);
str.startString(&pool); str.startString();
str.append("hello world!"); str.append("hello world!");
REQUIRE(str.isValid() == false); REQUIRE(str.isValid() == false);
@ -34,22 +34,22 @@ TEST_CASE("StringCopier") {
SECTION("Increases size of memory pool") { SECTION("Increases size of memory pool") {
MemoryPool pool(buffer, addPadding(JSON_STRING_SIZE(6))); MemoryPool pool(buffer, addPadding(JSON_STRING_SIZE(6)));
StringCopier str; StringCopier str(pool);
str.startString(&pool); str.startString();
str.append('h'); str.append('h');
str.save(&pool); str.save();
REQUIRE(1 == pool.size()); REQUIRE(1 == pool.size());
} }
} }
static const char* addStringToPool(MemoryPool* pool, const char* s) { static const char* addStringToPool(MemoryPool& pool, const char* s) {
StringCopier str; StringCopier str(pool);
str.startString(pool); str.startString();
str.append(s); str.append(s);
str.append('\0'); str.append('\0');
return str.save(pool); return str.save();
} }
TEST_CASE("StringCopier::save() deduplicates strings") { TEST_CASE("StringCopier::save() deduplicates strings") {
@ -57,9 +57,9 @@ TEST_CASE("StringCopier::save() deduplicates strings") {
MemoryPool pool(buffer, 4096); MemoryPool pool(buffer, 4096);
SECTION("Basic") { SECTION("Basic") {
const char* s1 = addStringToPool(&pool, "hello"); const char* s1 = addStringToPool(pool, "hello");
const char* s2 = addStringToPool(&pool, "world"); const char* s2 = addStringToPool(pool, "world");
const char* s3 = addStringToPool(&pool, "hello"); const char* s3 = addStringToPool(pool, "hello");
REQUIRE(s1 == s3); REQUIRE(s1 == s3);
REQUIRE(s2 != s3); REQUIRE(s2 != s3);
@ -67,16 +67,16 @@ TEST_CASE("StringCopier::save() deduplicates strings") {
} }
SECTION("Requires terminator") { SECTION("Requires terminator") {
const char* s1 = addStringToPool(&pool, "hello world"); const char* s1 = addStringToPool(pool, "hello world");
const char* s2 = addStringToPool(&pool, "hello"); const char* s2 = addStringToPool(pool, "hello");
REQUIRE(s2 != s1); REQUIRE(s2 != s1);
REQUIRE(pool.size() == 12 + 6); REQUIRE(pool.size() == 12 + 6);
} }
SECTION("Don't overrun") { SECTION("Don't overrun") {
const char* s1 = addStringToPool(&pool, "hello world"); const char* s1 = addStringToPool(pool, "hello world");
const char* s2 = addStringToPool(&pool, "wor"); const char* s2 = addStringToPool(pool, "wor");
REQUIRE(s2 != s1); REQUIRE(s2 != s1);
REQUIRE(pool.size() == 12 + 4); REQUIRE(pool.size() == 12 + 4);

View File

@ -12,8 +12,8 @@ using namespace ARDUINOJSON_NAMESPACE;
static void testCodepoint(uint32_t codepoint, std::string expected) { static void testCodepoint(uint32_t codepoint, std::string expected) {
char buffer[4096]; char buffer[4096];
MemoryPool pool(buffer, 4096); MemoryPool pool(buffer, 4096);
StringCopier str; StringCopier str(pool);
str.startString(&pool); str.startString();
CAPTURE(codepoint); CAPTURE(codepoint);
Utf8::encodeCodepoint(codepoint, str); Utf8::encodeCodepoint(codepoint, str);

View File

@ -32,8 +32,9 @@ deserialize(JsonDocument &doc, const TString &input, NestingLimit nestingLimit,
TFilter filter) { TFilter filter) {
Reader<TString> reader(input); Reader<TString> reader(input);
doc.clear(); doc.clear();
return makeDeserializer<TDeserializer>(doc.memoryPool(), reader, return makeDeserializer<TDeserializer>(
makeStringStorage(input)) doc.memoryPool(), reader,
makeStringStorage(input, doc.memoryPool()))
.parse(doc.data(), filter, nestingLimit); .parse(doc.data(), filter, nestingLimit);
} }
// //
@ -47,8 +48,9 @@ DeserializationError deserialize(JsonDocument &doc, TChar *input,
TFilter filter) { TFilter filter) {
BoundedReader<TChar *> reader(input, inputSize); BoundedReader<TChar *> reader(input, inputSize);
doc.clear(); doc.clear();
return makeDeserializer<TDeserializer>(doc.memoryPool(), reader, return makeDeserializer<TDeserializer>(
makeStringStorage(input)) doc.memoryPool(), reader,
makeStringStorage(input, doc.memoryPool()))
.parse(doc.data(), filter, nestingLimit); .parse(doc.data(), filter, nestingLimit);
} }
// //
@ -60,8 +62,9 @@ DeserializationError deserialize(JsonDocument &doc, TStream &input,
NestingLimit nestingLimit, TFilter filter) { NestingLimit nestingLimit, TFilter filter) {
Reader<TStream> reader(input); Reader<TStream> reader(input);
doc.clear(); doc.clear();
return makeDeserializer<TDeserializer>(doc.memoryPool(), reader, return makeDeserializer<TDeserializer>(
makeStringStorage(input)) doc.memoryPool(), reader,
makeStringStorage(input, doc.memoryPool()))
.parse(doc.data(), filter, nestingLimit); .parse(doc.data(), filter, nestingLimit);
} }

View File

@ -48,6 +48,10 @@ class JsonDocument : public Visitable {
return _pool.size(); return _pool.size();
} }
bool overflowed() const {
return _pool.overflowed();
}
size_t nesting() const { size_t nesting() const {
return _data.nesting(); return _data.nesting();
} }
@ -81,6 +85,7 @@ class JsonDocument : public Visitable {
return _pool; return _pool;
} }
// for internal use only
VariantData& data() { VariantData& data() {
return _data; return _data;
} }

View File

@ -241,7 +241,7 @@ class JsonDeserializer {
if (!variant) { if (!variant) {
// Save key in memory pool. // Save key in memory pool.
// This MUST be done before adding the slot. // This MUST be done before adding the slot.
key = _stringStorage.save(_pool); key = _stringStorage.save();
// Allocate slot in object // Allocate slot in object
VariantSlot *slot = object.addSlot(_pool); VariantSlot *slot = object.addSlot(_pool);
@ -334,7 +334,7 @@ class JsonDeserializer {
} }
bool parseKey() { bool parseKey() {
_stringStorage.startString(_pool); _stringStorage.startString();
if (isQuote(current())) { if (isQuote(current())) {
return parseQuotedString(); return parseQuotedString();
} else { } else {
@ -343,10 +343,10 @@ class JsonDeserializer {
} }
bool parseStringValue(VariantData &variant) { bool parseStringValue(VariantData &variant) {
_stringStorage.startString(_pool); _stringStorage.startString();
if (!parseQuotedString()) if (!parseQuotedString())
return false; return false;
const char *value = _stringStorage.save(_pool); const char *value = _stringStorage.save();
variant.setString(make_not_null(value), variant.setString(make_not_null(value),
typename TStringStorage::storage_policy()); typename TStringStorage::storage_policy());
return true; return true;

View File

@ -29,7 +29,8 @@ class MemoryPool {
: _begin(buf), : _begin(buf),
_left(buf), _left(buf),
_right(buf ? buf + capa : 0), _right(buf ? buf + capa : 0),
_end(buf ? buf + capa : 0) { _end(buf ? buf + capa : 0),
_overflowed(false) {
ARDUINOJSON_ASSERT(isAligned(_begin)); ARDUINOJSON_ASSERT(isAligned(_begin));
ARDUINOJSON_ASSERT(isAligned(_right)); ARDUINOJSON_ASSERT(isAligned(_right));
ARDUINOJSON_ASSERT(isAligned(_end)); ARDUINOJSON_ASSERT(isAligned(_end));
@ -48,6 +49,10 @@ class MemoryPool {
return size_t(_left - _begin + _end - _right); return size_t(_left - _begin + _end - _right);
} }
bool overflowed() const {
return _overflowed;
}
VariantSlot* allocVariant() { VariantSlot* allocVariant() {
return allocRight<VariantSlot>(); return allocRight<VariantSlot>();
} }
@ -91,9 +96,14 @@ class MemoryPool {
return str; return str;
} }
void markAsOverflowed() {
_overflowed = true;
}
void clear() { void clear() {
_left = _begin; _left = _begin;
_right = _end; _right = _end;
_overflowed = false;
} }
bool canAlloc(size_t bytes) const { bool canAlloc(size_t bytes) const {
@ -171,8 +181,10 @@ class MemoryPool {
#endif #endif
char* allocString(size_t n) { char* allocString(size_t n) {
if (!canAlloc(n)) if (!canAlloc(n)) {
_overflowed = true;
return 0; return 0;
}
char* s = _left; char* s = _left;
_left += n; _left += n;
checkInvariants(); checkInvariants();
@ -185,13 +197,16 @@ class MemoryPool {
} }
void* allocRight(size_t bytes) { void* allocRight(size_t bytes) {
if (!canAlloc(bytes)) if (!canAlloc(bytes)) {
_overflowed = true;
return 0; return 0;
}
_right -= bytes; _right -= bytes;
return _right; return _right;
} }
char *_begin, *_left, *_right, *_end; char *_begin, *_left, *_right, *_end;
bool _overflowed;
}; };
} // namespace ARDUINOJSON_NAMESPACE } // namespace ARDUINOJSON_NAMESPACE

View File

@ -239,7 +239,7 @@ class MsgPackDeserializer {
} }
bool readString(const char *&result, size_t n) { bool readString(const char *&result, size_t n) {
_stringStorage.startString(_pool); _stringStorage.startString();
for (; n; --n) { for (; n; --n) {
uint8_t c; uint8_t c;
if (!readBytes(c)) if (!readBytes(c))
@ -252,7 +252,7 @@ class MsgPackDeserializer {
return false; return false;
} }
result = _stringStorage.save(_pool); result = _stringStorage.save();
return true; return true;
} }

View File

@ -10,14 +10,16 @@ namespace ARDUINOJSON_NAMESPACE {
class StringCopier { class StringCopier {
public: public:
void startString(MemoryPool* pool) { StringCopier(MemoryPool& pool) : _pool(&pool) {}
pool->getFreeZone(&_ptr, &_capacity);
void startString() {
_pool->getFreeZone(&_ptr, &_capacity);
_size = 0; _size = 0;
} }
const char* save(MemoryPool* pool) { const char* save() {
ARDUINOJSON_ASSERT(_ptr); ARDUINOJSON_ASSERT(_ptr);
return pool->saveStringFromFreeZone(_size); return _pool->saveStringFromFreeZone(_size);
} }
void append(const char* s) { void append(const char* s) {
@ -34,6 +36,7 @@ class StringCopier {
if (_size >= _capacity) { if (_size >= _capacity) {
_ptr = 0; _ptr = 0;
_pool->markAsOverflowed();
return; return;
} }
@ -51,6 +54,7 @@ class StringCopier {
typedef storage_policies::store_by_copy storage_policy; typedef storage_policies::store_by_copy storage_policy;
private: private:
MemoryPool* _pool;
char* _ptr; char* _ptr;
size_t _size; size_t _size;
size_t _capacity; size_t _capacity;

View File

@ -13,11 +13,11 @@ class StringMover {
public: public:
StringMover(char* ptr) : _writePtr(ptr) {} StringMover(char* ptr) : _writePtr(ptr) {}
void startString(MemoryPool*) { void startString() {
_startPtr = _writePtr; _startPtr = _writePtr;
} }
const char* save(MemoryPool*) const { const char* save() const {
return _startPtr; return _startPtr;
} }

View File

@ -9,32 +9,15 @@
namespace ARDUINOJSON_NAMESPACE { namespace ARDUINOJSON_NAMESPACE {
template <typename TInput, typename Enable = void>
struct StringStorage {
typedef StringCopier type;
static type create(TInput&) {
return type();
}
};
template <typename TChar>
struct StringStorage<TChar*,
typename enable_if<!is_const<TChar>::value>::type> {
typedef StringMover type;
static type create(TChar* input) {
return type(reinterpret_cast<char*>(input));
}
};
template <typename TInput> template <typename TInput>
typename StringStorage<TInput>::type makeStringStorage(TInput& input) { StringCopier makeStringStorage(TInput&, MemoryPool& pool) {
return StringStorage<TInput>::create(input); return StringCopier(pool);
} }
template <typename TChar> template <typename TChar>
typename StringStorage<TChar*>::type makeStringStorage(TChar* input) { StringMover makeStringStorage(
return StringStorage<TChar*>::create(input); TChar* input, MemoryPool&,
typename enable_if<!is_const<TChar>::value>::type* = 0) {
return StringMover(reinterpret_cast<char*>(input));
} }
} // namespace ARDUINOJSON_NAMESPACE } // namespace ARDUINOJSON_NAMESPACE