Skip to content

Latest commit

 

History

History
279 lines (189 loc) · 21.3 KB

README.md

File metadata and controls

279 lines (189 loc) · 21.3 KB

Memwa

Table of Contents

 Introduction
    Terminology
 Benefits of Memwa
    Time
    Memory Use
    Alignment-Aware
    Resize In Place
    C++17 Compatible
    Reliance on Standard Functions
 Using Memwa
    Build Instructions
    Alignment
    Exceptions
    Recommendations
    With STL Containers
    Examples
 Allocators
    LinearAllocator
    StackAllocator
    PoolAllocator
    TinyObjectAllocator
 Testing Memwa
    Functionality
    Timing Tests
    Memory Tests
    Multi-threaded
 Design of Memwa
    Levels of Responsibility
        Block
        BlockInfo
        Single-Threaded Allocator
        Multi-Threaded Allocator
        AllocatorManager
        Supporting Classes
    Design Considerations
        Portability
        Time Performance
        Space Performance
        STL Container Compatiblility
        Embedded Software
 Writing Memwa Based Allocators

Introduction

Memwa is the Haitian Creole word for Memory.

This is a collection of memory allocators for C++.

Terminology

These terms are used within the documentation and code for Memwa.

  • Chunk An individual piece of memory provided by an Allocate function to consumers of Memwa.
  • Place The address of a chunk.
  • Block A large area of pre-allocated memory which is subdivided into many chunks.

Benefits of Memwa

Time

Memory Use

Alignment-Aware

Memwa is an alignment-aware allocator. When code requests a chunk of memory that is aligned on 1, 2, 4, 8, 16, or 32 byte boundaries, a Memwa allocator can fill that request. Although the C++ Standard says allocator "behavior is undefined if this is not a valid alignment value", Memwa will simply throw std::invalid_argument if the alignment is not correct.

Memwa may allocate extra bytes in each block so it can guarantee that each chunk is aligned on the appropriate byte boundary. For alignment sizes greater than 8, Memwa will add the alignment size to the block size. These extra bytes might be wasted depending on how the allocator algorithm uses the block size and alignment, or the extra bytes may be available for allocation into chunks. You can guarantee zero wasted bytes by never requesting alignment greater than 8 bytes. If you need alignments of 16, 32, or more, Memwa will provide that.

The alignment size should always be smaller than the block size.

Resize In Place

Sometimes you need to expand a chunk of memory, such as when you push an object onto the back of a std::vector, or add an element to a std::deque. This may require allocating a bigger chunk of memory, copying/moving all the elements to the new chunk, destructing all the elements in the old chunk, and then finally releasing the old chunk. A more efficient container would ask its allocator to simply expand the existing chunk, and then add the new elements into the annexed region of the new chunk. Unfortunately, STL containers don't work this way (yet) and STL allocators were not designed to support that need.

Some C++ containers now have a shrink_to_fit function that reduces the storage space to remove unused capacity. Allocators that support resize-in-place should allow chunks to shrink. The allocator will release the unused bytes at the end of the chunk.

The realloc function in the C Standard Library allows code to request a different chunk size. It will find a chunk that is big enough for the new size, allocate it, and then copy the data over to the new chunk. That can work for plain old data types, but merely copying the bytes of C++ objects is not acceptable for any objects whose constructors and destructors manage resources.

C++ needs an allocator that can resize in place so STL containers can expand or shrink their storage without copying or moving. Some Memwa allocators support resize-in-place. If the new size is smaller, the chunk is shrunk to that size. If it is bigger, the allocator determines if it can expand the chunk into unused memory. If it can expand, it does; otherwise it returns a value indicating that resize failed. When the C++ STL allocator has a function that supports resizing, Memwa will be ready to support it.

C++17 compatible

The 2017 version of C++ introduced several features related to memory allocation.

  • The std::align_val_t parameter. Several recently added new and delete operators use the std::align_val_t parameter for alignment-aware allocators.
  • The allocator trait for is_always_equal. Memwa's allocator adapter class provides the is_always_equal trait so compilers don't assume all Memwa allocators are equal.

Reliance on Standard Functions

Memwa only uses functions and types provided by the C++ Standard. The release version does not use any third party components so you don't have to link any other libraries into your software when you choose Memwa.

It makes as extensive use of std functions and types as possible.

Memwa uses functions in the std namespace where such functions provide the functionality Memwa needs. (e.g. - It uses std::align to align chunks on the appropriate byte boundaries instead of having its own function to do the same behavior.)

Using Memwa

Build Instructions

Alignment

For now, Memwa supports alignment on a power-of-two boundary up to 32. (e.g. - 1, 2, 4, 8, 16, or 32). This may change in the future. Some allocators support alignments as small as 1 or 2 bytes, while others support a minimum of 4 bytes.

Exceptions

Memwa will throw exceptions under these conditions.

std::bad_alloc if it is unable to allocate memory.
Before any Memwa allocator throws bad_alloc, it will attempt to free unused blocks of memory maintained by any other Memwa allocator.

