Skip to content

Latest commit

 

History

History
1569 lines (1246 loc) · 130 KB

WickedEngine-Documentation.md

File metadata and controls

1569 lines (1246 loc) · 130 KB

WickedEngine Documentation

This is the documentation for the C++ features of Wicked Engine

Contents

  1. Building and linking
  2. Basic runtime flow
  3. High Level Interface
    1. Application
    2. RenderPath
    3. RenderPath2D
    4. RenderPath3D
    5. RenderPath3D_Pathtracing
      1. Denoiser
    6. LoadingScreen
  4. System
    1. Entity-Component System
    2. Scene System
      1. NameComponent
      2. LayerComponent
      3. TransformComponent
      4. PreviousFrameTransformComponent
      5. HierarchyComponent
      6. MaterialComponent
      7. MeshComponent
      8. ImpostorComponent
      9. ObjectComponent
      10. RigidBodyPhysicsComponent
      11. SoftBodyPhysicsComponent
      12. ArmatureComponent
      13. LightComponent
      14. CameraComponent
      15. EnvironmentProbeComponent
      16. ForceFieldComponent
      17. DecalComponent
      18. AnimationDataComponent
      19. AnimationComponent
      20. WeatherComponent
      21. SoundComponent
      22. InverseKinematicsComponent
      23. SpringComponent
      24. ColliderComponent
      25. ScriptComponent
      26. Scene
    3. Job System
    4. Initializer
    5. Platform
    6. EventHandler
    7. Canvas
  5. Graphics
    1. Graphics API
      1. GraphicsDevice
        1. Debug device
        2. Creating resources
        3. Destroying resources
        4. Work submission
          1. Async compute
        5. Presenting to the screen
          1. HDR Display
        6. Resource binding
        7. Bindless resources
        8. Subresources
        9. Pipeline States and Shaders
        10. Render Passes
        11. GPU Barriers
        12. Textures
        13. GPU Buffers
        14. Updating GPU buffers
        15. GPU Queries
        16. RayTracingAccelerationStructure
        17. RayTracingPipelineState
        18. Variable Rate Shading
        19. Video decoding
      2. GraphicsDevice_DX11
      3. GraphicsDevice_DX12
      4. GraphicsDevice_Vulkan
    2. Renderer
      1. DrawScene
      2. Tessellation
      3. Occlusion Culling
      4. Shadow Maps
      5. UpdatePerFrameData
      6. UpdateRenderData
      7. Ray tracing (hardware accelerated)
      8. Ray tracing (compute)
      9. Path tracing
      10. Lightmap baking
      11. Scene BVH
      12. Decals
      13. Environment probes
      14. Post processing
      15. Instancing
      16. Stencil
      17. Loading Shaders
      18. Debug Draw
      19. Animation Skinning
      20. Custom Shaders
    3. Enums
    4. Image Renderer
    5. Font Renderer
    6. Emitted Particle System
    7. Hair Particle System
    8. Ocean
    9. Sprite
    10. SpriteFont
    11. GPUSortLib
    12. GPUBVH
  6. GUI
    1. GUI
    2. Widget
    3. Button
    4. Label
    5. TextInputField
    6. Slider
    7. CheckBox
    8. ComboBox
    9. Window
    10. ColorPicker
    11. TreeList
  7. Helpers
    1. Archive
    2. Color
    3. FadeManager
    4. Helper
    5. Primitive
      1. AABB
      2. Sphere
      3. Capsule
      4. Ray
      5. Frustum
      6. Hitbox2D
    6. wiMath
    7. wiRandom
    8. wiRectPacker
    9. wiResourceManager
    10. wiSpinLock
    11. wiArguments
    12. wiTimer
    13. wiVoxelGrid
    14. wiPathQuery
  8. Input
  9. Audio
    1. Sound
    2. SoundInstance
    3. SoundInstance3D
    4. SUBMIX_TYPE
    5. REVERB_PRESET
  10. Physics
    1. Rigid Body Physics
    2. Soft Body Physics
  11. Network
    1. Socket
    2. Connection
  12. Scripting
    1. Lua
    2. Lua_Globals
    3. Luna
  13. Tools
    1. Backlog
    2. Profiler
  14. Shaders
    1. Interop
    2. Shader Compiler

Building and linking

Wicked Engine is a static library that can be included and linked into a standard C++ application. This comes with some differences when you try to build it for different platforms, because those have different compilers and settings. The basics of the static library will apply for all:

  1. You must build WickedEngine itself
    • Windows: Use the WickedEngine.sln with Visual Studio and build the WickedEngine_Windows project
    • Linux: Use the following commands on linux to build the solution:

Ensure dependencies are installed on Linux:

sudo apt update
sudo apt install libsdl2-dev
sudo apt install build-essential

To build the engine on Linux, use cmake and then make:

mkdir build
cd build
cmake ..
make
  1. Set the $(SolutionDir)WickedEngine folder as "additional include directories" in your application's build system (for example, for me it resolves to: C:\PROJECTS\WickedEngine\WickedEngine)
  2. Set the $(SolutionDir)BUILD/$(Platform)/$(Configuration) as "additional library directories" in your application's build system (for example, for me it resolves to: C:\PROJECTS\WickedEngine\BUILD\x64\Debug in a Debug x64 configuration)
  3. Use #include "WickedEngine.h in your C++ application code
  4. Verify that you application compiles correctly and links to Wicked Engine.
  5. If succeeded, continue this guide with the Application initialization

If you have troubles, check out the Samples/Template projects which are setting up simple applications that use Wicked Engine and how they are configured.

Xbox, PlayStation: The WickedEngine.sln must be used with Visual Studio similarly to Windows, but with console extension files and additional instructions which are private now, but could be offered in the future.

Basic Runtime flow

The general flow of Wicked Engine's operation is the following:

  • main.cpp: your entry point with all the platform-specific functionality
    • wi::Application::SetWindow() is called once with operating system-specific window handle
    • wi::Application.Run() is called in an infinite loop, each call is one "frame"
      • wi::Application::Initialize() is called once if hasn't been called it before
        • wi::initializer::InitializeComponentsAsync() will be called to initialize all engine systems in the background
      • wi::Application::Update() is called every frame
        • wi::input::Update() is called to read input
        • lua script coroutines waiting for update() are resumed
        • wi::RenderPath::Update() is called every frame if any RenderPath is activated on the Application
          • wi::scene::Scene::Update() is called every frame if RenderPath is a RenderPath3D
      • wi::Application::Render() is called every frame
        • lua script coroutines waiting for render() are resumed
        • wi::RenderPath::Render() is called every frame if any RenderPath is activated on the Application
          • complex off-screen rendering logic is performed with multiple CPU threads (using mostly functions from the wi::renderer namespace)
      • wi::Application::Compose() is called every frame
        • wi::RenderPath::Compose() is called every frame if any RenderPath is activated on the Application
          • simple rendering logic is performed on the main thread directly onto the display (swap chain)
        • Application renders some info text on the very top like version, FPS, etc. if enabled (infoDisplay)
      • Submit all GPU work on the main thread (GraphicsDevice::SubmitCommandLists())

High Level Interface

The high level interface consists of classes that make up the general flow of Wicked Engine and allow for extending the engine with custom functionality. This is usually done by overriding the classes. The high level interface is not necessary to be used when coding in c++, but they provide a complete simple runtime logic that is maintained by Wicked Engine.

Application

[Header] [Cpp] This is the main runtime component that has the Run() function which is called every frame. It should be included in the application entry point while calling Run() in an infinite loop. The Application is designed to handle a simple game-like functionality, with switching to different game screens with fading and rendering to a single display.

To use the application, the user should at least set the operating system window to render to with the SetWindow(), providing the operating system-specific window handle to it. This will be the main window that the application will draw its contents to.

It's also recommended to call Initialize() of the application if you do anything with the engine features outside of RenderPath, for example in your main function. This will kick off engine initialization immediately, instead of the first call to Run(). Using engine features before application.Initialize() was called, can be undefined behaviour. If you only use engine features within application's Update() and other overridable functions, those are ensured to be running when it is valid to do.

An example of minimal application initialization:

Application app;
app.SetWindow(hWnd); // operating-system dependent window handle is given
while(true)
{
	app.Run(); // app.Initialize() will be called in here, don't use engine features before this.
}

The example above is sufficient to rely on doing things inside the startup.lua file, which - if exists - will be executed at the appropriate time.

An example of minimal application initialization with extended c++ class:

class Game : public Application
{
	virtual ~Game() = default; // in c++ it's recommended to always make class destructors that are used in inheritance virtual

	void Update(float deltatime) override // override Application's Update function
	{
		Application::Update(deltatime);
		// your per-application update logic can go here
	}
}
Game game;
game.SetWindow(hWnd);
while(true)
{
	game.Run();
}

An example of application initialization and some engine usage immediately:

Application app;
app.SetWindow(hWnd);
app.Initialize(); // before using engine feature LoadModel, application.Initialize() will ensure all engine system initializations are kicked off. Most things are safe to use while initalization is running in the background...
LoadModel("something.wiscene");
while(true)
{
	app.Run();
}

An example of initializing application and blocking until whole engine was initialized:

Application app;
app.SetWindow(hWnd);
app.Initialize(); // this returns immediately, but engine sub-system begin to be initialized in the background...
wi::initializer::WaitForInitializationsToFinish(); // block until all engine sub-systems are ready. If this is called before Run(), then initialization screen won't be shown.
while(true)
{
	app.Run();
}

The Application has many uses that the user is not necessarily interested in. The most important part is that it manages the RenderPaths. There can be one active RenderPath at a time, which will be updated and rendered to the screen every frame. However, because a RenderPath is a highly customizable class, there is no limitation what can be done within the RenderPath, for example supporting multiple concurrent RenderPaths if required. RenderPaths can be switched with a fade out screen easily. Loading Screen can be activated as an active Renderpath and it will load and switch to an other RenderPath if desired. A RenderPath can be simply activated with the Application::ActivatePath() function, which will also perform the fadeout when the fadeSeconds and fadeColor arguments are used.
The Application does the following every frame while it is running:

  1. FixedUpdate()
    Calls FixedUpdate for the active RenderPath and wakes up scripts that are waiting for fixedupdate(). The frequency off calls will be determined by Application::setTargetFrameRate(float framespersecond). By default (parameter = 60), FixedUpdate will be called 60 times per second.
  2. Update(float deltatime)
    Calls Update for the active RenderPath and wakes up scripts that are waiting for update()
  3. Render()
    Calls Render for the active RenderPath and wakes up scripts that are waiting for render()
  4. Compose() Calls Compose for the active RenderPath

RenderPath

[Header] This is an empty base class that can be activated with a Application. It calls its Start(), Update(), FixedUpdate(), Render(), Compose(), Stop(), etc. functions as needed. Override this to perform custom gameplay or rendering logic. The RenderPath inherits from wiCanvas, and the canvas properties will be set by the Application when the RenderPath was activated, and while the RenderPath is active.
The order in which the functions are executed every frame:

  1. PreUpdate()
    This will be called once per frame before any script that calls Update().
  2. FixedUpdate()
    This will be called in a manner that is deterministic, so logic will be running in the frequency that is specified with Application::setTargetFrameRate(float framespersecond)
  3. Update(float deltatime)
    This will be called once per frame, and the elapsed time in seconds since the last Update() is provided as parameter
  4. PostUpdate()
    This will be called once per frame after any script that calls Update().
  5. Render() const
    This will be called once per frame. It is const, so it shouldn't modify state. When running this, it is not defined which thread it is running on. Multiple threads and job system can be used within this. The main purpose is to record mass rendering commands in multiple threads and command lists. Command list can be safely retrieved at this point from the graphics device.
  6. Compose(CommandList cmd) const
    It is called once per frame. It is running on a single command list that it receives as a parameter. These rendering commands will directly record onto the last submitted command list in the frame. The render target is the back buffer at this point, so rendering will happen to the screen.

Apart from the functions that will be run every frame, the RenderPath has the following functions:

  1. Load()
    Intended for loading resources before the first time the RenderPath will be used. It will be called by LoadingScreen to load resources in the background.
  2. Start()
    Start will always be called when a RenderPath is activated by the Application
  3. Stop()
    Stop will be always called when the current RenderPath was the active one in Application, but an other one was activated.

RenderPath2D

