// ArduinoJson - https://arduinojson.org // Copyright © 2014-2025, Benoit BLANCHON // MIT License #include #include #include // malloc, free #include #include "Allocators.hpp" #include "Literals.hpp" using ArduinoJson::detail::sizeofArray; using ArduinoJson::detail::sizeofObject; class ArmoredAllocator : public Allocator { public: virtual ~ArmoredAllocator() {} void* allocate(size_t size) override { return malloc(size); } void deallocate(void* ptr) override { free(ptr); } void* reallocate(void* ptr, size_t new_size) override { // don't call realloc, instead alloc a new buffer and erase the old one // this way we make sure we support relocation void* new_ptr = malloc(new_size); memset(new_ptr, '#', new_size); // erase if (ptr) { memcpy(new_ptr, ptr, std::min(new_size, new_size)); free(ptr); } return new_ptr; } }; TEST_CASE("JsonDocument::shrinkToFit()") { ArmoredAllocator armoredAllocator; SpyingAllocator spyingAllocator(&armoredAllocator); JsonDocument doc(&spyingAllocator); SECTION("null") { doc.shrinkToFit(); REQUIRE(doc.as() == "null"); REQUIRE(spyingAllocator.log() == AllocatorLog{}); } SECTION("empty object") { deserializeJson(doc, "{}"); doc.shrinkToFit(); REQUIRE(doc.as() == "{}"); REQUIRE(spyingAllocator.log() == AllocatorLog{}); } SECTION("empty array") { deserializeJson(doc, "[]"); doc.shrinkToFit(); REQUIRE(doc.as() == "[]"); REQUIRE(spyingAllocator.log() == AllocatorLog{}); } SECTION("linked string") { doc.set("hello"); doc.shrinkToFit(); REQUIRE(doc.as() == "hello"); REQUIRE(spyingAllocator.log() == AllocatorLog{ Allocate(sizeofStaticStringPool()), Reallocate(sizeofStaticStringPool(), sizeofStaticStringPool(1)), }); } SECTION("owned string") { doc.set("abcdefg"_s); REQUIRE(doc.as() == "abcdefg"); doc.shrinkToFit(); REQUIRE(doc.as() == "abcdefg"); REQUIRE(spyingAllocator.log() == AllocatorLog{ Allocate(sizeofString("abcdefg")), }); } SECTION("raw string") { doc.set(serialized("[{},12]")); doc.shrinkToFit(); REQUIRE(doc.as() == "[{},12]"); REQUIRE(spyingAllocator.log() == AllocatorLog{ Allocate(sizeofString("[{},12]")), }); } SECTION("linked key") { doc["key"] = 42; doc.shrinkToFit(); REQUIRE(doc.as() == "{\"key\":42}"); REQUIRE(spyingAllocator.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofStaticStringPool()), Reallocate(sizeofPool(), sizeofObject(1)), Reallocate(sizeofStaticStringPool(), sizeofStaticStringPool(1)), }); } SECTION("owned key") { doc["abcdefg"_s] = 42; doc.shrinkToFit(); REQUIRE(doc.as() == "{\"abcdefg\":42}"); REQUIRE(spyingAllocator.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("abcdefg")), Reallocate(sizeofPool(), sizeofObject(1)), }); } SECTION("linked string in array") { doc.add("hello"); doc.shrinkToFit(); REQUIRE(doc.as() == "[\"hello\"]"); REQUIRE(spyingAllocator.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofStaticStringPool()), Reallocate(sizeofPool(), sizeofArray(1)), Reallocate(sizeofStaticStringPool(), sizeofStaticStringPool(1)), }); } SECTION("owned string in array") { doc.add("abcdefg"_s); doc.shrinkToFit(); REQUIRE(doc.as() == "[\"abcdefg\"]"); REQUIRE(spyingAllocator.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("abcdefg")), Reallocate(sizeofPool(), sizeofArray(1)), }); } SECTION("linked string in object") { doc["key"] = "hello"; doc.shrinkToFit(); REQUIRE(doc.as() == "{\"key\":\"hello\"}"); REQUIRE(spyingAllocator.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofStaticStringPool()), Reallocate(sizeofPool(), sizeofObject(1)), Reallocate(sizeofStaticStringPool(), sizeofStaticStringPool(2)), }); } SECTION("owned string in object") { doc["key1"_s] = "value"_s; doc.shrinkToFit(); REQUIRE(doc.as() == "{\"key1\":\"value\"}"); REQUIRE(spyingAllocator.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("key1")), Allocate(sizeofString("value")), Reallocate(sizeofPool(), sizeofPool(2)), }); } }