std::invalid_argument if any of these conditions are met.

  • The alignment parameter to a Release or Allocate function is greater than the allocator's initial alignment.
  • The requested size is greater than what the allocator can provide.
  • The requested size does not match the size specified for a PoolAllocator or TinyObjectAllocator.

With STL Containers

Memwa provides an adapter template class so programmers can use Memwa allocators with STL containers. Some allocators work well with some containers, while others work better with different containers.

Recommendations

  • Configure the allocators to pre-allocate blocks whose sizes are the same as the CPU caches.
    Sometimes you have lots of data that can all fit on one memory page. Several Intel chips have L1 caches of 32KB and L2 caches of 256KB. If all of the data fits in the cache, then the CPU will not have to load another memory page into its local cache when it churns through that data. Memwa can allocate blocks that are the same size as the CPU caches.

  • Configure the allocators to pre-allocate blocks whose sizes are smaller than the CPU caches.
    This recommendation is the exact opposite of the previous one, and that is because sometimes you have lots of data spread across many pages. Sometimes it is hard to put all those different chunks of data onto the same page, even though the different data chunks might be used together. You can still reduce the likelihood of a cache miss by configuring Memwa to pre-allocate chunks much smaller than a cache. For example, by choosing a block size of 4KB, the CPU can load 8 blocks into an CPU cache that is 32KB. Whether you make each page the same size as a cache or smaller than a cache depends on what gives your code the greatest performance. Unlike other memory allocation libraries, Memwa gives you a choice.

  • Use the hint parameter to increase likelihood of locality.
    The Allocate functions accept a hint parameter - which is defaulted to nullptr - so the next chunk of memory will (likely) be allocated within the same block as the previous chunk of memory. If the two chunks of memory are used together often, and those two chunks are on the same memory block, the CPU will not have to load two different pages into its local caches to use that data. If they are on different blocks, the CPU will consume many cycles to load the memory into caches.

  • Don't underalign.
    Underalignment is when alignment is too small. Underalignment wastes time.
    Underalignment is when a program wants to align an object on a boundary that is much bigger than the object size. If an object is 8 bytes, it is better to align it on 8 byte boundaries than to align it on 2 or 4 byte boundaries. Overalignment can lead to suboptimal behavior since the compiler must create additional instructions to fetch data that is partly in one 4-byte word and then get the rest of the data that is in another 4 byte word.

  • Don't overalign.
    Overalignment is when alignment is too large. Overalignment wastes space.
    Overalignment is when an object is aligned on a smaller byte boundary than it should be. For example, if an object is 4 bytes, it is a waste of space to align it on 8 byte boundaries. The first 4 bytes of a chunk would store the object, and the second 4 bytes are wasted. It's more space efficient to align those objects on 4 byte boundaries.

  • Create a new Memwa allocator for each purpose. Don't use an allocator for multiple purposes.
    For example, if you have an allocator for a std::set, then don't use the same allocator for a different container or a different component. If your code is multi-threaded, then the different components will race against each other to use the allocator.
    Each allocator has to maintain a collection of blocks. To deallocate a chunk, it does a binary search through the blocks to find which block owns that chunk. It is quicker for separate allocators to search a small number of blocks, than for a single allocator to search a large number of blocks.

Examples

Allocators

LinearAllocator

The purpose of LinearAllocator is to allocate chunks of memory very quickly that never need to be released. LinearAllocator will pre-allocate blocks of memory and then suballocate chunks from within those blocks as requested. It maintains no information about the locations or sizes of those chunks so it can't release them.

Uses:

  • For chunks of any size from 1 byte to the size of a block.
  • For alignments of any size from 1 byte to 32 bytes.
  • Chunks of memory allocated at program start-up and never released.
  • Static const objects that are never destructed.

Limitations:

  • Does not allow releasing ever.
  • Does not allow resizing ever.

StackAllocator

StackAllocator will pre-allocate 1 or more large blocks and then suballocate chunks from within that block. It uses the 4 bytes before each chunk to store the size of the previous chunk. By storing the size, an allocator can release chunks at the end of a stack. It can't release chunks before the end. It allows resizing only on chunks at the top of each block.

Uses:

  • For alignments of any size from 4 bytes to 32 bytes. This allocator is not intended for alignment on 1 or 2 byte boundaries.
  • For objects of any size from 4 bytes to the size of a block. This allocator is not intended for chunks smaller than 4 bytes.
  • For memory that are released in the opposite order in which they were allocated. (first-in-last-out)
  • Chunks of memory allocated at program start-up and never released.
  • Static const objects that are never destructed.
  • Suitable for stack containers.

Limitations:

  • You can use it for sizes smaller than a pointer, but this is not recommended.
  • Best used for objects that are released in reverse order from how they are allocated.
  • Not suitable for node-based STL containers such as std::list, std::map, or std::set.
  • Can handle resizing, but only for the most recently allocated chunk. (Or for chunks that happen to be at the top of each block.)

PoolAllocator

The PoolAllocator will pre-allocate large blocks of memory and then subdivide each block into equal size chunks. The allocator then places a pointer inside each free chunk to point to the next free chunk; so all the free chunks form a singly linked list. Allocating a chunk merely requires removing it from the head of the linked list, and releasing it requires adding it back to the head of the list. Both of those are constant time actions.