[Header] [Cpp] Capable of handling 2D rendering to offscreen buffer in Render() function, or just the screen in Compose() function. It has some functionality to render wiSprite and wiSpriteFont onto rendering layers and stenciling with 3D rendered scene. It has a GUI that is automatically updated and rendered if any elements have been added to it.

RenderPath3D

[Header] [Cpp] Base class for implementing 3D rendering paths. It also supports everything that the Renderpath2D does.

The post process chain is also implemented here. This means that the order of the post processes and the resources that they use are defined here, but the individual post process rendering on a lower level is implemented in the wi::renderer as core engine features. Read more about post process implementation in the wi::renderer section.

The post process chain is implemented in RenderPath3D::RenderPostprocessChain() function and divided into multiple parts:

  • HDR
    These are using the HDR scene render targets, they happen before tone mapping. For example: Temporal AA, Motion blur, Depth of field
  • LDR
    These are running after tone mapping. For example: Color grading, FXAA, chromatic aberration. The last LDR post process result will be output to the back buffer in the Compose() function.
  • Other
    These are running in more specific locations, depending on the render path. For example: SSR, SSAO, cartoon outline

The HDR and LDR post process chain are using the "ping-ponging" technique, which means when the first post process consumes texture1 and produces texture2, then the following post process will consume texture2 and produce texture1, until all post processes are rendered.

RenderPath3D_PathTracing

[Header] [Cpp] Implements a compute shader based path tracing solution. In a static scene, the rendering will converge to ground truth. When something changes in the scene (something moves, ot material changes, etc...), the convergence will be restarted from the beginning. The raytracing is implemented in wi::renderer and multiple shaders. The ray tracing is available on any GPU that supports compute shaders.

Denoiser

To enable denoising for path traced images, you can use the Open Image Denoise library. To enable this functionality, the engine will try to include the "WickedEngine/OpenImageDenoise/oidn.hpp" file. If this file could be included, it will attempt to link with OpenImageDenoise.lib and tbb.lib and any other lib that is supplied in the OpenImageDenoise release version The libs should be located in the $(SolutionDir)BUILD\$(Platform)\$(Configuration) path, which is for example: C:/PROJECTS/WickedEngine/BUILD/x64/Release. It is also required to provide the OpenImageDenoise.dll and tbb.dll and any other dll near the exe to correctly launch the application after this, which is for example: C:\PROJECTS\WickedEngine\BUILD\x64\Release\Editor_Windows. If you satisfy these steps, the denoiser will work automatically after you rebuild the engine and the path tracer reached the target sample count, or lightmap baking is stopped.

LoadingScreen

[Header] [Cpp] The LoadingScreen is an example RenderPath that can be used as a drop-in loading screen for your application. It is designed to be a simple 2D screen that is shown during resource loading. You can add generic tasks with addLoadingFunction(), or to load an other RenderPath using addLoadingComponent(). You can specify what will be called after loading is finished by adding a task to with onFinished(). You can query the progress of loading with the getProgress() function which returns the percentage of loading as an integer in the range 0-100. You can use it to display a loading progress bar for example. The loading will start when Start() is called for the loading screen, but you nedd not call this yourself usually. When you activate your loading screen with the Application class, it will switch to the loading screen when appropriate and start it automatically. With this you can achieve nice fade-in-out of loading screen easily, as the Application's ActivatePath() simply has a parameter for fading duration.

Because the LoadingScreen is a RenderPath2D, it is safe to load a Scene directly into the main scene (with LoadModel()), because RenderPath2D doesn't utilize scene at all for rendering. If you use a custom loading screen that is also performing 3D rendering with a scene, you would instead load into a separate scene with LoadModel(), and merge into your main scene in the LoadingScreen::onFinished() callback.

A simple helper feature of this LoadingScreen class is the BackgroundMode setting. You can set a background texture easily for the LoadingScreen which will simply be shown as a full screen image during loading. For this, you should at least specify the LoadingScreen::bacgroundTexture resource, which you can simply load from an asset file with wi::resourcemanager::Load(). You can set the BackgroundMode of this image via LoadingScreen::background_mode with the following modes:

  • Fill: fill the whole screen, will cut off parts of the image if aspects don't match (default)
  • Fit: fit the image completely inside the screen, will result in black bars on screen if aspects don't match
  • Stretch: fill the whole screen, and stretch the image if needed

If you want to render other things on the LoadingScreen, then you can also override RenderPath2D methods such as Render() or Compose() and implement custom rendering for yourself.

System

You can find out more about the Entity-Component system and other engine-level systems under ENGINE/System filter in the solution.

Entity-Component System

[Header]

ComponentManager

The Component Manager is responsible of binding data (component) to an Entity (identifier). The component can be a c++ struct that contains data for an entity. It supports serialization of this data, and if this is used, then the component structs must have a Serialize() function. Otherwise the component can be any c++ struct that can be moved.

Entity

Entity is an identifier (number) that can reference components through ComponentManager containers. An entity is always valid if it exists. It's not required that an entity has any components. An entity has a component, if there is a ComponentManager that has a component which is associated with the same entity.

Using the entity-component system

To use the entity-component system, you must use the ComponentManager to store components of the T type, where T is an type of c++ struct. To bind a component to an entity, this procedure should be followed:

struct MyComponent
{
	float3 position;
	float speed;
};
ComponentManager<MyComponent> components; // create a component manager
Entity entity = CreateEntity(); // create a new entity ID
MyComponent& component = components.Create(entity); // create a component and bind it to the entity

When you create a new component, it will be added to the end of the array, in a contiguous memory allocation. The ComponentManager can be indexed and iterated efficiently:

for(size_t i = 0; i < components.GetCount(); ++i)
{
	MyComponent& component = components[i];
	Entity entity = components.GetEntity(i);
}

You can see that we can both iterate through components and entities this way by the same index. It is useful when you just want to go through everything one-by one. But it is not useful when you want to query a component for an entity but you don't know its index. Indices can change when components are removed, so you cannot usually rely on remembering indices for entities yourself. However, the component manager does manage this information internally. To query a component for a given entity, use the GetComponent() function:

MyComponent* component = components.GetComponent(entity);
if(component != nullptr) // check for null, which can be returned if component doesn't exist for the entity
{
	// use component
}

You can also get the index of an entity in a component manager:

size_t index = components.GetIndex(entity);
if(index != ~0ull) // check if it's valid. If invalid, then the index returned will be the max value of 64 bit uint
{
	MyComponent& component = components[index];
}

It is NOT valid to use the array operator with entity ID:

// NOT VALID! result will be undefined (could be a valid component, or overindexing crash)
MyComponent& component = components[entity];

A useful pattern is to offload the iteration of a large component manager to the job system, to parallelize the work:

wi::jobsystem::context ctx;
wi::jobsystem::Dispatch(ctx, (uint32_t)components.GetCount(), 64, [&](wi::jobsystem::JobArgs args){
	MyComponent& component = components[args.jobIndex];
	Entity entity = components.GetEntity(args.jobIndex);
	// Do things...
});
// optionally, do unrelated things here...
wi::jobsystem::Wait(ctx); // wait for Dispatch to finish

In this example, each entity/component will be iterated by one job, and 64 batch of jobs will be executed on one thread as a group. This can greatly improve performance when iterating a large component manager.

One thing that you must look out for is pointer invalidation of the Component Manager. It is always safe to use Entity IDs whenever, but it is faster to use a pointer or an index. Those can, hovewer change as components are added or removed. You must look out for index and pointer invalidation when components are removed, because the ordering of the components can change there (the component manager is always kept dense, without holes). When adding/creating components, pointers can occasionally get invalidated the same way as for example std::vector, because reallocation of memory can happen to keep everyting contiguous. This can lead to unexpected consequences:

MyComponent& component1 = components[0];
MyComponent& component2 = components.Create(CreateEntity()); // here the component1 reference/pointer can get invalidated
component1.speed = 10; // POINTER INVALIDATION HAZARD!

In the example above, a new component is created while the reference to the first element is still going to be used, this can lead to crash if the memory got reallocated while creating the component2. The correct solution would be to first create all components, then use them after all creations finished. Or re-query component pointers with GetComponent(entity) after a creation happens.

The Scene structure of the engine is essentially a collection of ComponentManagers, which let you access everything in the scene in the same manner.

Scene System

[Header] [Cpp] The logical scene representation using the Entity-Component System

  • GetScene
    Returns a global scene instance. This is a convenience feature for simple applications that need a single scene. The RenderPath3D will use the global scene by default, but it can be set to a custom scene if needed.
  • LoadModel()
    There are two flavours to this. One of them immediately loads into the global scene. The other loads into a custom scene, which is useful to manage the contents separately. This function will return an Entity that represents the root transform of the scene - if the attached parameter was true, otherwise it will return INVALID_ENTITY and no root transform will be created.
  • LoadModel2()
    This is an alternative usage of LoadModel, which lets you give the root entity ID as a parameter. Apart from that, it works the same way as LoadModel(), just that attachments will be made to your specified root entity.
  • Intersects(Ray/Capsule/Sphere, filterMask, layerMask, lod)
    Intersection function with scene entities and primitives. You can specify various settings to filter intersections. The filterMask lets you specify an engine-defined type enum bitmask combination, so you can choose to intersect with objects, and/or colliders, and various specifications. The layerMask lets you filter the intersections with layer bits, where binary OR of the parameter and entity layers will decide active entities. The lod parameter lets you force a lod level for meshes.
  • Pick
    Allows to pick the closest object with a RAY (closest ray intersection hit to the ray origin). The user can provide a custom scene or layermask to filter the objects to be checked.
  • SceneIntersectSphere
    Performs sphere intersection with all objects and returns the first occured intersection immediately. The result contains the incident normal and penetration depth and the contact object entity ID.
  • SceneIntersectCapsule
    Performs capsule intersection with all objects and returns the first occured intersection immediately. The result contains the incident normal and penetration depth and the contact object entity ID.

Below you will find the structures that make up the scene. These are intended to be simple structures that will be held in ComponentManagers. Keep these structures minimal in size to use cache efficiently when iterating a large amount of components.

Note on bools: using bool in C++ structures is inefficient, because they take up more space in memory than required. Instead, bitmasks will be used in every component that can store up to 32 bool values in each bit. This also makes it easier to add bool flags and not having to worry about serializing them, because the bitfields themselves are already serialized (but the order of flags must never change without handling possible side effects with serialization versioning!). C++ enums are used in the code to manage these bool flags, and the bitmask storing these is always called uint32_t _flags; For example:

enum FLAGS
{
	EMPTY = 0, // always have empty as first flag (value of zero)
	RENDERABLE = 1 << 0,
	DOUBLE_SIDED = 1 << 1,
	DYNAMIC = 1 << 2,
};
uint32_t _flags = EMPTY; // default value is usually EMPTY, indicating that no flags are set

// How to query the "double sided" flag:
bool IsDoubleSided() const { return _flags & DOUBLE_SIDED; }

// How to set the "double sided" flag
void SetDoubleSided(bool value) { if (value) { _flags |= DOUBLE_SIDED; } else { _flags &= ~DOUBLE_SIDED; } }

It is good practice to not implement constructors and destructors for components. Wherever possible, initialization of values in declaration should be preferred. If desctructors are defined, move contructors, etc. will also need to be defined for compatibility with the ComponentManager, so default constructors and destructors should be preferred. Member objects should be able to desctruct themselves implicitly. If pointers need to be stored within the component that manage object lifetime, std::unique_ptr or std::shared_ptr can be used, which will be destructed implicitly.

NameComponent

[Header] [Cpp] A string to identify an entity with a human readable name.

LayerComponent

[Header] [Cpp] A bitmask that can be used to filter entities. Depending on what other components the entity has, the layer will be used by different systems, to for example determine visibility, filter collisions, or picking. A bitmask of 0 will make the entity inactive in most systems. The lower 8 bits will be used by ray tracing as an instance inclusion mask for instances (ObjectComponent, EmittedParticle, HairParticle..)

TransformComponent

[Header] [Cpp] Orientation in 3D space, which supports various common operations on itself.

PreviousFrameTransformComponent

[Header] [Cpp] Absolute orientation in the previous frame (a matrix).

HierarchyComponent

[Header] [Cpp] An entity can be part of a transform hierarchy by having this component. Some other properties can also be inherieted, such as layer bitmask. If an entity has a parent, then it has a HierarchyComponent, otherwise it's not part of a hierarchy.

MaterialComponent

[Header] [Cpp] Several properties that define a material, like color, textures, etc...

MeshComponent

