This is the documentation for Enlighten.
Allocating working memory
Overview
This example demonstrates how to solve for the input lighting and output irradiance. Four additional temporary working memory buffers are required for the per-frame update loop:
- One for the input lighting
- One for the irradiance
- One to receive the final output (persistent if using temporal solves)
- One to receive the intermediate bounce output (persistent if using temporal solves)
The intermediate bounce output is generated as a secondary output of the irradiance solver, and is used as the input to the EndInputLighting
pass.
When using temporal solves, the bounce output also accumulates lighting difference values; therefore it is necessary to use a unique persistent buffer for each system. The same applies to the final output textures since the temporal solver may only update this partially (or not at all) depending on the lighting changes.
Setting up a memory allocator
You must set a memory allocator before you can create or use any Enlighten object, including the IPrecompute
object.
This is a change from the behaviour of previous Enlighten releases.
The file GeoBase/GeoMemory.h
contains a virtual class MemoryAllocator
and the associated function SetMemoryAllocator()
.
Enlighten ships with a default memory allocator, GeoMemoryDefault
, which is implemented using the standard C library memory functions. You can either use this allocator, or instead implement a custom allocator to tie into your own memory management or to track memory usage.
Example using Enlighten's default memory allocator
Use this code in your main function, to ensure that the memory allocator exists for the full duration of your program:
Geo::GeoMemoryDefault alloc; Geo::SetMemoryAllocator(&alloc);
Using a different memory allocator
To use a different allocator, derive from the MemoryAllocator
class and call SetMemoryAllocator()
as above.
The Enlighten runtime does not use the realloc
function, so you can provide a dummy implementation for runtime use. However, if you are using the low-level precompute API then a proper implementation of realloc
is required.
Ensure correct alignment if you change the actual allocation/de-allocation; otherwise errors may occur.
Working buffer for the input lighting
The input lighting task requires a small amount of scratchspace working memory. It depends on the list of lights passed to the task to work out the size of the required buffer:
// reserve workspace memory Geo::u32 workspaceSizeRequired = Enlighten::CalcRequiredScratchSpaceMemory(lights, numLights); void * scratchMemory = GEO_ALIGNED_MALLOC(workspaceSizeRequired, 16);
When solving input lighting for many systems, simply allocate a single buffer large enough for the largest task and then reuse it.
Working buffer for the solver
The solve functions also require some scratchspace working memory. The size depends on a given system's budget and the number of dependencies. Use the RadSystemCore
structure to query Enlighten and thus get a working buffer of exactly the right size:
// reserve workspace memory Geo::u32 irradianceMemoryRequired = Enlighten::CalcRequiredIrradianceTaskWorkspaceMemory(&myRadSystemCore); void* irradianceWorkspace = GEO_ALIGNED_MALLOC(irradianceMemoryRequired, 128);
Output texture buffers
Allocate a buffer to receive the radiosity output:
// Create radiosity output texture. Geo::s32 outputTextureSize = radCore->m_MetaData.m_OutputWidth * radCore->m_MetaData.m_OutputHeight; Geo::s32 outputTextureMemorySize = outputTextureSize * 8; // fp16 output = 8 bytes per pixel void* irradianceOutput = GEO_ALIGNED_MALLOC(outputTextureMemorySize, 16);
Optionally allocate buffer(s) for directional irradiance output:
// Create directional output texture(s). Geo::s32 outputTextureSize = radCore->m_MetaData.m_OutputWidth * radCore->m_MetaData.m_OutputHeight; Geo::s32 outputTextureMemorySize = outputTextureSize * 4; // 8 bit per channel output = 4 bytes per pixel // A single output texture for luminance-based directional irradiance void* directionalOutput = GEO_ALIGNED_MALLOC(outputTextureMemorySize, 16); // or alternatively, three output textures for colour-separated directional irradiance void* directionalOutputR = GEO_ALIGNED_MALLOC(outputTextureMemorySize, 16); void* directionalOutputG = GEO_ALIGNED_MALLOC(outputTextureMemorySize, 16); void* directionalOutputB = GEO_ALIGNED_MALLOC(outputTextureMemorySize, 16);
The code sample above is simplified to assume a linear buffer with no extra padding per row (that is, the stride is set to the width). In practice, although the output texture has a width which is a multiple of 4, its data stride may be different to the actual width, depending on the graphics driver. In this case, it is more convenient to allocate the output buffer to be the same data size of the texture, and pass the appropriate stride to the Enlighten solver.
Persistent data buffer
Use the RadSystemCore
to ask for the persistent data requirements:
Geo::s32 bounceOutputSize = Enlighten::CalcRequiredPersistentDataSize(&myRadSystemCore); void* bounceOutput = GEO_ALIGNED_MALLOC(bounceOutputSize, 16);