Uses:

  • For alignments of any size from 4 bytes to 32 bytes.
  • For node-based STL containers that always allocate objects of the same size. (e.g. - std::list, std::set, or std::map)
  • For code that must allocate from hundreds to millions of objects that are all the same size.
  • For memory that could be allocated and released in any order.
  • Works well for chunks from 4 bytes up to block size.

Limitations:

  • Does not allow resizing ever.
  • Not intended for alignment on 1 or 2 byte boundaries.
  • Not recommended for chunks smaller than a pointer.
  • Not as space-efficient as TinyObjectAllocator for small objects.

TinyObjectAllocator

TinyObjectAllocator is very similar to PoolAllocator in behavior, but had slightly different features and limitations. It is a type of pool allocator specialized for small objects and small alignments. This allocator provides a time and space efficient way to allocate many small chunks of memory. Unlike the other allocators which use pointers and size fields to keep track of allocated memory, TinyObjectAllocator uses stealth indexes within each unallocated block so the indexes form a singly linked list. Allocating a chunk merely requires removing it from the head of a linked list of stealth indexes, and releasing it requires adding it back to the head of the list. Both of those are constant time actions.

This allocator was based on a rewrite of the SmallObjectAllocator in the Loki project. This does not suffer some of the limitations of Loki's SmallObjectAllocator.

  • The blocks in Loki were not sorted, so searching for a chunk to release within an allocator required a linear search every time. This allocator keeps its blocks sorted by address, so searching is always a O(log N) operation.
  • Loki favored making each allocator into a singleton, which meant all code that used the allocator would have to wait on other code in multi-threaded programs just to do a simple allocation or release. Memwa allows programs to make different allocators for different uses, so that code which uses one allocator would not have to wait on code using another.
  • By using the allocator as a singleton, the allocator would maintain blocks for many different parts of the code, so it would have a large number of blocks to search through when releasing a chunk. Since Memwa encourages each part of the code to create its own allocator, each of those allocators only needs to search a smaller number of blocks when releasing a chunk.

Uses:

  • For hundreds or thousands of objects that are all the same size.
  • For objects whose alignments can be on 1 or 2 byte boundaries. (If the object should be aligned on a boundary of 4 or more, consider PoolAllocator instead.)
  • For objects whose sizes are smaller than a pointer. (Pointers are usually 4 or 8 bytes.)
  • Works well for chunks from 1 byte up to 128 bytes.
  • For memory that could be allocated and released in any order.
  • For node-based STL containers that always allocate objects of the same size. (e.g. - std::list, std::set, or std::map)

Limitations:

  • Does not allow resizing ever.
  • Not as time efficient as PoolAllocator when allocating and releasing large amounts of chunks often.

Testing Memwa

Functionality

The purpose of functionality tests is to demonstrate that each allocator provides an expected output for particular input values.

Negative Tests

These tests provide invalid input parameters (e.g. - wrong size or wrong alignment) to see if the allocator will throw an exception.

Positive Tests

These tests show the allocators return correct values for valid input.

Timing Tests

Memory Tests

Multi-threaded

Design of Memwa

Levels of Responsibility

Block

The Block is the lowest level of Memwa. Each instance of a Block class (e.g. - PoolBlock, LinearBlock, etc...) maintains a single block of memory. This class maintains internal pointers and indexes used for the block. All of its actions are constant-time and often trivial so that the block does not impose any runtime costs on higher classes in Memwa. Each block object is designed for trivial copying so they can be stored in containers.

BlockInfo

BlockInfo maintains a container of blocks and common information (e.g. - block-size, alignment, and object-size) about those blocks. It provides functions to search blocks for an address, allocate a chunk from a block, and to release a chunk from a block. Its internal duties include creating new blocks, removing empty blocks, and keeping blccks sorted. Each allocator object has a single BlockInfo object to maintain its blocks.

Single-Threaded Allocator

The single-threaded allocators implement Memwa's allocator interface class. These classes will check if the input parameters are valid, and will then call the appropriate functions in their BlockInfo data members.

Multi-Threaded Allocator

The multi-threaded allocators are mere subclasses of the single-threaded allocators. Each function in the multi-threaded allocators will lock a mutex and then call the appropriate function in the base class. The lock uses RAII so that any exception will automatically unlock the mutex.

AllocatorManager

The AllocatorManager is at the top of the Memwa project. It provides a factory function to create any type of Memwa allocator on demand, to destroy a Memwa allocator, and to trim unused memory from Memwa allocators.

Supporting Classes

Design Considerations

Portability

Time Performance

The lowest levels of each Memwa allocator require only O(1) operations to allocate and release a chunk. A Block should only need to adjust one or two pointers and/or indexes for each allocation and release. Most functions at the next level up required O(1) operations for almost all allocations and releases, use O(log N) operations occasionally, and use O(N) operations sparingly.

Space Performance

STL Container Compatiblility

Embedded Software

Writing Memwa Based Allocators