[Header] [Cpp] A mesh is an array of triangles. A mesh can have multiple parts, called MeshSubsets. Each MeshSubset has a material and it is using a range of triangles of the mesh. This can also have GPU resident data for rendering.

ImpostorComponent

[Header] [Cpp] Supports efficient rendering of the same mesh multiple times (but as an approximation, such as a billboard cutout). A mesh can be rendered as impostors for example when it is not important, but has a large number of copies.

ObjectComponent

[Header] [Cpp] An ObjectComponent is an instance of a mesh that has physical location in a 3D world. Multiple ObjectComponents can have the same mesh, and in this case the mesh will be rendered multiple times efficiently. It is expected that an entity that has ObjectComponent, also has TransformComponent and AABB (axis aligned bounding box) component.

RigidBodyPhysicsComponent

[Header] [Cpp] Stores properties required for rigid body physics simulation and a handle that will be used by the physicsengine internally.

SoftBodyPhysicsComponent

[Header] [Cpp] Stores properties required for soft body physics simulation and a handle that will be used by the physicsengine internally.

ArmatureComponent

[Header] [Cpp] A skeleton used for skinning deformation of meshes.

LightComponent

[Header] [Cpp] A light in the scene that can shine in the darkness. It is expected that an entity that has LightComponent, also has TransformComponent and AABB (axis aligned bounding box) component.

CameraComponent

[Header] [Cpp]

EnvironmentProbeComponent

[Header] [Cpp] It is expected that an entity that has EnvironmentProbeComponent, also has TransformComponent and AABB (axis aligned bounding box) component.

ForceFieldComponent

[Header] [Cpp]

DecalComponent

[Header] [Cpp] It is expected that an entity that has DecalComponent, also has TransformComponent and AABB (axis aligned bounding box) component.

AnimationDataComponent

[Header] [Cpp]

AnimationComponent

[Header] [Cpp]

WeatherComponent

[Header] [Cpp]

SoundComponent

[Header] [Cpp] Holds a Sound and a SoundInstance, and it can be placed into the scene via a TransformComponent. It can have a 3D audio effect it has a TransformComponent.

InverseKinematicsComponent

[Header] [Cpp] If an entity has an InverseKinematicComponent (IK), and part of a transform hierarchy (has both TransformComponent and a HierarchyComponent), then it can be targetted to an other TransformComponent. The parent transforms will be computed in order to let the IK reach the target if possible. The parent transforms will be only rotated. For example, if a hand tries to reach for an object, the hand and shoulder will move accordingly to let the hand reach. The chain_length can be specified to let the IK system know how many parents should be computed. It can be greater than the real chain length, in that case there will be no more simulation steps than the length of hierarchy chain. The iteration_count can be specified to increase accuracy of the computation. If animations are also playing on the affected entities, the IK system will override the animations.

SpringComponent

[Header] [Cpp] An entity can have a SpringComponent to achieve a "jiggle" or "soft" animation effect programatically. The effect will work automatically if the transform is changed by animation system for example, or in any other way. The parameter stiffness specifies how fast the transform tries to go back to its initial position. The parameter damping specifies how fast the transform comes to rest position. The wind_affection parameter specifies how much the global wind applies to the spring.

ColiderComponent

[Header] [Cpp] The collider component will specify a collider shape to be used in simple fake physics simulations. It will affect Springs and GPU particle effect.

ScriptComponent

[Header] [Cpp] ScriptComponent can reference a lua script and run it every frame, while also providing some additional data to script like a local GetEntity() function. The script can be written to reference additional component data by using its unique GetEntity() function. A ScriptComponent can also call other scripts, which can be used to implement multiple scripts on one entity.

Scene

[Header] [Cpp] A scene is a collection of component arrays. The scene is updating all the components in an efficient manner using the job system. It can be serialized and saved/loaded from disk efficiently.

  • Update(float deltatime)
    This function runs all the requied systems to update all components contained within the Scene.

Job System

[Header] [Cpp] Manages the execution of concurrent tasks

  • context
    Defines a single workload that can be synchronized. It is used to issue jobs from within jobs and properly wait for completion. A context can be simply created on the stack because it is a simple atomic counter.
  • Execute
    This will schedule a task for execution on a separate thread for a given workload
  • Dispatch
    This will schedule a task for execution on multiple parallel threads for a given workload
  • Wait
    This function will block until all jobs have finished for a given workload. The current thread starts working on any work left to be finished.

Initializer

[Header] [Cpp] Initializes all engine systems either in a blocking or an asynchronous way.

Platform

[Header] You can get native platform specific functionality.

Event Handler

[Header] The event system can be used to execute system-wide tasks. Any system can "subscribe" to events and any system can "fire" events.

  • Subscribe
    The first parameter is the event ID. Core system events are negative numbers. The user can choose any positive number to create custom events. The second parameter is a function to be executed, with a userdata argument. The userdata argument can hold any custom data that the user desires. The return value is a wi::eventhandler::Handle type. When this is object is destroyed, the event subscription for the function will be removed. Multiple functions can be subscribed to a single event ID.
  • FireEvent
    The first argument is the event id, that says which events to invoke. The second argument will be passed to the subscribed event's userdata parameter. All events that are subsribed to the specified event id will run immediately at the time of the call of FireEvent. The order of execution among events is the order of which they were subscribed.

Canvas

[Header] The canvas specifies a DPI-aware drawing area. Use the GetLogical... functions to get parameters with DPI aware scale factors applied. Use the GetPhysical... functions to get the real pixel counts.

Graphics

Everything related to rendering graphics will be discussed below

Graphics API

[Header] The graphics API wrappers

GraphicsDevice

[Header] [Cpp] This is the interface for the graphics API abstraction. It is higher level than a graphics API, but these are the lowest level of rendering commands the engine offers.

Debug device

When creating a graphics device, it can be created in special debug mode to assist development. For this, the user can specify debugdevice as command line argument to the application, or set the debuglayer argument of the graphics device constructor to true when creating the device. The errors will be posted to the Console Output window in the development environment.

Creating resources

Functions like CreateTexture(), CreateBuffer(), etc. can be used to create corresponding GPU resources. Using these functions is thread safe. The resources will not necessarily be created immediately, but by the time the GPU will want to use them. These functions immediately return false if there were any errors, such as wrong parameters being passed to the description parameters, and they will return true if everything is correct. If there was an error, please use the debug device functionality to get additional information. When passing a resource to these functions that is already created, it will be destroyed, then created again with the newly provided parameters.

Destroying resources

Resources will be destroyed automatically by the graphics device when they are no longer used.

Work submission

Rendering commands that expect a CommandList as a parameter are not executed immediately. They will be recorded into command lists and submitted to the GPU for execution upon calling the SubmitCommandLists() function. The CommandList is a simple handle that associates rendering commands to a CPU execution timeline. The CommandList is not thread safe, so every CommandList can be used by a single CPU thread at a time to record commands. In a multithreading scenario, each CPU thread should have its own CommandList. CommandLists can be retrieved from the GraphicsDevice by calling GraphicsDevice::BeginCommandList() that will return a CommandList handle that is free to be used from that point by the calling thread. All such handles will be in use until SubmitCommandLists() or PresentEnd() was called, where GPU submission takes place. The command lists will be submitted in the order they were retrieved with GraphicsDevice::BeginCommandList(). The order of submission correlates with the order of actual GPU execution. For example:

CommandList cmd1 = device->BeginCommandList();
CommandList cmd2 = device->BeginCommandList();

Read_Shadowmaps(cmd2); // CPU executes these commands first
Render_Shadowmaps(cmd1); // CPU executes these commands second

//...

CommandList cmd_present = device->BeginCommandList();

device->SubmitCommandLists(cmd_present); // CPU submits work for GPU
// The GPU will execute the Render_Shadowmaps() commands first, then the Read_Shadowmaps() commands second

When submitting command lists with SubmitCommandLists(), the CPU thread can be blocked in cases when there is too much GPU work submitted already that didn't finish. It's not appropriate to start recording new command lists until SubmitCommandLists() finished.

Furthermore, the BeginCommandList() is thread safe, so the user can call it from worker threads if ordering between command lists is not a requirement (such as when they are producing workloads that are independent of each other).

Async compute

Async workload can be performed on the compute queue with CommandList granularity. The BeginCommandList() function has an optional QUEUE_TYPE parameter, which can be specified to execute the command list on the specified queue. By default, if no such parameter is specified, the work will be executed on the main graphics queue (QUEUE_GRAPHICS), serially to other such graphics work. In case the graphics device or API doesn't support async compute (such as DX11), then the QUEUE_GRAPHICS will be assumed. The main graphics queue can execute both graphics and compute work, but the QUEUE_COMPUTE can only execute compute work. Two queues can also be synchronized with CommandList granularity, with the WaitCommandList() function. This function inserts a dependency barrier before the first parameter command list, that synchronizes with the second parameter command list. For example:

CommandList cmd0 = device->BeginCommandList(QUEUE_GRAPHICS);
CommandList cmd1 = device->BeginCommandList(QUEUE_COMPUTE);
device->WaitCommandList(cmd1, cmd0); // cmd1 waits for cmd0 to finish
CommandList cmd2 = device->BeginCommandList(QUEUE_GRAPHICS); // cmd2 doesn't wait, it runs async with cmd1
CommandList cmd3 = device->BeginCommandList(QUEUE_GRAPHICS);
device->WaitCommandList(cmd3, cmd1); // cmd3 waits for cmd1 to finish

device->SubmitCommandLists(); // execute all of the above

The WaitCommandList() function is a GPU wait operation, so it will not block CPU execution. Furthermore, it is not required to use this between two CommandLists that are on the same queue, because the synchronization between those is implicit.

Important: The SHADER_RESOURCE state cannot be used on the compute queue. The device could convert these to SHADER_RESOURCE_COMPUTE while issuing Barrier() commands. However, the starting resource state must be correctly specified, because those cannot be converted. Consider always choosing a SHADER_RESOURCE_COMPUTE starting resource state if the resource is going to be used in a compute queue, and transition them to regular SHADER_RESOURCE only before the resource is going to be used in a pixel shader. The graphics queue with compute commands doesn't have this limitation however.

Presenting to the screen

To present to the screen (an operating system window), first create a SwapChain with the CreateSwapChain() function that will be associated with a window. The SwapChain acts as a special kind of RenderPass, so there is a BeginRenderPass() function with an overload that accepts a SwapChain parameter instead of a RenderPass. Simply use this BeginRenderPass() and EndRenderPass() to draw to the SwapChain. The final presentation will happen when calling SubmitCommandLists().

HDR Display

To present content to a HDR display, set the SwapChainDesc::allow_hdr to true when creating the SwapChain. Also, select a texture format for the swapChain that can be used for HDR content. The formats that are capable of HDR, are:

  • R10G10B10A2_UNORM, which supports both the SRGB (which is SDR) and HDR10_ST2084 (HDR10) color spaces.
  • R16G16B16A16_FLOAT, which supports the HDR_LINEAR color space.

If the display associated with the SwapChain doesn't support HDR output, HDR will be disabled and the SwapChain can fall back to an appropriate format that is supported. To check the final color space of the SwapChain, call the GraphicsDevice::GetSwapChainColorSpace() function providing a valid SwapChain as argument. The function returns the actual COLOR_SPACE of the SwapChain. To check whether the display associated with the SwapChain is HDR capable, call the GraphicsDevice::IsSwapChainSupportsHDR() function providing a valid SwapChain as argument. This will return whether the display supports HDR or not, regardless of the current format of the SwapChain.

It is not enough to set up a HDR SwapChain to render correct HDR content, because great care must be taken to blend elements in linear color space, and correctly convert to the display's color space before presenting. This is a responsibility of shaders.

Resource binding

The resource binding model is based on DirectX 11. That means, resources are bound to slot numbers that are simple integers.

Declare a resource bind point with HLSL syntax:

Texture2D<float4> myTexture : register(t5);

The slot number t5 must be specified to avoid the HLSL compiler assign an arbitrary slot to it and will be accessible from the CPU.

The user can bind the texture resource from the CPU side:

Texture myTexture;
// after texture was created, etc:
device->BindResource(&myTexture, 5, cmd);

Other than this, resources like Texture can have different subresources, so an extra parameter can be specified to the BindResources() function called subresource:

Texture myTexture;
// after texture was created, etc:
device->BindResource(&myTexture, 5, cmd, 42);

