From e4d8dc02e87614854f82f2d960c1e38a10511519 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Art=C3=B6m=20Bakri=20Al-Sarmini?= <3sz3tt+git@gmail.com> Date: Thu, 2 Apr 2020 15:20:25 +0300 Subject: [PATCH] Fixes #1971 (memory leak in basic_json::push_back) --- include/nlohmann/json.hpp | 4 +-- single_include/nlohmann/json.hpp | 4 +-- test/src/unit-allocator.cpp | 48 ++++++++++++++++++++++++++++++++ 3 files changed, 50 insertions(+), 6 deletions(-) diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index b72410d7..da3d57b1 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -4878,9 +4878,7 @@ class basic_json // add element to array (move semantics) m_value.array->push_back(std::move(val)); - // invalidate object: mark it null so we do not call the destructor - // cppcheck-suppress accessMoved - val.m_type = value_t::null; + // if val is moved from, basic_json move constructor marks it null so we do not call the destructor } /*! diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 77bd1739..4d0b84f4 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -19421,9 +19421,7 @@ class basic_json // add element to array (move semantics) m_value.array->push_back(std::move(val)); - // invalidate object: mark it null so we do not call the destructor - // cppcheck-suppress accessMoved - val.m_type = value_t::null; + // if val is moved from, basic_json move constructor marks it null so we do not call the destructor } /*! diff --git a/test/src/unit-allocator.cpp b/test/src/unit-allocator.cpp index 3518c4ae..c751d3c5 100644 --- a/test/src/unit-allocator.cpp +++ b/test/src/unit-allocator.cpp @@ -234,3 +234,51 @@ TEST_CASE("controlled bad_alloc") } } } + +namespace +{ +template +struct allocator_no_forward +{ + typedef std::remove_const_t value_type; + template + struct rebind + { + typedef allocator_no_forward other; + }; + + T* allocate(size_t sz) + { + return static_cast(malloc(sz * sizeof(T))); + } + + void deallocate(T* p, size_t) + { + free(p); + } + + void construct(T* p, const T& arg) + { + ::new (static_cast(p)) T(arg); + } +}; +} + +TEST_CASE("bad my_allocator::construct") +{ + SECTION("my_allocator::construct doesn't forward") + { + using bad_alloc_json = nlohmann::basic_json; + + bad_alloc_json json; + json["test"] = bad_alloc_json::array_t(); + json["test"].push_back("should not leak"); + } +}