Add a tag to the string memory pool in fmt to discriminate between pool vs heap allocated buffers. Heap allocated buffers are used as a fallback mechanism when the buffer runs out of space.

master
faluco 4 years ago committed by faluco
parent 0465f6badd
commit a2f6e13138

@ -1277,7 +1277,7 @@ template <typename T> const T& unwrap(const std::reference_wrapper<T>& v) {
class dynamic_arg_list { class dynamic_arg_list {
public: public:
static constexpr std::size_t max_pool_string_size = 140; static constexpr std::size_t max_pool_string_size = 256;
private: private:
// Workaround for clang's -Wweak-vtables. Unlike for regular classes, for // Workaround for clang's -Wweak-vtables. Unlike for regular classes, for

@ -26,9 +26,11 @@ int format_float(char* buf, std::size_t size, const char* format, int precision,
} }
#define NODE_POOL_SIZE (10000u) #define NODE_POOL_SIZE (10000u)
static constexpr uint8_t memory_heap_tag = 0xAA;
class dyn_node_pool class dyn_node_pool
{ {
using type = std::array<uint8_t, dynamic_arg_list::max_pool_node_size>; /// The extra byte is used to store the memory tag at position 0 in the array.
using type = std::array<uint8_t, dynamic_arg_list::max_pool_node_size + 1>;
public: public:
dyn_node_pool() { dyn_node_pool() {
@ -44,13 +46,18 @@ public:
std::lock_guard<std::mutex> lock(m); std::lock_guard<std::mutex> lock(m);
if (free_list.empty()) { if (free_list.empty()) {
return nullptr; // Tag that this allocation was performed by the heap.
auto *p = new type;
(*p)[0] = memory_heap_tag;
return p->data() + 1;
} }
auto* p = free_list.back(); auto* p = free_list.back();
free_list.pop_back(); free_list.pop_back();
return p; // Tag that this allocation was performed by the pool.
p[0] = 0;
return p + 1;
} }
void dealloc(void* p) { void dealloc(void* p) {
@ -59,7 +66,14 @@ public:
} }
std::lock_guard<std::mutex> lock(m); std::lock_guard<std::mutex> lock(m);
free_list.push_back(reinterpret_cast<uint8_t *>(p)); uint8_t* base_ptr = reinterpret_cast<uint8_t *>(p) - 1;
if (*base_ptr == memory_heap_tag) {
// This pointer was allocated using the heap.
delete reinterpret_cast<type *>(base_ptr);
return;
}
free_list.push_back(base_ptr);
} }
private: private:

Loading…
Cancel
Save