By default, the subresource parameter is -1, which means that the entire resource will be bound. For more information about subresources, see the Subresources section.

There are different resource types regarding how to bind resources. These slots have separate binding points from each other, so they don't interfere between each other.

  • Shader resources
    These are read only resources. GPUBuffers and Textures can be bound as shader resources if they were created with a BindFlags in their description that has the SHADER_RESOURCE bit set. Use the GraphicsDevice::BindResource() function to bind a single shader resource, or the GraphicsDevice::BindResources() to bind a bundle of shader resources, occupying a number of slots starting from the bind slot. Use the GraphicsDevice::UnbindResources() function to unbind several shader resource slots manually, which is useful for removing debug device warnings.
  • UAV
    Unordered Access Views, in other words resources with read-write access. GPUBuffers and Textures can be bound as shader resources if they were created with a BindFlags in their description that has the UNORDERED_ACCESS bit set. Use the GraphicsDevice::BindUAV() function to bind a single UAV resource, or the GraphicsDevice::BindUAVs() to bind a bundle of UAV resources, occupying a number of slots starting from the bind slot. Use the GraphicsDevice::UnbindUAVs() function to unbind several UAV slots manually, which is useful for removing debug device warnings.
  • Constant buffers
    Only GPUBuffers can be set as constant buffers if they were created with a BindFlags in their description that has the CONSTANT_BUFFER bit set. The resource can't be a constant buffer at the same time when it is also a shader resource or a UAV or a vertex buffer or an index buffer. Use the GraphicsDevice::BindConstantBuffer() function to bind constant buffers.
  • Samplers
    Only Sampler can be bound as sampler. Use the GraphicsDevice::BindSampler() function to bind samplers.

There are some limitations on the maximum value of slots that can be used, these are defined as compile time constants in Graphics device SharedInternals. The user can modify these and recompile the engine if the predefined slots are not enough. This could slightly affect performance.

Remarks:

  • Vulkan and DX12 devices make an effort to combine descriptors across shader stages, so overlapping descriptors will not be supported with those APIs to some extent. For example it is OK, to have a constant buffer on slot 0 (b0) in a vertex shader while having a Texture2D on slot 0 (t0) in pixel shader. However, having a StructuredBuffer on vertex shader slot 0 (t0) and a Texture2D in pixel shader slot 0 (t0) will not work correctly, as only one of them will be bound to a pipeline state. This is made for performance reasons.
  • This slot based binding model has a CPU overhead that has to be kept in mind. Avoid binding massive amount of resources. The maximum slot numbers that should be used are: 14 for BindConstantBuffer(), 16 for BindResource(), 16 for BindUAV() and 8 for BindSampler(). Consider using a bindless model when you need to bind more than a couple of resources.
Bindless resources

Wicked Engine supports bindless resource management, this can greatly improve performance and removes resource binding constraints to allow great flexibility.

Related functions to this feature:

  • GetDescriptorIndex() : returns an int that identifies the resource in bindless space. The queried resource can be a Sampler or a GPUResource. If the resource is not usable (for example if it was not created), then the function returns -1. In this case, the shaders must not use the resource, but instead rely on dynamic branching to avoid it, because this would be undefined behaviour and could result in a GPU hang. Otherwise, the index can be used by shaders to index into a descriptor heap.
  • PushConstants() : This is an easy way to set a small amount of 32-bit values on the GPU, usable by shaders that declared a PUSHCONSTANT(name, type) block. There can be one push constant block per pipeline (graphics, compute or raytracing). The push constants will be bound to the last set pipeline, so only use this after binding a graphics pipeline state or compute shader.

The shaders can use bindless descriptors with the following syntax example:

Texture2D<float4> bindless_textures[] : register(t0, space5);
struct PushConstants
{
	uint materialIndex;
	int textureindex;
};
PUSHCONSTANT(push, PushConstants);
// ...
float4 color = bindless_textures[push.textureindex].Sample(sam, uv);

The corresponding data can be fed from the CPU like this:

struct PushConstants
{
	uint materialIndex;
	int textureindex;
};
PushConstants push;
push.materialIndex = device->GetDescriptorIndex(materialCB, SRV);
push.textureindex = device->GetDescriptorIndex(texture, SRV);
device->PushConstants(&push, sizeof(push), cmd);

Note: a descriptor array of ConstantBuffer could be supported by some hardware/API (eg. Vulkan) in a limited form even though bindless descriptors are supported, so avoid relying on it too much.

The regular slot based binding model will work alongside the bindless model seamlessly. The slot based bindings will always use space0 implicitly, space1 and greater should be used for the bindless model. Aim to keep space numbers as low as possible.

Subresources

Resources like textures can have different views. For example, if a texture contains multiple mip levels, each mip level can be viewed as a separate texture with one mip level, or the whole texture can be viewed as a texture with multiple mip levels. When creating resources, a subresource that views the entire resource will be created. Functions that expect a subresource parameter can be provided with the value -1 that means the whole resource. Usually, this parameter is optional.

Other subresources can be create with the GraphicsDevice::CreateSubresource() function. The function will return an int value that can be used to refer to the subresource view that was created. In case the function returns -1, the subresource creation failed due to an incorrect parameter. Please use the debug device functionality to check for errors in this case. The use of CreateSubresource() function is thread safe on the device, but not on the resource, so only 1 thread must modify a given resource at a time.

The subresource indices are valid as long as the resource is valid that they were created with.

Pipeline States and Shaders

PipelineStates are used to define the graphics pipeline state, that includes which shaders are used, which blend mode, rasterizer state, input layout, depth stencil state and primitive topology, as well as sample mask are in effect. These states can only be bound atomically in a single call to GraphicsDevice::SetPipelineState(). This does not include compute shaders, which do not participate in the graphics pipeline state and can be bound individually using GraphicsDevice::BindComputeShader() method.

The pipeline states are subject to shader compilations. Shader compilation will happen when a pipeline state is bound inside a render pass for the first time. This is required because the render target formats are necessary information for compilation, but they are not part of the pipeline state description. This choice was made for increased flexibility of defining pipeline states. However, unlike APIs where state subsets (like RasterizerDesc, or BlendStateDesc) can be bound individually, the grouping of states is more optimal regarding CPU time, because state hashes are computed only once for the whole pipeline state at creation time, as opposed to binding time for each individual state. This approach is also less prone to user error when the developer might forget setting any subset of state and the leftover state from previous render passes are incorrect.

Shaders still need to be created with GraphicsDevice::CreateShader() in a similar way to CreateTexture(), etc. The CreateShader() function expects a wiGraphics::ShaderStage enum value which will define the type of shader:

  • MS: Mesh Shader
  • AS: Amplification Shader, or Task Shader
  • VS: Vertex Shader
  • HS: Hull Shader, or Tessellation Control Shader
  • DS: Domain Shader, or Tessellation Evaluation Shader
  • GS: Geometry Shader
  • PS: Pixel Shader
  • CS: Compute Shader
  • LIB: Library shader

Depending on the graphics device implementation, the shader code must be different format. For example, DirectX expects DXIL shaders, Vulkan expects SPIR-V shaders. The engine can compile HLSL shader source code to DXIL or SPIRV format with the Shader Compiler.

Note: As an optional parameter, CreatePipelineState() function also accepts a RenderPassInfo pointer. If this is provided, then all the information will be available for pipeline creation to happen immediately. This means the pipeline will be immediately in final usable state after the function returns, but it can take longer time for this to complete. If the parameter is not provided, the pipeline could be created at a later time, when the render pass information is available on first use. This can have the effect of the longer pipeline compilation happening at an unexpected time. But in this case, the pipeline will also be compatible with more render pass types if it's used at different places.

Render Passes

Render passes are defining regions in GPU execution where a number of render targets or depth buffers will be used to render into them. Render targets and depth buffers are defined as RenderPassImages. The RenderPassImages have a pointer to the texture, state the resource type (RENDERTARGET, DEPTH_STENCIL or RESOLVE), state the subresource index, the load and store operations, and the layout transitions for the textures.

  • RENDERTARGET: The image will be used as a (color) render target. The order of these images define the shader color output order.

  • DEPTH_STENCIL: The image will be used as a depth (and/or stencil) buffer.

  • RESOLVE: The image will be used as MSAA resolve destination. The resolve source is chosen among the RENDERTARGET images in the same render pass, in the order they were declared in. The declaration order of the RENDERTARGET and RESOLVE images must match to correctly deduce source and destination targets for resolve operations.

  • RESOLVE_DEPTH same as RESOLVE but for depth stencil. You can also specify Min or Max resolve operation with the RenderPassImage::DepthResolveMode enum. This feature must be supported by GPU and indicated by GraphicsDeviceCapability::DEPTH_RESOLVE_MIN_MAX and GraphicsDeviceCapability::STENCIL_RESOLVE_MIN_MAX (if image also has stencil)

  • Load Operation:
    Defines how the texture contents are initialized at the start of the render pass. LoadOp::LOAD says that the previous texture content will be retained. LoadOp::CLEAR says that the previous contents of the texture will be lost and instead the texture clear color will be used to fill the texture. LoadOp::DONTCARE says that the texture contents are undefined, so this should only be used when the developer can ensure that the whole texture will be rendered to and not leaving any region empty (in which case, undefined results will be present in the texture).

  • Store operation:
    Defines how the texture contents are handled after the render pass ends. StoreOp::STORE means that the contents will be preserved. StoreOp::DONTCARE means that the contents won't be necessarily preserved, they are only temporarily valid within the duration of the render pass, which can save some memory bandwidth on some platforms (specifically tile based rendering architectures, like mobile GPUs).

  • Layout transition:
    Define the layout_before, layout (only for DEPTH_STENCIL) and layout_after members to have an implicit transition performed as part of the render pass, that works like an IMAGE_BARRIER, but can be more optimal. The layout_before states the starting state of the resource. The resource will be transitioned from layout_before to layout within the render pass. The layout states how the resource is accessed within the render pass. For DEPTH_STENCIL type, it must be either DEPTHSTENCIL or DEPTHSTENCIL_READONLY. For RESOLVE type, the subpass_layout have no meaning, it is implicitly defined. At the end of the render pass, the resources will be transitioned from layout to layout_after.

Notes:

  • When RenderPassBegin() is called, RenderPassEnd() must be called after on the same command list before the command list gets submitted.
  • It is not allowed to call RenderPassBegin() inside a render pass.
  • It is not allowed to call CopyResource(), CopyTexture(), CopyBuffer() inside a render pass.
  • It is not allowed to call Dispatch(), DispatchIndirect(), DispatchMesh(), DispatchMeshIndirect(), DispatchRays() inside a render pass.
  • It is not allowed to call UpdateBuffer() inside the render pass.
  • It is not allowed to call Barrier() inside the render pass.
  • It is not allowed to call ClearUAV() inside the render pass.
GPU Barriers

GPUBarriers can be used to state dependencies between GPU workloads. There are different kinds of barriers:

  • MEMORY_BARRIER
    Memory barriers are used to wait for UAV writes to finish, or in other words to wait for shaders to finish that are writing to a UNORDERED_ACCESS resource. The GPUBarrier::memory.resource member is a pointer to the GPUResource to wait on. If it is nullptr, than the barrier means "wait for every UAV write that is in flight to finish".
  • IMAGE_BARRIER
    Image barriers are stating resource state transition for textures. The most common use case for example is to transition from RENDERTARGET to SHADER_RESOURCE, which means that the RenderPass that writes to the texture as render target must finish before the barrier, and the texture can be used as a read only shader resource after the barrier. There are other cases that can be indicated using the GPUBarrier::image.layout_before and GPUBarrier::image.layout_after states. The GPUBarrier::image.resource is a pointer to the resource which will have its state changed. If the texture's layout (as part of the TextureDesc) is not SHADER_RESOURCE, the layout must be transitioned to SHADER_RESOURCE before binding as shader resource. The image layout can also be transitioned using a RenderPass, which should be preferred to GPUBarriers.
  • BUFFER_BARRIER
    Similar to IMAGE_BARRIER, but for GPU Buffer state transitions.

It is very important to place barriers in the right places; missing a barrier could lead to corruption of rendered results, crashes and generally undefined behaviour. The debug layer will help to detect errors and missing barriers.

Textures

