From 27d6db8d311b6e244bcfb84c2488441a128b573a Mon Sep 17 00:00:00 2001 From: Roman Gershman Date: Fri, 10 May 2024 22:31:52 +0300 Subject: [PATCH 1/2] fix: avoid over-allocating memory for 8 byte alignment or less Signed-off-by: Roman Gershman --- include/jsoncons/detail/heap_string.hpp | 33 ++++++++++++++++++------- 1 file changed, 24 insertions(+), 9 deletions(-) diff --git a/include/jsoncons/detail/heap_string.hpp b/include/jsoncons/detail/heap_string.hpp index fe3501b61..54ed1da47 100644 --- a/include/jsoncons/detail/heap_string.hpp +++ b/include/jsoncons/detail/heap_string.hpp @@ -60,6 +60,7 @@ namespace detail { pointer p_; std::size_t length_; uint8_t offset_; + uint8_t align_pad_; ~heap_string() noexcept = default; @@ -136,16 +137,29 @@ namespace detail { std::size_t len = aligned_size(length*sizeof(char_type)); std::size_t align = alignof(storage_type); - std::size_t mem_len = (align-1)+len; - + char* q = nullptr; + char* storage = nullptr; byte_allocator_type byte_alloc(alloc); - byte_pointer ptr = byte_alloc.allocate(mem_len); - - char* q = extension_traits::to_plain_pointer(ptr); - - char* storage = align_up(q, align); + uint8_t align_pad = 0; + + if (align <= 8) { + byte_pointer ptr = byte_alloc.allocate(len); + if (reinterpret_cast(ptr) % align == 0) { + storage = extension_traits::to_plain_pointer(ptr); + q = storage; + align_pad = 0; + } else { + byte_alloc.deallocate(ptr, len); + } + } - JSONCONS_ASSERT(storage >= q); + if (storage == nullptr) { + align_pad = (align-1); + byte_pointer ptr = byte_alloc.allocate(align_pad+len); + q = extension_traits::to_plain_pointer(ptr); + storage = align_up(q, align); + JSONCONS_ASSERT(storage >= q); + } heap_string_type* ps = new(storage)heap_string_type(extra, byte_alloc); @@ -157,6 +171,7 @@ namespace detail { ps->p_ = std::pointer_traits::pointer_to(*p); ps->length_ = length; ps->offset_ = (uint8_t)(storage - q); + ps->align_pad_ = align_pad; return std::pointer_traits::pointer_to(*ps); } @@ -170,7 +185,7 @@ namespace detail { char* p = q - ptr->offset_; - std::size_t mem_size = (alignof(storage_type)-1)+ aligned_size(ptr->length_*sizeof(char_type)); + std::size_t mem_size = ptr->align_pad_ + aligned_size(ptr->length_*sizeof(char_type)); byte_allocator_type byte_alloc(ptr->get_allocator()); byte_alloc.deallocate(p,mem_size + ptr->offset_); } From 07e33158e20eee98ced572ca09fd94cbccf795b0 Mon Sep 17 00:00:00 2001 From: Roman Gershman Date: Wed, 15 May 2024 09:04:00 +0300 Subject: [PATCH 2/2] chore: address comments Signed-off-by: Roman Gershman --- include/jsoncons/detail/heap_string.hpp | 26 ++++++++++++------------- 1 file changed, 13 insertions(+), 13 deletions(-) diff --git a/include/jsoncons/detail/heap_string.hpp b/include/jsoncons/detail/heap_string.hpp index 54ed1da47..7f516ea95 100644 --- a/include/jsoncons/detail/heap_string.hpp +++ b/include/jsoncons/detail/heap_string.hpp @@ -15,7 +15,7 @@ #include // std::allocator #include -namespace jsoncons { +namespace jsoncons { namespace detail { inline char* @@ -30,8 +30,8 @@ namespace detail { { Extra extra_; Allocator alloc_; - - Allocator& get_allocator() + + Allocator& get_allocator() { return alloc_; } @@ -53,7 +53,7 @@ namespace detail { struct heap_string : public heap_string_base { using char_type = CharT; - using allocator_type = typename std::allocator_traits::template rebind_alloc; + using allocator_type = typename std::allocator_traits::template rebind_alloc; using allocator_traits_type = std::allocator_traits; using pointer = typename allocator_traits_type::pointer; @@ -62,7 +62,7 @@ namespace detail { uint8_t offset_; uint8_t align_pad_; - ~heap_string() noexcept = default; + ~heap_string() noexcept = default; const char_type* c_str() const { return extension_traits::to_plain_pointer(p_); } const char_type* data() const { return extension_traits::to_plain_pointer(p_); } @@ -111,10 +111,10 @@ namespace detail { using heap_string_type = heap_string; private: - using byte_allocator_type = typename std::allocator_traits::template rebind_alloc; + using byte_allocator_type = typename std::allocator_traits::template rebind_alloc; using byte_pointer = typename std::allocator_traits::pointer; - using heap_string_allocator_type = typename std::allocator_traits::template rebind_alloc; + using heap_string_allocator_type = typename std::allocator_traits::template rebind_alloc; public: using pointer = typename std::allocator_traits::pointer; @@ -144,17 +144,17 @@ namespace detail { if (align <= 8) { byte_pointer ptr = byte_alloc.allocate(len); - if (reinterpret_cast(ptr) % align == 0) { - storage = extension_traits::to_plain_pointer(ptr); - q = storage; - align_pad = 0; + q = extension_traits::to_plain_pointer(ptr); + + if (reinterpret_cast(q) % align == 0) { + storage = q; } else { byte_alloc.deallocate(ptr, len); } } if (storage == nullptr) { - align_pad = (align-1); + align_pad = uint8_t(align-1); byte_pointer ptr = byte_alloc.allocate(align_pad+len); q = extension_traits::to_plain_pointer(ptr); storage = align_up(q, align); @@ -163,7 +163,7 @@ namespace detail { heap_string_type* ps = new(storage)heap_string_type(extra, byte_alloc); - auto psa = launder_cast(storage); + auto psa = launder_cast(storage); CharT* p = new(&psa->c)char_type[length + 1]; std::memcpy(p, s, length*sizeof(char_type));