UnityGame/Library/PackageCache/com.unity.collections/Documentation~/allocator-rewindable.md
2024-10-27 10:53:47 +03:00

5.6 KiB

Rewindable allocator overview

A rewindable allocator is a custom allocator that works in a similar way to a linear allocator. It's fast and thread safe. A rewindable allocator pre-allocates blocks of memory in advance.

When you request memory from a rewindable allocator, it selects a range of memory from its pre-allocated block and assigns it to use. The minimum alignment of rewindable allocations is 64 bytes. After it uses all the existing blocks of memory, the rewindable allocator allocates another block of memory.

It doubles the size of the new block until it reaches a maximum block size. When it reaches this point, the rewindable allocator adds the maximum block size to its previous block size to increase its block size linearly.

One advantage of rewindable allocator is that you don't need to free individual allocations. As its name implies, a rewindable allocator can rewind and free all your allocations at one point.

When you rewind an allocator, the allocator keeps the memory blocks that it used before to improve performance and disposes the rest of the blocks. When you request to free or dispose a memory allocation from a rewindable allocator, it's a no-op unless you set the enable block free flag of the rewindable allocator. When you set the flag to enable block free, the rewindable allocator rewinds a memory block when it frees the last allocation from the block.

Declare and create a rewindable allocator

To create a rewindable allocator, you must do the following:

  • Allocate memory to hold the rewindable allocator
  • Add an entry in the global allocator table to register the allocator
  • Pre-allocate the allocator's first memory block to initialize it.

You can use the wrapper AllocatorHelper to create a rewindable allocator.

The following example declares and creates a rewindable allocator:

// Example user structure
internal struct ExampleStruct
{
    // Use AllocatorHelper to help creating a rewindable alloctor
    AllocatorHelper<RewindableAllocator> rwdAllocatorHelper;

    // Rewindable allocator property for accessibility
    public ref RewindableAllocator RwdAllocator => ref rwdAllocatorHelper.Allocator;

    // Create the rewindable allocator
    void CreateRewindableAllocator(AllocatorManager.AllocatorHandle backgroundAllocator, int initialBlockSize, bool enableBlockFree = false)
    {
        // Allocate the rewindable allocator from backgroundAllocator and register the allocator
        rwdAllocatorHelper = new AllocatorHelper<RewindableAllocator>(backgroundAllocator);

        // Allocate the first memory block with initialBlockSize in bytes, and indicate whether
        // to enable the rewindable allocator with individual block free through enableBlockFree
        RwdAllocator.Initialize(initialBlockSize, enableBlockFree);
    }
}

Use a rewindable allocator to allocate memory

For Native- collection types, allocation from a rewindable allocator is similar to a classic allocator, except you must use CollectionHelper.CreateNativeArray to create a NativeArray from a rewindable allocator. When you use a rewindable allocator to create a Native- collection type, its safety handle is added to the list of child safety handles of the rewindable allocator.

For Unsafe- collection types, you must use AllocatorManager.Allocate to allocate memory from a rewindable allocator.

You don't need to dispose individual allocations. When all allocations aren't needed anymore, call the Rewind method of a rewindable allocator to free all its allocations. When you rewind the rewindable allocator, it invalidates and unregisters its child allocators, and invalidates all its child safety handles. For Native- collection types, the disposal safety checks throw an exception if the rewindable allocator has rewound.

This example method UseRewindableAllocator shows how to use a rewindable allocator to create and allocate native containers:

[!code-csUse rewindable allocator to allocate memory]

Free all allocated memory of a rewindable allocator

When you Rewind the rewindable allocator, it performs the following operations:

  • Invalidates and unregisters all the allocator handle's child allocators
  • Invalidates all its child safety handles

The example method FreeRewindableAllocator shows how to Free all allocations from the rewindable allocator, with Rewind.

[!code-csFree all allocations from the rewindable allocator]

Dispose a rewindable allocator

To dispose a rewindable allocator, you must do the following:

  • Dispose all the memory blocks of the rewindable allocator from Allocator.Persistant.
  • Unregister the allocator
  • Dispose the memory used to store the allocator

The following example adds a method DisposeRewindableAllocatorthat disposes a rewindable allocator using Dispose:

[!code-csDispose a rewindable allocator]

Full example of a rewindable allocator

The following is a full example of how to use a rewindable allocator:

[!code-csFull example of a rewindable allocator]