Texture type resources are used to store images which will be sampled or written by the GPU efficiently. There are a lot of texture types, such as Texture1D, Texture2D, TextureCube, Texture3D, etc. used in shaders, that correspond to the simple Texture type on the CPU. The texture type will be determined when creating the resource with GraphicsDevice::CreateTexture(const TextureDesc*, const SubresourceData*, Texture*) function. The first argument is the TextureDesc that determines the dimensions, size, format and other properties of the texture resource. The SubresourceData is used to initialize the texture contents, it can be left as nullptr, when the texture contents don't need to be initialized, such as for textures that will be rendered into. The Texture is the texture that will be initialized.

SubresourceData usage:

  • The SubresourceData parameter points to an array of SubresourceData structs. The array size is determined by the TextureDesc::ArraySize * TextureDesc::MipLevels, so one structure for every subresource.
  • The SubresourceData::pData pointer should point to the texture contents on the CPU that will be uploaded to the GPU.
  • The SubresourceData::rowPitch member is indicating the distance (in bytes) from the beginning of one line of a texture to the next line. System-memory pitch is used only for 2D and 3D texture data as it is has no meaning for the other resource types. Specify the distance from the first pixel of one 2D slice of a 3D texture to the first pixel of the next 2D slice in that texture in the slicePitch member.
  • The SubresourcData::slicePitch member is indicating the distance (in bytes) from the beginning of one depth level to the next. System-memory-slice pitch is only used for 3D texture data as it has no meaning for the other resource types.
  • For more complete information, refer to the DirectX 11 documentation on this topic, which should closely match.

Related topics: Creating Resources, Destroying Resources, Binding resources, Subresources

GPU Buffers

GPUBuffer type resources are used to store linear data which will be read or written by the GPU. There are a lot of buffer types, such as structured buffers, raw buffers, vertex buffers, index buffers, typed buffers, etc. used in shaders, that correspond to the simple GPUBuffer type on the CPU. The buffer type will be determined when creating the buffer with GraphicsDevice::CreateBuffer(const GPUBufferDesc*, const void*, GPUBuffer*) function. The first argument is the GPUBufferDesc that determines the dimensions, size, format and other properties of the texture resource. The pInitialData is used to initialize the buffer contents, it can be left as nullptr. The GPUBuffer is the buffer that will be initialized.

Related topics: Creating Resources, Destroying Resources, Binding resources, Subresources, Updating GPU Buffers

Updating GPU buffers

A GPUBuffer's Usage parameter specifies how the buffer memory is accessed.

  • DEFAULT: The buffer memory is visible to the GPU but not the CPU. This means that it will observe maximum GPU performance, but special care needs to be taken to write the buffer contents. The GPU could write the memory from a shader or copy operation for example. You can also use the UpdateBuffer() function to update such a buffer from CPU (which uses a GPU copy).
  • UPLOAD: The buffer can be written by the CPU and read by the GPU. Once such a GPUBuffer was created, it's memory is persistently mapped for CPU access, and can be accessed through the GPUResource::mapped_data pointer. It's perfect to update a DEFAULT buffer from this by first filling the UPLOAD buffer from the CPU, then let the GPU copy its contents to the DEFAULT buffer with a shader or copy operation.
  • READBACK: The buffer can be written by the GPU and the contents read by the CPU. The buffer memory is persistently mapped after creation, and accessible through the GPUResource::mapped_data pointer.
GPU Queries

The GPUQueryHeap can be used to retrieve information from GPU to CPU. There are different query types (GpuQueryType enum):

  • TIMESTAMP is used to write a timestamp value to the designated query inside the heap. Use GraphicsDevice::QueryEnd() to record this type.
  • OCCLUSION is used to retrieve depth test passed sample count for drawing commands between GraphicsDevice::QueryBegin() and GraphicsDevice::QueryEnd().
  • OCCLUSION_BINARY is the same as OCCLUSION, but only returns whether any samples passed depth test or not. It can be more optimal than OCCLUSION.

The GPUQueryHeap is designed to retrieve query results in a bulk, instead of one by one, which can be implemented more optimally. However, retrieving queries one by one will still be possible if needed. The GraphicsDevice::QueryResolve() function will issue a GPU operation that will write the query results to a CPU visible heap. The GraphicsDevice::QueryRead() function can be called on the CPU timeline to read resolved query data in bulk. Reading queries must be done when the GPU finished executing the GraphicsDevice::QueryResolve(), which is usually after a few frames of latency.

RayTracingAccelerationStructure

The acceleration strucuture can be bottom level or top level, and decided by the description strucutre's type field. Depending on the type, either the toplevel or bottomlevel member must be filled out before creating the acceleration structure. When creating the acceleration structure with the grpahics device (GraphicsDevice::CreateRaytracingAccelerationStructure()), sufficient backing memory is allocated, but the acceleration structure is not built. Building the acceleration structure is performed on the GPU timeline, via the GraphicsDevice::BuildRaytracingAccelerationStructure(). To build it from scratch, leave the src argument of this function to nullptr. To update the acceleration structure without doing a full rebuild, specify a src argument. The src can be the same acceleration structure, in this case the update will be performed in place. To be able to update an acceleration structure, it must have been created with the RaytracingAccelerationStructureDesc::FLAG_ALLOW_UPDATE flag.

RayTracingPipelineState

Binding a ray tracing pipeline state is required to dispatch ray tracing shaders. A ray tracing pipeline state holds a collection of shader libraries and hitgroup definitions. It also declares information about max resource usage of the pipeline.

Variable Rate Shading

Variable Rate Shading can be used to decrease shading quality while retaining depth testing accuracy. The shading rate can be set up in different ways:

  • BindShadingRate(): Set the shading rate for the following draw calls. The first parameter is the shading rate, which is by default RATE_1X1 (the best quality). The increasing enum values are standing for decreasing shading rates.
  • Shading rate image: Set the shading rate for the screen via a tiled texture. The texture must be set as a RenderPassAttachment of RATE_SOURCE type. The texture must be using the R8_UINT format. In each pixel, the texture contains the shading rate value for a tile of pixels (8x8, 16x16 or 32x32). The tile size can be queried via GetVariableRateShadingTileSize(). The shading rate values that the texture contains are not the raw values from ShadingRate enum, but they must be converted to values that are native to the graphics API used using the WriteShadingRateValue() function. The shading rate texture must be written with a compute shader and transitioned to RATE_SOURCE with a GPUBarrier before setting it with BindShadingRateImage(). It is valid to set a nullptr instead of the texture, indicating that the shading rate is not specified by a texture.
  • Or setting the shading rate from a vertex or geometry shader with the SV_ShadingRate system value semantic.

The final shading rate will be determined from the above methods using the maximum shading rate (least detailed) which is applicable to the screen tile. In the future it might be considered to expose the operator to define this.

To read more about variable rate shading, refer to the DirectX specifications.

Video Decoding

Video decoding can be used to decode compressed video bitstream in real time on the GPU. Currently only the H264 format is supported. To decode a video, the following steps need to be taken:

  • Provide H264 slice data in a GPUBuffer. The slice data offset needs to be aligned within the buffer with GraphcisDevice::GetVideoDecodeBitstreamAlignment(). The bitstream buffer needs to be an UPLOAD buffer currently.
  • Create VideoDecoder object, while providing appropriate paraemters parsed from H264 such as resolution, picture parameters, sequence parameters. The decode format must be Format::NV12 which is a multi planar YUV420 format.
  • Create a "Decode Picture Buffer" (DPB), which is a texture array storing reference and decoded images in the native video format (currently only Format::NV12 is supported). Indicate in the misc_flags that it will be used as ResourceMiscFlags::VIDEO_DECODE.
  • Run the GraphicsDevice::VideoDecode operation providing correct arguments to decode a single video frame. You are responsible to provide correct H264 arguments and DPB picture indices.
  • You will need to manually read from the DPB texture (eg. in a shader) and resolve the YUV format to RGB if you want.

All this is demonstrated in the wi::video implementation.

GraphicsDevice_DX11

The DirectX11 interface has been removed after version 0.56

GraphicsDevice_DX12

[Header] [Cpp] DirectX12 implementation for rendering interface

GraphicsDevice_Vulkan

[Header] [Cpp] Vulkan implementation for rendering interface

Renderer

[Header] [Cpp] This is a collection of graphics technique implentations and functions to draw a scene, shadows, post processes and other things. It is also the manager of the GraphicsDevice instance, and provides other helper functions to load shaders from files on disk.

Apart from graphics helper functions that are mostly independent of each other, the renderer also provides facilities to render a Scene. This can be done via the high level DrawScene, DrawSky, etc. functions. These don't set up render passes or viewports by themselves, but they expect that they are set up from outside. Most other render state will be handled internally, such as constant buffers, stencil, blendstate, etc.. Please see how the scene rendering functions are used in the High level interface RenderPath3D implementations (for example RenderPath3D_TiledForward.cpp)

Other points of interest here are utility graphics functions, such as CopyTexture2D, GenerateMipChain, and the like, which provide customizable operations, such as border expand mode, Gaussian mipchain generation, and things that wouldn't be supported by a graphics API.

The renderer also hosts post process implementations. These functions are independent and have clear input/output parameters. They shouldn't rely on any other extra setup from outside.

Read about the different features of the renderer in more detail below:

DrawScene

Renders the scene from the camera's point of view that was specified as parameter. Only the objects within the camera Frustum will be rendered. The objects will be sorted from front-to back. This is an optimization to reduce overdraw, because for opaque objects, only the closest pixel to the camera will contribute to the rendered image. Pixels behind the frontmost pixel will be culled by the GPU using the depth buffer and not be rendered. The sorting is implemented with RenderQueue internally. The RenderQueue is responsible to sort objects by distance and mesh index, so instanced rendering (batching multiple drawable objects into one draw call) and front-to back sorting can both work together.

The renderPass argument will specify what kind of render pass we are using and specifies shader complexity and rendering technique. The cmd argument refers to a valid CommandList. The flags argument can contain various modifiers that determine what kind of objects to render, or what kind of other properties to take into account:

  • DRAWSCENE_OPAQUE: Opaque object will be rendered
  • DRAWSCENE_TRANSPARENT: Transparent objects will be rendered. Objects will be sorted back-to front, for blending purposes
  • DRAWSCENE_OCCLUSIONCULLING: Occluded objects won't be rendered. Occlusion culling can be globally switched on/off using wi::renderer::SetOcclusionCullingEnabled()
  • DRAWSCENE_TESSELLATION: Enable tessellation (if hardware supports it). Tessellation can be globally switched on/off using wi::renderer::SetTessellationEnabled()
  • DRAWSCENE_HAIRPARTICLE: Allow drawing hair particles
  • DRAWSCENE_IMPOSTOR : Allow drawing impostors
  • DRAWSCENE_OCEAN : Allow drawing the ocean
  • DRAWSCENE_SKIP_PLANAR_REFLECTION_OBJECTS : Don't draw the objects which are rendering planar reflections. This is to avoid rendering the reflector object itself into the planar reflections, which could produce fully occluded reflections.

Tessellation

Tessellation can be used when rendering objects. Tessellation requires a GPU hardware feature and can enable displacement mapping on vertices or smoothing mesh silhouettes dynamically while rendering objects. Tessellation will be used when tessellation parameter to the DrawScene was set to true and the GPU supports the tessellation feature. Tessellation level can be specified per MeshComponent's tessellationFactor parameter. Tessellation level will be modulated by distance from camera, so that tessellation factor will fade out on more distant objects. Greater tessellation factor means more detailed geometry will be generated.

Occlusion Culling

Occlusion culling is a technique to determine which objects are within the camera, but are completely behind an other objects, such that they wouldn't be rendered. The depth buffer already does occlusion culling on the GPU, however, we would like to perform this earlier than submitting the mesh to the GPU for drawing, so essentially do the occlusion culling on CPU. A hybrid approach is used here, which uses the results from a previously rendered frame (that was rendered by GPU) to determine if an object will be visible in the current frame. For this, we first render the object into the previous frame's depth buffer, and use the previous frame's camera matrices, however, the current position of the object. In fact, we only render bounding boxes instead of objects, for performance reasons. Occlusion queries are used while rendering, and the CPU can read the results of the queries in a later frame. We keep track of how many frames the object was not visible, and if it was not visible for a certain amount, we omit it from rendering. If it suddenly becomes visible later, we immediately enable rendering it again. This technique means that results will lag behind for a few frames (latency between cpu and gpu and latency of using previous frame's depth buffer). These are implemented in the functions wi::renderer::OcclusionCulling_Render() and wi::renderer::OcclusionCulling_Read().

Shadow Maps

