...one of the most highly
regarded and expertly designed C++ library projects in the
world.
— Herb Sutter and Andrei
Alexandrescu, C++
Coding Standards
Variable-length containers in this library all use dynamically allocated
memory to store their contents. Callers can gain control over the strategy
used for allocation by specifying a storage_ptr
in select constructors
and function parameter lists. A storage_ptr
has these properties:
memory_resource
.
memory_resource*
or polymorphic_allocator
do not
acquire ownership; the caller is responsible for ensuring that the lifetime
of the resource extends until it is no longer referenced.
make_shared_resource
acquires
shared ownership of the memory resource; the lifetime of the resource
is extended until all copies of the storage pointer are destroyed.
is_deallocate_trivial
before
type-erasing the resource, allowing the value to be queried at run-time.
This lists all of the allocation-related types and functions available when using the library:
Table 1.4. Functions and Types
Name |
Description |
---|---|
Returns a pointer to a memory resource instance which always throws an exception upon allocation. This is used to to achieve the invariant that no parsing or container operation will dynamically allocate memory. |
|
A customization point allowing a memory resource type to indicate that calls to deallocate are trivial. |
|
A function returning a smart pointer with shared ownership of a newly allocated memory resource. |
|
The abstract base class representing an allocator. |
|
A memory resource which allocates large blocks of memory and has a trivial deallocate function. Allocated memory is not freed until the resource is destroyed, making it fast for parsing but not suited for performing modifications. |
|
An Allocator
which uses a reference to a |
|
A memory resource that uses a single caller provided buffer. No dynamic allocations are used. This is fast for parsing but not suited for performing modifications. |
|
A smart pointer through which a |
The default memory resource uses the global operator new
and operator delete
to allocate memory. This resource is not reference counted and has a non-trivial
deallocate function. All default-constructed storage_ptr
objects reference the
same memory resource:
storage_ptr sp1; storage_ptr sp2; assert( sp1.get() != nullptr ); // always points to a valid resource assert( sp1.get() == sp2.get() ); // both point to the default resource assert( *sp1.get() == *sp2.get() ); // the default resource compares equal
Default-constructed library containers use the default memory resource:
array arr; // default construction object obj; string str; value jv; assert( jv.storage().get() == storage_ptr().get() ); // uses the default memory resource assert( jv.storage().get() == arr.storage().get() ); // both point to the default resource assert( *arr.storage() == *obj.storage() ); // containers use equivalent resources
The default memory resource is well suited for general usage. It offers reasonable performance for parsing, and conservative memory usage for modification of the contents of containers.
Note | |
---|---|
This memory resource is not guaranteed to be the same as the result of
|
Consider the pattern of memory allocation during parsing: when an array, object, or string is encountered the parser accumulates elements in its temporary storage area. When all of the elements are known, a single memory allocation is requested from the resource when constructing the value. Thus, parsing only allocates and constructs containers at their final size. Memory is not reallocated; that is, a memory buffer never needs to grow by allocating a new larger buffer and deallocating the previous buffer.
The monotonic_resource
optimizes this
memory allocation pattern by allocating increasingly large blocks of global
memory internally and parceling those blocks out in smaller pieces to fulfill
allocation requests. It has a trivial deallocate function. The monotonic
resource does not actually deallocate memory until the resource is destroyed.
Thus, it is ideally suited for the use-case where JSON is parsed, and the
resulting value is then inspected but not modified.
The resource to use when constructing values may be specified in calls to
parse
as shown here:
monotonic_resource mr; value const jv = parse( "[1,2,3]", &mr );
Or, to parse into a value with shared ownership of the memory resource:
value parse_value( string_view s) { return parse( s, make_shared_resource< monotonic_resource >() ); }
A monotonic resource may be optionally constructed with an initial buffer to use first, before going to the heap. This allows the caller to use stack space and avoid dynamic allocations for most parsed JSON, falling back to dynamic allocation from the heap if the input JSON is larger than average, as shown here:
template< class Handler > void do_rpc( string_view s, Handler&& h ) { unsigned char buffer[ 8192 ]; // Small stack buffer to avoid most allocations during parse monotonic_resource mr( buffer ); // This resource will use our local buffer first value const jv = parse( s, &mr ); // Parse the input string into a value that uses our resource h( jv ); // Call the handler to perform the RPC command }
A static_resource
constructs from a caller-provided buffer, and satisfies all memory allocation
requests from the buffer. Once the buffer is exhausted, subsequent calls
to allocate throw the exception std::bad_alloc
.
The resource offers a simple invariant: dynamic heap allocations are never
performed.
To use the resource, construct it with a local buffer:
unsigned char buffer[ 8192 ]; static_resource mr( buffer ); // The resource will use our local buffer
The function get_null_resource
returns a global
instance of the null resource. This resource offers a simple invariant: all
calls to allocate will throw the exception std::bad_alloc
.
An instance of the null resource can be used to make parsing guarantee that
allocations from the heap are never made. This is explored in more detail
in a later section.
The containers array
, object
, and value
all propagate the memory resource
they were constructed with to child elements:
monotonic_resource mr; array arr( &mr ); // construct an array using our resource arr.emplace_back( "boost" ); // insert a string assert( *arr[0].as_string().storage() == mr ); // the resource is propagated to the string
This propagation acts recursively, containers within containers will all have the resource propagated. Once a container is constructed, its memory resource can never be changed.
It is important to note that storage_ptr
supports both shared-ownership
and reference lifetime models. Construction from a memory resource pointer
does not transfer ownership:
{ monotonic_resource mr; array arr( &mr ); // construct an array using our resource assert( ! arr.storage().is_shared() ); // no shared ownership }
When using a memory resource in this fashion, including the case where a
storage pointer or container is constructed from a polymorphic_allocator
, the caller
must ensure that the lifetime of the resource is extended until it is no
longer referenced by any variables; otherwise, undefined behavior is possible.
Shared ownership is achieved using the function make_shared_resource
, which creates
a new, reference-counted memory resource using a dynamic memory allocation
and returns it as a storage_ptr
:
storage_ptr sp = make_shared_resource< monotonic_resource >(); string str( sp ); assert( sp.is_shared() ); // shared ownership assert( str.storage().is_shared() ); // shared ownership
When a storage pointer is constructed this way, the lifetime of the referenced memory resource is extended until all variables which reference it are destroyed.
To implement custom memory allocation strategies, derive your class from
memory_resource
and implement the functions do_allocate
,
do_deallocate
, and do_is_equal
as seen in this example below,
which logs each operation it performs to the console:
class logging_resource : public boost::container::pmr::memory_resource { private: void* do_allocate( std::size_t bytes, std::size_t align ) override { std::cout << "Allocating " << bytes << " bytes with alignment " << align << '\n'; return ::operator new( bytes ); } void do_deallocate( void* ptr, std::size_t bytes, std::size_t align ) override { std::cout << "Deallocating " << bytes << " bytes with alignment " << align << " @ address " << ptr << '\n'; return ::operator delete( ptr ); } bool do_is_equal( memory_resource const& other ) const noexcept override { // since the global allocation and deallocation functions are used, // any instance of a logging_resource can deallocate memory allocated // by another instance of a logging_resource return dynamic_cast< logging_resource const* >( &other ) != nullptr; } };