I’m working with C# in a game engine and frequently need to pass around subsets of collections of objects to different synchronous functions. I need to avoid making heap allocations for each call, and have investigated a few solutions. One that seemed elegant conceptually was to stackalloc a buffer and store references to the objects in it, then pass that buffer as a Span to whatever function I needed to call.
However, I know the docs state that stackalloc can only be used with unmanaged types. I’ve read a few descriptions of why this is the case, including this SO answer. My limited understanding is that the garbage collector needs to track all of the potential references to objects for the purposes of moving and freeing them. But stackalloc creates a value type with no fields, just a raw blob of memory with no type info, so the GC inherently can’t know anything about it.
Through use of the Unsafe and MemoryMarshal classes, I’ve written a workaround that allocates and reinterprets a buffer on the stack:
IntPtr* ptr = stackalloc IntPtr[count];
ref T tRef = ref Unsafe.AsRef<T>(ptr); // T is some object
Span<T> buff = MemoryMarshal.CreateSpan(ref tRef, count);
... // Copy references into buff and pass it to some function
This works as expected so far. However, I haven’t used this solution extensively yet and don’t know if there will be any unexpected behavior over sustained GC runs.
I know the objects won’t be freed as they’re still referenced by the original heap allocated collections. But I want to ensure the objects won’t be moved by the GC while calling a function with these stack buffers. Is this possible?
Will the GC account for the references stored in this memory because it’s now being pointed to by a Span? If not, is it possible to avoid the referenced objects from being moved using the Span with the fixed statement and pinning them? If that won’t work, is there any other way to ensure they won’t be moved for the duration of the function call, or is this just not possible whatsoever?
6
I am explicitly attempting to avoid allocating a new heap array for
each call I make.
For your purposes, there seems to be a built-in solution – ArrayPool<T>
:
Using the ArrayPool class to rent and return buffers (using the
Rent and Return methods) can improve performance in situations where
arrays are created and destroyed frequently, resulting in significant
memory pressure on the garbage collector.
1