The DrawShadowmaps() function will render shadow maps for each active dynamic light that are within the camera frustum. There are two types of shadow maps, 2D and Cube shadow maps. The maximum number of usable shadow maps are set up with calling SetShadowProps2D() or SetShadowPropsCube() functions, where the parameters will specify the maximum number of shadow maps and resolution. The shadow slots for each light must be already assigned, because this is a rendering function and is not allowed to modify the state of the Scene and lights. The shadow slots will be set up in the UpdatePerFrameData() function that is called every frame by the RenderPath3D.

UpdatePerFrameData

This function prepares the scene for rendering. It must be called once every frame. It will modify the Scene and other rendering related resources. It is called after the Scene was updated. It performs frustum culling and other management tasks, such as packing decal rects into atlas and several other things.

UpdateRenderData

Begin rendering the frame on GPU. This means that GPU compute jobs are kicked, such as particle simulations, texture packing, mipmap generation tasks that were queued up, updating per frame GPU buffer data, animation vertex skinning and other things.

Ray tracing (hardware accelerated)

Hardware accelerated ray tracing API is available, so a variety of renderer features are available using that. If the hardware support is available, the Scene will allocate a top level acceleration structure, and the meshes will allocate bottom level acceleration structures for themselves if it's requested by some effect. Requesting is possible by using wi::scene::Scene::SetAccelerationStructureUpdateRequested(true). Updating acceleration structures is done by calling wi::renderer::UpdateRaytracingAccelerationStructures(cmd), which can be called on the async compute or graphics queue. The updates will happen on the GPU timeline, so provide a CommandList as argument. The top level acceleration structure will be rebuilt from scratch. The bottom level acceleration structures will be rebuilt from scratch once, and then they will be updated (refitted).

After the acceleration structures are updated, ray tracing shaders can use it after binding to a shader resource slot.

Ray tracing (compute)

If hardware accelerated ray tracing is not available, certain effects will fall back to software ray tracing implementation using compute resources. The following effects support software ray tracing implementation:

  • path tracing
  • lightmap baking
  • DDGI
  • Surfel GI The following do not support software ray tracing at the moment for practical reason (because it would be too slow to be useful in real time):
  • ray traced shadows
  • ray traced ambient occlusion
  • ray traced reflections
  • ray traced diffuse

Path tracing

The wi::renderer::RayTraceScene() function will render the scene with path tracing from the camera view that was bound using wi::renderer::BindCameraCB() for the current CommandList. The result will be written to a texture that is provided as parameter. You also provide the sample index parameter, since path tracing will render only 1 sample in this call, but for sample index greater than 0, it will blend results into previous result texture. The texture must have been created with UNORDERED_ACCESS bind flags, because it will be written in compute shaders. The scene BVH structure must have been already built to use this, it can be accomplished by calling wi::renderer::UpdateRaytracingAccelerationStructures(). The RenderPath3D_Pathracing uses this ray tracing functionality to render a path traced scene. The RayTraceScene() also have optional texture parameters that can be filled to aid when using denoiser, or compositing with rasterized effects. For further details, check the RenderPath3D_Pathracing implementation.

Other than path tracing, the scene BVH can be rendered by using the RayTraceSceneBVH function. This will render the bounding box hierarchy to the screen as a heatmap. Blue colors mean that few boxes were hit per pixel, and with more bounding box hits the colors go to green, yellow, red, and finally white. This is useful to determine how expensive the scene is with regards to ray tracing performance.

Lightmap baking

The lightmap baking is also using ray tracing on the GPU to precompute static lighting for objects in the scene. Objects that contain lightmap atlas texture coordinate set can start lightmap rendering by setting ObjectComponent::SetLightmapRenderRequest(true). Every object that have this flag set will have their lightmaps updated by performing ray traced rendering by the wi::renderer internally every frame. Every frame only renders 1 sample, so results will be updated in real time, and you will see how the result accumulates over time. You can stop accumulation whenever you feel it converged enough. Lightmap texture coordinates can be generated by a separate tool, such as the Wicked Engine Editor application. The lightmaps can be optionally compressed in BC6H format to save memory and improve performance. The compression will be performed only after lightmap rendering is stopped, so the compression will not interfere with lightmap accumulation. If you are using a denoiser, it will be performed after lightmap baking is stopped. To denoise lightmaps, follow the same steps as the path tracer denoiser setup described in the Denoiser section.

Scene BVH

The scene BVH can be rebuilt from scratch using the wi::renderer::UpdateRaytracingAccelerationStructures() function. The ray tracing features require the scene BVH to be built before using them. This is using the wiGPUBVH facility to build the BVH using compute shaders running on the GPU when hardware ray tracing is not available. Otherwise it uses ray tracing acceleration structure API objects.

Decals

The DecalComponents in the scene can be rendered differently based on the rendering technique that is being used. The two main rendering techinques are forward and deferred rendering.

Forward rendering is aimed at low bandwidth consumption, so the scene lighting is computed in one rendering pass, right when rendering the scene objects to the screen. In this case, all of the decals are also processed in the object rendering shaders automatically. In this case a decal atlas will need to be generated, because all decal textures must be available to sample in a single shader, but this is handled automatically inside the UpdateRenderData function.

Deferred rendering techniques are outputting G-buffers, and compute lighting in a separate pass that is decoupled from the object rendering pass. Decals are rendered one by one on top of the albedo buffer using the function wi::renderer::DrawDeferredDecals(). All the lighting will be computed on top after the decals were rendered to the albedo in a separate render pass to complete the final scene.

Environment probes

EnvironmentProbeComponents can be placed into the scene and if they are tagged as invalid using the EnvironmentProbeComponent::SetDirty(true) flag, they will be rendered to a cubemap array that is accessible in all shaders. They will also be rendered in real time every frame if the EnvironmentProbeComponent::SetRealTime(true) flag is set. The cubemaps of these contain indirect specular information from a point source that can be used as approximate reflections for close by objects.

The probes also must have TransformComponent associated to the same entity. This will be used to position the probes in the scene, but also to scale and rotate them. The scale/position/rotation defines an oriented bounding box (OBB) that will be used to perform parallax corrected environment mapping. This means that reflections inside the box volume will not appear as infinitely distant, but constrained inside the volume, to achieve more believable illusion.

Post processing

There are several post processing effects implemented in the wi::renderer. They are all implemented in a single function with their name starting with Postprocess_, like for example: wi::renderer::Postprocess_SSAO(). They are aimed to be stateless functions, with their function signature clearly indicating input-output parameters and resources.

The chain/order of post processes is not implemented here, only the single effects themselves. The order of post processes is more of a user responsibility, so it should be implemented in the High Level Interface. For reference, a default post-process chain is implemented in RenderPath3D::RenderPostProcessChain() function.

Instancing

Instanced rendering will be always used automatically when rendering objects. This means, that ObjectComponents that share the same mesh will be batched together into a single draw call, even if they have different transformations, colors, or dithering parameters. This can greatly reduce CPU overhead when lots of objects are duplicated (for example trees, rocks, props).

ObjectComponents can override stencil settings for the material. In case multiple ObjectComponents share the same mesh, but some are using different stencil overrides, those could be removed from the instanced batch, so there is some overhead to modifying stencil overrides for objects.

Tip: to find out more about how instancing is used to batch objects together, take a look at the RenderMeshes() function inside wi::renderer.cpp

Stencil

If a depth stencil buffer is bound when using DrawScene(), the stencil buffer will be written. The stencil is a 8-bit mask that can be used to achieve different kinds of screen space effects for different objects. MaterialComponents and ObjectComponents have the ability to set the stencil value that will be rendered. The 8-bit stencil value is separated into two parts:

  • The first 4 bits are reseved for engine-specific values, such as to separate skin, sky, common materials from each other. these values are contained in wiEnums in the STENCILREF enum.
  • The last 4 bits are reserved for user stencil values. The application can decide what it wants to use these for.

Please use the wi::renderer::CombineStencilrefs() function to specify stencil masks in a future-proof way.

The Pipeline States have a DepthStencilState type member which will control how the stencil mask is used in the graphics pipeline for any custom rendering effect. The functionality is supposed to be equivalent and closely matching of what DirectX 11 provides, so for further information, refer to the DirectX 11 documentation of Depth Stencil States.

Loading Shaders

While the GraphicsDevice is responsible of creating shaders and pipeline states, loading the shader files themselves are not handled by the graphics device. The wi::renderer::LoadShader(ShaderStage::) is a helper function that provides this feature. This is internally loading shaders from a common shader path, that is by default the "../WickedEngine/shaders" directory (relative to the application working directory), so the filename provided to this function must be relative to that path. Every system in the engine that loads shaders uses this function to load shaders from the same folder, which makes it very easy to reload shaders on demand with the wi::renderer::ReloadShaders() function. This is useful for when the developer modifies a shader and recompiles it, the engine can reload it while the application is running. The developer can modify the common shader path with the wi::renderer::SetShaderPath() to any path of preference. The developer is free to load shaders with a custom loader, from any path too, but the wi::renderer::ReloadShaders() functionality might not work in that case for those shaders.

Debug Draw

Debug geometry can be rendered by calling the wi::renderer::DrawDebugWorld() function and setting up debug geometries, or enabling debug features. The DrawDebugWorld() is already called by RenderPath3D, so the developer can simply just worry about configure debug drawing features and add debug geometries and drawing will happen at the right place (if the developer decided to use RenderPath3D in their application).

The user provided debug geometry features at present are:

  • DrawBox(): provide a transformation matrix and color to render a wireframe box mesh with the desired transformation and color. The box will be only rendered for a single frame.
  • DrawLine(): provide line segment begin, end positions and a color value. The line will be only rendered for a single frame.
  • DrawPoint(): provide a position, size and color to render a point as a "X" line geometry that is always camera facing. The point will be only rendered for a single frame.
  • DrawTriangle(): provide 3 positions and colors that represent a colored triangle. There is an option to draw the triangle in wireframe mode instead of solid. The triangle will be only rendered for a single frame.
  • DrawSphere(): provide a sphere and a color, its bounds will be drawn as line geometry. The sphere will be only rendered for a single frame.
  • DrawCapsule(): provide a capsule and a color, its bounds will be drawn as line geometry. The capsule will be only rendered for a single frame.

Configuring other debug rendering functionality:

  • SetToDrawDebugBoneLines(): Bones will be rendered as lines
  • SetToDrawDebugPartitionTree(): Scene object bounding boxes will be rendered.
  • SetToDrawDebugEnvProbes(): Environment probes will be rendered as reflective spheres and their affection range as oriented bounding boxes.
  • SetToDrawDebugEmitters(): Emitter mesh geometries will be rendered as wireframe meshes.
  • SetToDrawDebugForceFields(): Force field volumes will be rendered.
  • SetToDrawDebugCameras(): Cameras will be rendered as oriented boxes.

Animation Skinning

MeshComponents that have their armatureID associated with an ArmatureComponent will be skinned inside the wi::renderer::UpdateRenderData() function. This means that their vertex buffer will be animated with compute shaders, and the animated vertex buffer will be used throughout the rest of the frame.

Custom Shaders

Apart from the built in material shaders, the developer can create a library of custom shaders from the application side and assign them to materials. The wi::renderer::RegisterCustomShader() function is used to register a custom shader from the application. The function returns the ID of the custom shader that can be input to the MaterialComponent::SetCustomShaderID() function.

The CustomShader is essentially the combination of a Pipeline State Object for each RENDERPASS that specifies whether it is to be drawn in a transparent or opaque, or other kind of pass within a RENDERPASS. The developer is responsible of creating a fully valid pipeline state to render a mesh. If a pipeline state is left as empty for a combination of RENDERPASS, then the material will simply be skipped and not rendered in that pass. The other part of CustomShader is a name, that can be used as simple identifier for the user, and a filterMask that will be used to indicate what kind of material this is to multiple systems. You can set the filterMask to any combination of wi::enums::FILTER for your purposes, but you have to include FILTER_OPAQUE or FILTER_TRANSPARENT to indicate if this is going to be rendered in opaque or transparent passes (or both).

The MaterialComponent::userdata can also be used to provide a small amount of custom material data that will be available to shaders. You can access that in shaders from ShaderMaterial::userdata. If the built in user data is not enough for your purposes, you can create additional GPUBuffer objects and send descriptor indices in user data to access extended user data indirectly.

To look at an example, take a look at the built in Hologram custom shader and see how exactly to create a valid pipeline state, shader, etc.

