mirror of
				https://github.com/bblanchon/ArduinoJson.git
				synced 2025-11-04 00:21:36 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			144 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			144 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
// ArduinoJson - https://arduinojson.org
 | 
						|
// Copyright © 2014-2023, Benoit BLANCHON
 | 
						|
// MIT License
 | 
						|
 | 
						|
#include <ArduinoJson.h>
 | 
						|
#include <catch.hpp>
 | 
						|
 | 
						|
#include <stdlib.h>  // malloc, free
 | 
						|
#include <string>
 | 
						|
 | 
						|
using ArduinoJson::detail::sizeofArray;
 | 
						|
using ArduinoJson::detail::sizeofObject;
 | 
						|
 | 
						|
class ArmoredAllocator : public Allocator {
 | 
						|
 public:
 | 
						|
  ArmoredAllocator() : _ptr(0), _size(0) {}
 | 
						|
  virtual ~ArmoredAllocator() {}
 | 
						|
 | 
						|
  void* allocate(size_t size) override {
 | 
						|
    _ptr = malloc(size);
 | 
						|
    _size = size;
 | 
						|
    return _ptr;
 | 
						|
  }
 | 
						|
 | 
						|
  void deallocate(void* ptr) override {
 | 
						|
    REQUIRE(ptr == _ptr);
 | 
						|
    free(ptr);
 | 
						|
    _ptr = 0;
 | 
						|
    _size = 0;
 | 
						|
  }
 | 
						|
 | 
						|
  void* reallocate(void* ptr, size_t new_size) override {
 | 
						|
    REQUIRE(ptr == _ptr);
 | 
						|
    // 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);
 | 
						|
    memcpy(new_ptr, _ptr, std::min(new_size, _size));
 | 
						|
    memset(_ptr, '#', _size);  // erase
 | 
						|
    free(_ptr);
 | 
						|
    _ptr = new_ptr;
 | 
						|
    return new_ptr;
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  void* _ptr;
 | 
						|
  size_t _size;
 | 
						|
};
 | 
						|
 | 
						|
void testShrinkToFit(JsonDocument& doc, std::string expected_json,
 | 
						|
                     size_t expected_size) {
 | 
						|
  // test twice: shrinkToFit() should be idempotent
 | 
						|
  for (int i = 0; i < 2; i++) {
 | 
						|
    doc.shrinkToFit();
 | 
						|
 | 
						|
    REQUIRE(doc.capacity() == expected_size);
 | 
						|
    REQUIRE(doc.memoryUsage() == expected_size);
 | 
						|
 | 
						|
    std::string json;
 | 
						|
    serializeJson(doc, json);
 | 
						|
    REQUIRE(json == expected_json);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
TEST_CASE("JsonDocument::shrinkToFit()") {
 | 
						|
  ArmoredAllocator armoredAllocator;
 | 
						|
  JsonDocument doc(4096, &armoredAllocator);
 | 
						|
 | 
						|
  SECTION("null") {
 | 
						|
    testShrinkToFit(doc, "null", 0);
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("empty object") {
 | 
						|
    deserializeJson(doc, "{}");
 | 
						|
    testShrinkToFit(doc, "{}", sizeofObject(0));
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("empty array") {
 | 
						|
    deserializeJson(doc, "[]");
 | 
						|
    testShrinkToFit(doc, "[]", sizeofArray(0));
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("linked string") {
 | 
						|
    doc.set("hello");
 | 
						|
    testShrinkToFit(doc, "\"hello\"", 0);
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("owned string") {
 | 
						|
    doc.set(std::string("abcdefg"));
 | 
						|
    testShrinkToFit(doc, "\"abcdefg\"", 8);
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("linked raw") {
 | 
						|
    doc.set(serialized("[{},123]"));
 | 
						|
    testShrinkToFit(doc, "[{},123]", 0);
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("owned raw") {
 | 
						|
    doc.set(serialized(std::string("[{},12]")));
 | 
						|
    testShrinkToFit(doc, "[{},12]", 8);
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("linked key") {
 | 
						|
    doc["key"] = 42;
 | 
						|
    testShrinkToFit(doc, "{\"key\":42}", sizeofObject(1));
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("owned key") {
 | 
						|
    doc[std::string("abcdefg")] = 42;
 | 
						|
    testShrinkToFit(doc, "{\"abcdefg\":42}", sizeofObject(1) + 8);
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("linked string in array") {
 | 
						|
    doc.add("hello");
 | 
						|
    testShrinkToFit(doc, "[\"hello\"]", sizeofArray(1));
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("owned string in array") {
 | 
						|
    doc.add(std::string("abcdefg"));
 | 
						|
    testShrinkToFit(doc, "[\"abcdefg\"]", sizeofArray(1) + 8);
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("linked string in object") {
 | 
						|
    doc["key"] = "hello";
 | 
						|
    testShrinkToFit(doc, "{\"key\":\"hello\"}", sizeofObject(1));
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("owned string in object") {
 | 
						|
    doc["key"] = std::string("abcdefg");
 | 
						|
    testShrinkToFit(doc, "{\"key\":\"abcdefg\"}", sizeofArray(1) + 8);
 | 
						|
  }
 | 
						|
 | 
						|
  SECTION("unaligned") {
 | 
						|
    doc.add(std::string("?"));  // two bytes in the string pool
 | 
						|
    REQUIRE(doc.memoryUsage() == sizeofObject(1) + 2);
 | 
						|
 | 
						|
    doc.shrinkToFit();
 | 
						|
 | 
						|
    // the new capacity should be padded to align the pointers
 | 
						|
    REQUIRE(doc.capacity() == sizeofObject(1) + sizeof(void*));
 | 
						|
    REQUIRE(doc.memoryUsage() == sizeofObject(1) + 2);
 | 
						|
    REQUIRE(doc[0] == "?");
 | 
						|
  }
 | 
						|
}
 |