Enums

[Header] This is a collection of enum values used by the wi::renderer to identify graphics resources. Usually arrays of the same resource type are declared and the XYZENUM_COUNT values tell the length of the array. The other XYZENUM_VALUE represents a single element within that array. This makes the code easy to manage, for example:

enum CBTYPE
{
	CBTYPE_MESH, // = 0
	CBTYPE_SOMETHING_ELSE, // = 1
	CBTYPE_AN_OTHER_THING, // = 2
	CBTYPE_COUNT // = 3
};
GPUBuffer buffers[CBTYPE_COUNT]; // this example array contains 3 elements
//...
device->BindConstantBuffer(&buffers[CBTYPE_MESH], 0, cmd); // makes it easy to reference an element

This is widely used to make code straight forward and easy to add new objects, without needing to create additional declarations, except for the enum values.

Image Renderer

[Header] [Cpp] This can render images to the screen in a simple manner. You can draw an image like this:

wi::image::SetCanvas(canvas); // setting the canvas area is required to set the drawing area and perform DPI scaling (this is only for the current thread)
wi::image::Draw(myTexture, wiImageParams(10, 20, 256, 128), cmd);

The example will draw a 2D texture image to the position (10, 20), with a size of 256 x 128 pixels to the current render pass. There are a lot of other parameters to customize the rendered image, for more information see wiImageParams structure.

  • wi::image::Params
    Describe all parameters of how and where to draw the image on the screen.

Font Renderer

[Header] [Cpp] This can render fonts to the screen in a simple manner. You can render a font as simple as this:

wi::font::SetCanvas(canvas); // setting the canvas area is required to set the drawing area and perform DPI scaling (this is only for the current thread)
wi::font::Draw("write this!", wiFontParams(10, 20), cmd);

Which will write the text write this! to 10, 20 pixel position onto the screen. There are many other parameters to describe the font's position, size, color, etc. See the wiFontParams structure for more details.

  • wiFontParams
    Describe all parameters of how and where to draw the font on the screen.

The wiFont can load and render .ttf (TrueType) fonts. The default "Liberation Sans" (Arial compatible) font style is embedded into the engine ([liberation_sans.h] file). The developer can load additional fonts from files by using wiFont::AddFontStyle() functions. These can either load from a file, or take a provided byte data for the font. The AddFontStyle() will return an int that will indicate the font ID within the loaded font library. The wiFontParams::style can be set to the font ID to use a specific font that was previously loaded. If the developer added a font before wiFont::Initialize was called, then that will be the default font and the "Liberation Sans" font will not be created.

Emitted Particle System

[Header] [Cpp] GPU driven emitter particle system, used to draw large amount of camera facing quad billboards. Supports simulation with force fields and fluid simulation based on Smooth Particle Hydrodynamics computation.

Hair Particle System

[Header] [Cpp] GPU driven particles that are attached to a mesh surface. It can be used to render vegetation. It participates in force fields simulation.

Ocean

[Header] [Cpp] Ocean renderer using Fast Fourier Transforms simulation. The ocean surface is always rendered relative to the camera, like an infinitely large water body.

Sprite

[Header] [Cpp] A helper facility to render and animate images. It uses the wiImage renderer internally

  • Anim
    Several different simple animation utilities, like animated textures, wobbling, rotation, fade out, etc...

SpriteFont

[Header] [Cpp] A helper facility to render fonts. It uses the wiFont renderer internally. It performs string conversion

TextureHelper

[Header] [Cpp] This is used to generate procedural textures, such as uniform colors, noise, etc...

GPUSortLib

[Header] [Cpp] This is a GPU sorting facility using the Bitonic Sort algorithm. It can be used to sort an index list based on a list of floats as comparison keys entirely on the GPU.

GPUBVH

[Header] [Cpp] This facility can generate a BVH (Bounding Volume Hierarcy) on the GPU for a Scene. The BVH structure can be used to perform efficient RAY-triangle intersections on the GPU, for example in ray tracing. This is not using the ray tracing API hardware acceleration, but implemented in compute, so it has wide hardware support.

GUI

The custom GUI, implemented with engine features

GUI

[Header] [Cpp] The wiGUI is responsible to run a GUI interface and manage widgets.

GUI Scaling: To ensure correct GUI scaling, GUI elements should be designed for the current window size. If they are placed inside RenderPath2D::ResizeLayout() function according to current screen size, it will ensure that GUI will be scaled on a Resolution or DPI change event, which is recommended.

EventArgs

This will be sent to widget callbacks to provide event arguments in different formats

wiWidget

The base widget interface can be extended with specific functionality. The GUI will store and process widgets by this interface.

Button

A simple clickable button. Supports the OnClick event callback.

Label

A simple static text field.

TextInputField

Supports text input when activated. Pressing Enter will accept the input and fire the OnInputAccepted callback. Pressing the Escape key while active will cancel the text input and restore the previous state. There can be only one active text input field at a time.

Slider

A slider, that can represent float or integer values. The slider also accepts text input to input number values. If the number input is outside of the slider's range, it will expand its range to support the newly assigned value. Upon changing the slider value, whether by text input or sliding, the OnSlide event callback will be fired.

CheckBox

The checkbox is a two state item which can represent true/false state. The OnClick event callback will be fired upon changing the value (by clicking on it)

ComboBox

Supports item selection from a list of text. Can set the maximum number of visible items. If the list of items is greater than the max allowed visible items, then a vertical scrollbar will be presented to allow showing hidden items. Upon selection, the OnSelect event callback will be triggered.

Window

A window widget is able to hold any number of other widgets. It can be moved across the screen, minimized and resized by the user. The window does not manage lifetime of attached widgets since 0.49.0!

ColorPicker

Supports picking a HSV (or HSL) color, displaying its RGB values. On selection, the OnColorChanged event callback will be fired and the user can read the new RGB value from the event argument.

Helpers

A collection of engine-level helper classes

Archive

[Header] [Cpp] This is used for serializing binary data to disk or memory. An archive file always starts with the 64-bit version number that it was serialized with. An archive of greater version number than the current archive version of the engine can't be opened safely, so an error message will be shown if this happens. A certain archive version will not be forward compatible with the current engine version if the current archive version barrier number is greater than the archive's own version number.

Color

[Header] Utility to convert to/from float color data to 32-bit RGBA data (stored in a uint32_t as RGBA, where each channel is 8 bits)

FadeManager

[Header] [Cpp] Simple helper to manage a fadeout screen. Fadeout starts at transparent, then fades smoothly to an opaque color (such as black, in most cases), then a callback occurs which the user can handle with their own event. After that, the color will fade back to transperent. This is used by the Application to fade from one RenderPath to an other.

Helper

[Header] [Cpp] Many helper utility functions, like screenshot, readfile, messagebox, splitpath, sleep, etc...

Primitive

[Header] [Cpp] Primitives that can be intersected with each other

AABB

[Header] [Cpp] Axis aligned bounding box. There are multiple ways to construct it, for example from min-max corners, or center and halfextent.

Sphere

[Header] [Cpp] Sphere with a center and radius.

Capsule

[Header] [Cpp] It's like two spheres connected by a cylinder. Base and Tip are the two endpoints, radius is the cylinder's radius.

Ray

[Header] [Cpp] Line with a starting point (origin) and direction. The direction's reciprocal is precomputed to perform fast intersection of many primitives with one ray.

Frustum

[Header] [Cpp] Six planes, most commonly used for checking if an intersectable primitive is inside a camera.

Hitbox2D

[Header] [Cpp] A rectangle, essentially an 2D AABB.

Math

[Header] [Cpp] Math related helper functions, like lerp, triangleArea, HueToRGB, etc...

Random

[Header] [Cpp] Uniform random number generator with a good distribution.

RectPacker

[Header] [Cpp] Provides the ability to pack multiple rectangles into a bigger rectangle, while taking up the least amount of space from the containing rectangle.

ResourceManager

[Header] [Cpp] This can load images and sounds. It will hold on to resources until there is at least something that is referencing them, otherwise deletes them. One resource can have multiple owners, too. This is thread safe.

  • Load() : Load a resource, or return a resource handle if it already exists. The resources are identified by file names. The user can specify import flags (optional). The user can provide a file data buffer that was loaded externally (optional). This function will return a resource handle. The resource handle equals to nullptr if it was not loaded successfully, otherwise a valid handle is returned.
  • Contains() : Check whether a resource exists or not.
  • Clear() : Clear all resources. This will clear the resource library, but resources that are still used somewhere will remain usable.

The resource manager can support different modes that can be set with SetMode(MODE param) function:

  • DISCARD_FILEDATA_AFTER_LOAD : this is the default behaviour. The resource will not hold on to file data, even if the user specified IMPORT_RETAIN_FILEDATA flag when loading the resource. This will result in the resource manager unable to serialize (save) itself.
  • ALLOW_RETAIN_FILEDATA : this mode can be used to keep the file data buffers alive inside resources. This way the resource manager can be serialized (saved). Only the resources that still hold onto their file data will be serialized (saved). When loading a resource, the user can specify IMPORT_RETAIN_FILEDATA flag to keep the file data for a specific resource instead of discarding it.
  • ALLOW_RETAIN_FILEDATA_BUT_DISABLE_EMBEDDING : Keep all file data, but don't write them while serializing. This is useful to disable resource embedding temporarily without destroying file data buffers.

The resource manager can always be serialized in read mode. File data retention will be based on existing file import flags and the global resource manager mode.

SpinLock

[Header] [Cpp] This can be used to guarantee exclusive access to a block in multithreaded race condition scenario instead of a mutex. The difference to a mutex that this doesn't let the thread to yield, but instead spin on an atomic flag until the spinlock can be locked.

Arguments

[Header] [Cpp] This is to store the startup parameters that were passed to the application from the operating system "command line". The user can query these arguments by name.

Timer

[Header] [Cpp] High resolution stopwatch timer

wiVoxelGrid

[Header] [Cpp] An axis aligned bounding box volume that contains a 3D grid of voxels. It can be used to voxelize the scene, and used by PathQuery for path finding.

The voxel grid is created with the init() function, providing the dimensions of the 3D grid. This function will allocate the necessary memory to store the grid, and fill every voxel to empty. To simply clear the grid, use cleardata() function, which will not allocate memory, only clear all voxels to empty.

The volume of the grid is specified by the center and voxelSize members. To set the voxelSize, use the set_voxelsize() function instead of directly modifying the voxelSize. You can also align it easily to an existing AABB with the from_aabb() function. Setting these doesn't modify voxel data, so it probably needs to be revoxelized after modifications.

The main feature is that you can inject primitives into the voxel grid, this is the voxelization process. You can inject triangles, AABB, Sphere and Capsule primitives with the inject_triangle(), inject_aabb(), inject_sphere() and the inject_capsule() functions, respectively. All the primitives are input to these functions in world space, and they will be mapped into the voxel grid internally by these functions based on the grid center, voxelSize and resolution. These functions can also accept a bool option called subtract which - if set to true - will modify the voxelization process to remove voxels intead of add them. The voxelization process is performance sensitive, but it can be done from different threads as it is thread safe, using atomic operations to modify voxel data.

To access individual voxels, you can use the check_voxel() function to check if a voxel is empty or validm and the set_voxel() function to set a voxel to empty or valid. These operations are not thread safe! Also these functions will both accept world position, or voxel coordinates. You can additionally convert world positions to voxel coordinates by using the world_to_coord() function, or do the reverse with the coord_to_world() function.

Note: There are helper functions to voxelize a whole object or the whole scene, accessible from the Scene object. These are called VoxelizeObject() and VoxelizeScene().

wiPathQuery

[Header] [Cpp] Path query can find paths from start to goal position within a voxel grid. To run a path finding query, first prepare a voxel grid with scene data, then process it with the path query.

The usage of the path query at minimum is to call the process() function, with start and goal positions and a voxel grid. When the function returns, you can check path finding success with the is_successful() function, and get_next_waypoint() function to find the next waypoint in the path from the start that goes towards the goal.

The path query can be additionally configured with the flying bool param. If you set flying to true, then the path finding will search a path in empty voxels (in the air), otherwise by default it will try to find a path that goes on the ground.

The traversing entity can have a logical size, configured by the agent_height and agent_width parameters, that specify the approximate size of traversing entity in voxels. agent_height will specify how many empty voxels must be above the traversed path at any waypoint. agent_width specifies how many empty voxels must be in the horizontal directions. With these you can specify how far the path should keep away from walls and obstacles, and also to not allow going in between tight openings that the logical size would not allow.

Note: processing a path query can take a long time, depending on how far the goal is from the start. Consider doing multiple path queries on multiple threads, or doing them asynchronously across the frame, the Job System can be used to track completion of asynchronous tasks like this.

Input

[Header] [Cpp] This is the high level input interface. Use this to read input devices in a platform-independent way.

  • Initialize
    Creates all necessary resources, the engine initialization already does this via wiInitializer, so the user probably doesn't have to worry about this.
  • Update
    This is called once per frame and necessary to process inputs. This is automatically called in Application, so the user doesn't need to worry about it, unless they are reimplementing the high level interface for themselves.
  • Down
    Checks whether a button is currently held down.
  • Press
    Checks whether a button was pressed in the current frame.
  • Hold
    Checks whether a button was held down for a specified amount of time. It can be checked whether it was held down exactly for the set amount of time, or at least for that long.
  • GetPointer
    Returns the mouse position. The first two values of the returned vector are mouse pointer coordinates in window space. The third value is the relative scroll button difference since last frame.
  • SetPointer
    Sets the mouse pointer position in window-relative coordinates.
  • HidePointer
    Hides or shows the mouse pointer.
  • GetAnalog
    Returns the gamepad's analog position.
  • SetControllerFeedback
    Utilize various features of the gamepad, such as vibration, or LED color.

BUTTON

This collection of enums containing all the usable digital buttons, either on keyboard or a controller. If you wish to specify a letter key on the keyboard, you can use the plain character value and cast to this enum, like:

bool is_P_key_pressed = wiInput::Press((wiInput::BUTTON)'P');

KeyboardState

The keyboard's full state, where each button's state is stored as either pressed or released. Can be queried via wiRawInput

MouseState

The mouse pointer's absolute and relative position. Can be queried via wiRawInput

ControllerState

The controller's button and analog states. Can be queried via wiRawInput or wiXInput

ControllerFeedback

The controller force feedback features that might be supported. Can be provided to wiRawInput or wiXInput

Touch

A touch contact point. Currently it is supported in UWP (Universal Windows Platform) applications.

  • GetTouches
    Get a vector containing current Touch contact points

XInput

[Header] [Cpp] Low level wrapper for XInput API (capable of handling Xbox controllers). This functionality is used by the more generic wiInput interface, so the user probably doesn't need to use this.

RawInput

[Header] [Cpp] Low level wrapper for RAWInput API (capable of handling human interface devices, such as mouse, keyboard, controllers). This functionality is used by the more generic wiInput interface, so the user probably doesn't need to use this.

Audio

[Header] [Cpp] The namespace that is a collection of audio related functionality.

  • CreateSound
  • CreateSoundInstance
  • Play
  • Pause
  • Stop
  • SetVolume
  • GetVolume
  • ExitLoop
  • SetSubmixVolume
  • GetSubmixVolume
  • Update3D
  • SetReverb

Sound

Represents a sound file in memory. Load a sound file via wiAudio interface.

SoundInstance

An instance of a sound file that can be played and controlled in various ways through the wiAudio interface. (To ensure looped playback of its Sound, SoundInstance::SetLooped(true) must be called before audio::CreateSoundInstance, subsequent SetLooped calls have no effect even if occurring before calling Play.)

SoundInstance3D

This structure describes a relation between listener and sound emitter in 3D space. Used together with a SoundInstance in wiAudio::Update3D() function

SUBMIX_TYPE

Groups sounds so that different properties can be set for a whole group, such as volume for example

REVERB_PRESET

Can make different sounding 3D reverb effect globally

  • REVERB_PRESET_DEFAULT
  • REVERB_PRESET_GENERIC
  • REVERB_PRESET_FOREST
  • REVERB_PRESET_PADDEDCELL
  • REVERB_PRESET_ROOM
  • REVERB_PRESET_BATHROOM
  • REVERB_PRESET_LIVINGROOM
  • REVERB_PRESET_STONEROOM
  • REVERB_PRESET_AUDITORIUM
  • REVERB_PRESET_CONCERTHALL
  • REVERB_PRESET_CAVE
  • REVERB_PRESET_ARENA
  • REVERB_PRESET_HANGAR
  • REVERB_PRESET_CARPETEDHALLWAY
  • REVERB_PRESET_HALLWAY
  • REVERB_PRESET_STONECORRIDOR
  • REVERB_PRESET_ALLEY
  • REVERB_PRESET_CITY
  • REVERB_PRESET_MOUNTAINS
  • REVERB_PRESET_QUARRY
  • REVERB_PRESET_PLAIN
  • REVERB_PRESET_PARKINGLOT
  • REVERB_PRESET_SEWERPIPE
  • REVERB_PRESET_UNDERWATER
  • REVERB_PRESET_SMALLROOM
  • REVERB_PRESET_MEDIUMROOM
  • REVERB_PRESET_LARGEROOM
  • REVERB_PRESET_MEDIUMHALL
  • REVERB_PRESET_LARGEHALL
  • REVERB_PRESET_PLATE

Physics

[Header] [Cpp] You can find the physics system related functionality under ENGINE/Physics filter in the solution. It uses the entity-component system to perform updating all physics components in the world.

  • Initialize
    This must be called before using the physics system, but it is automatically done by wiInitializer
  • IsEnabled
    Is the physics system running?
  • SetEnabled
    Enable or disable physics system
  • RunPhysicsUpdateSystem
    Run physics simulation on input components.

Rigid Body Physics

Rigid body simulation requires RigidBodyPhysicsComponent for entities and TransformComponent. It will modify TransformComponents with physics simulation data, so after simulation, TransformComponents will contain absolute world matrix.

Kinematic rigid bodies are those that are not simulated by the physics system, but they will affect the rest of the simulation. For example, an animated kinematic rigid body will strictly follow the animation, but affect any other rigid bodies it comes in contact with. Set a rigid body to kinematic by having the RigidBodyPhysicsComponent::KINEMATIC flag in the RigidBodyPhysicsComponent::_flags bitmask.

Deactivation happens after a while for rigid bodies that participated in the simulation for a while, this means after that they will no longer participate in the simulation. This behaviour can be disabled with the RigidBodyPhysicsComponent::DISABLE_DEACTIVATION flag.

There are multiple Collision Shapes of rigid bodies that can be used:

  • BOX: simple oriented bounding box, fast.
  • SPHERE: simple sphere, fast.
  • CAPSULE: Two spheres with a cylinder between them.
  • CONVEX_HULL: A simplified mesh.
  • TRIANGLE_MESH: The original mesh. It is always kinematic.

Soft Body Physics

Soft body simulation requires SoftBodyPhysicsComponent for entities as well as MeshComponent. When creating a soft body, the simulation mesh will be computed from the MeshComponent vertices and mapping tables from physics to graphics indices that associate graphics vertices with physics vertices. The physics vertices will be simulated in world space and copied to the SoftBodyPhysicsComponent::vertex_positions_simulation array as graphics vertices. This array can be uploaded as vertex buffer as is.

The ObjectComponent's transform matrix is used as a manipulation matrix to the soft body, if the ObjectComponent's mesh has a soft body associated with it. The manipulation matrix means, that pinned soft body physics vertices will receive that transformation matrix. The simulated vertices will follow those manipulated vertices.

Pinning the soft body vertices means that those physics vertices have zero weights, so they are not simulated, but instead drive the simulation, as in other vertices will follow them, a bit like how kinematic rigid bodies manipulate simulated rigid bodies.

The pinned vertices can also be manipulated via skinning animation. If the soft body mesh is skinned and an animation is playing, then the pinned vertices will follow the animation.

Network

[Header] [Cpp] Simple interface that provides UDP networking features.

  • Initialize
  • CreateSocket
  • Send
  • ListenPort
  • CanReceive
  • Receive

Socket

This is a handle that must be created in order to send or receive data. It identifies the sender/recipient.

Connection

An IP address and a port number that identifies the target of communication

Scripting

This is the place for the Lua scipt interface. For a complete reference about Lua scripting interface, please see the ScriptingAPI-Documentation

LuaBindings

The systems that are bound to Lua have the name of the system as their filename, postfixed by _BindLua.

Lua

[Header] [Cpp] The Lua scripting interface on the C++ side. This allows to execute lua commands from the C++ side and manipulate the lua stack, such as pushing values to lua and getting values from lua, among other things.

Lua_Globals

[Header] Hardcoded lua script in text format. This will be always executed and provides some commonly used helper functionality for lua scripts.

Luna

[Header] Helper to allow bind engine classes from C++ to Lua

Tools

This is the place for tools that use engine-level systems

Backlog

[Header] [Cpp] Used to log any messages by any system, from any thread. It can draw itself to the screen. It can execute Lua scripts. If there was a wii:backlog::LogLevel::Error or higher severity message posted on the backlog, the contents of the log will be saved to the temporary user directory as wiBacklog.txt.

Profiler

[Header] [Cpp] Used to time specific ranges in execution. Support CPU and GPU timing. Can write the result to the screen as simple text at this time.

Shaders

Shaders are written in HLSL shading language and compiled into the native shader binary format that can vary based on platform and graphics device requirements.

There are some macros used to declare resources with binding slots that can be read from the C++ application via code sharing. These macros are used to declare resources:

  • CBUFFER(name, slot)
    Declares a constant buffer
CBUFFER(myCbuffer, 0);
{
	float4 values0;
	uint4 values2;
};

// Load as if reading a global value
float loaded_value = values0.y;

Note: When creating constant buffers, the structure must be strictly padded to 16-byte boundaries according to HLSL rules. Apart from that, the following limitation is in effect for Vulkan compatibility:

Incorrect padding to 16-byte:

CBUFFER(cbuf, 0)
{
	float padding;
	float3 value; // <- the larger type can start a new 16-byte slot, so the C++ and shader side structures could mismatch
	float4 value2;
};

Correct padding:

CBUFFER(cbuf, 0)
{
	float3 value; // <- the larger type must be placed before the padding
	float padding;
	float4 value2;
};

Interop

The interop between shaders and C++ code is handled by shared header (.h) files. Modifying the shared header files will need recompilation of both engine and shaders, otherwise undefined behaviour will occur. For cases when only shaders need visibility, consider using shader header files (.hlsli), which should not be shared with the engine C++ code.

[Header] Shader Interop is used for declaring shared structures or values between C++ Engine code and shader code. There are several ShaderInterop files, postfixed by the subsystem they are used for to keep them minimal and more readable:
ShaderInterop_BVH.h
ShaderInterop_DDGI.h
ShaderInterop_EmittedParticle.h
ShaderInterop_FFTGenerator.h
ShaderInterop_Font.h
ShaderInterop_FSR2.h
ShaderInterop_GPUSortLib.h
ShaderInterop_HairParticle.h
ShaderInterop_Image.h
ShaderInterop_Ocean.h
ShaderInterop_Postprocess.h
ShaderInterop_Raytracing.h
ShaderInterop_Renderer.h
ShaderInterop_SurfelGI.h
ShaderInterop_Terrain.h
ShaderInterop_VoxelGrid.h
ShaderInterop_VXGI.h
ShaderInterop_Weather.h

The ShaderInterop also contains the resource macros to help share code between C++ and HLSL, and portability between shader compilers. Read more about macros in the Shaders section

Shader Compiler

The built in Visual Studio shader compiler can compile HLSL shaders with the usual visual studio build process. This can be used to some extent, but the recommended way is to use the [Wicked Engine Shader Compiler Interface (wiShaderCompiler)]. This supports various tools to compile shaders to different shader formats (hlsl5, hlsl6, spirv), and can be used at runtime, for example to compile shader permutations easily, and also use engine features like the job system to parallelize compile tasks with more control than a common build system.

Offline Shader Compilation: The OfflineShaderCompiler tool can be built and used to compile shaders in a command line process. It can also be used to generate a shader dump, which is a header file that can be included into C++ code and compiled, so all shaders will be embedded into the executable, this way they won't be loaded as separate files by applications. However, the shader reload feature will not work in this case for those shaders that are embedded. The shader dump will be contained in wiShaderDump.h file when generated by the offline shader compiler using the shaderdump command line argument. If this file is detected by the time the engine is compiled, shaders will be embedded inside the compiled executable. The offline shader compiler can also be used to compile shader normally into separate .cso files with .wishadermeta metadata files that will be used to detect when each shader needs to be rebuilt automatically.