Diligent Graphics > Diligent Engine > Samples and Tutorials > Asteroids
Asteroids
This sample is designed to be a performance benchmark and is based on this demo developed by Intel. It renders 50,000 unique textured asteroids. Every asteroid is a combination of one of 1000 unique meshes and one of 10 unique textures. The sample uses original D3D11 and D3D12 native implementations, and adds implementation using Diligent Engine API to allow comparing performance of different rendering modes.
Building and Running the Sample
The full sample source code is available at GitHub. To build the sample, download the engine, and open EngineAll.sln solution in build\Win32 folder. Asteroids demo only supports x64 platform. Set Asteroids project as the startup project and run it.
Controls
The sample uses the following keys:
- ‘1’ – switch to native D3D11 mode
- ‘2’ – switch to native D3D12 mode
- ‘3’ – switch to Diligent D3D11 mode
- ‘4’ – switch to Diligent D3D12 mode
- ‘M’ – toggle multithreaded rendering
- ‘+’ – increment number of threads
- ‘-‘ – decrement number of threads
These fights tadalafil professional create gaps between relationships and sometimes create huge misunderstandings. Nuts: Almonds are cialis generico uk very excellent sources of fat and Zinc. Most of the men in the age group of 50+ find difficulty in achieving an erection and maintaining it and in some cases, vision changes that can purchase viagra online turn the world a shade of blue. (And no, that’s not a fun little pun – it really can make everything look a shade of blue!) And any problems with overall health, especially with the products involved. DHT, which stands for dihydro-testosterone also know as the “hair killer” contributes largely to pattern baldness in men alone; minoxidil on the other hand is way too Cheap – the buy viagra in stores is definitely supposed to be cheaper than the branded one.
Implementation Details
Main loop of the sample consists of two parts: update and render. The fist part performs simulation of asteroid movement and is not related to graphics API. The second part is where all API-specific commands are executed. The application displays total frame time, update time and render time in the title bar:
The application is intentionally CPU-bound as its goal is to analyze CPU-side efficiency of different graphics APIs. Most of the time render function spends handling 50,000 asteroids. It also renders background and text sprites, but these steps take negligible fraction of total time and are ignored.
Native D3D11 Implementation
The main part of the native D3D11 render routine first sets all the required states (vertex and index buffers, shaders, rasterizer, blend and depth-stencil states etc.) and then iterates over all 50,000 asteroids. For every object, the function does the following:
- Maps dynamic constant buffer and uploads attributes of the current object
- Sets object’s texture as PS shader resource
- Calls draw command
The asteroid rendering loop is given in the listing below:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
for (UINT drawIdx = 0; drawIdx < NUM_ASTEROIDS; ++drawIdx) { auto staticData = &staticAsteroidData[drawIdx]; auto dynamicData = &dynamicAsteroidData[drawIdx]; D3D11_MAPPED_SUBRESOURCE mapped = {}; ThrowIfFailed(mDeviceCtxt->Map(mDrawConstantBuffer, 0, D3D11_MAP_WRITE_DISCARD, 0, &mapped)); auto drawConstants = (DrawConstantBuffer*) mapped.pData; XMStoreFloat4x4(&drawConstants->mWorld, dynamicData->world); XMStoreFloat4x4(&drawConstants->mViewProjection, viewProjection); drawConstants->mSurfaceColor = staticData->surfaceColor; drawConstants->mDeepColor = staticData->deepColor; mDeviceCtxt->Unmap(mDrawConstantBuffer, 0); mDeviceCtxt->PSSetShaderResources(0, 1, &mTextureSRVs[staticData->textureIndex]); mDeviceCtxt->DrawIndexedInstanced(dynamicData->indexCount, 1, dynamicData->indexStart, staticData->vertexStart, 0); } |
Native D3D12 Implementation
The main part of the native D3D12 render routine uses dynamic resource indexing feature available in D3D12. It lets the shaders dynamically select the resource in run time. For this particular example, the demo selects asteroid texture right in the shader as shown in the code snippet below:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
Texture2DArray<float4> Tex[NUM_UNIQUE_TEXTURES] : register(t0); cbuffer DrawConstantBuffer// : register(b0) { float4x4 mWorld; float4x4 mViewProjection; float4 mSurfaceColor; float4 mDeepColor; uint mTextureIndex; }; float4 asteroid_ps(in float4 position : SV_Position, in VSOut vs_output) : SV_Target { //... float3 detailTex = 0.0f; detailTex += blendWeights.x * Tex[mTextureIndex].Sample(Sampler, coords1).xyz; detailTex += blendWeights.y * Tex[mTextureIndex].Sample(Sampler, coords2).xyz; detailTex += blendWeights.z * Tex[mTextureIndex].Sample(Sampler, coords3).xyz; //... } |
Note that mTextureIndex comes from the constant buffer, which was not allowed in PS5.0 or earlier versions. The render routine sets pipeline state and root signature, binds all textures and then iterates over all 50,000 asteroids. For every object, it sets constant buffer view and runs draw command. The asteroid rendering loop is given in the listing below:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
auto constantsPointer = frame->mDrawConstantBuffersGPUVA + sizeof(DrawConstantBuffer) * drawStart; for (UINT drawIdx = drawStart; drawIdx < drawEnd; ++drawIdx) { auto staticData = &staticAsteroidData[drawIdx]; auto dynamicData = &dynamicAsteroidData[drawIdx]; XMStoreFloat4x4(&drawConstantBuffers[drawIdx].mWorld, dynamicData->world); XMStoreFloat4x4(&drawConstantBuffers[drawIdx].mViewProjection, viewProjection); // Set root cbuffer cmdLst->SetGraphicsRootConstantBufferView(RP_DRAW_CBV, constantsPointer); constantsPointer += sizeof(DrawConstantBuffer); cmdLst->DrawIndexedInstanced(dynamicData->indexCount, 1, dynamicData->indexStart, staticData->vertexStart, 0); } |
Diligent Engine Implementation
As Diligent Engine does not operate with shader registers directly, but uses shader resource binding objects, there are several ways main render loop can be implemented. The sample implements three modes:
- Dynamic. In this mode, asteroid texture is marked as dynamic resource, and there is single SRB object. For every asteroid, its texture is first bound to the shader variable in the SRB object, and SRB resources are committed to the GPU pipeline.
- Mutable. In this mode, asteroid texture is marked as mutable resource, and there are 50,000 SRB objects, one for every asteroid. The texture is bound only once to every SRB object during the initialization.
- Texture Mutable. In this mode, asteroid texture is marked also as mutable resource, but there are only 10 SRB objects, one for every unique texture. The texture is bound only once to every SRB object during the initialization. Also, since there are only 10 SRB objects, they can be first transitioned to correct states before the loop and then committed without COMMIT_SHADER_RESOURCES_FLAG_TRANSITION_RESOURCES flag to avoid unnecessary operations.
The following listing shows the asteroid rendering loop implemented using the Diligent Engine API:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 |
for (UINT drawIdx = startIdx; drawIdx < startIdx+numAsteroids; ++drawIdx) { auto staticData = &staticAsteroidData[drawIdx]; auto dynamicData = &dynamicAsteroidData[drawIdx]; { MapHelper<DrawConstantBuffer> drawConstants(pCtx, mDrawConstantBuffer, MAP_WRITE_DISCARD, 0); XMStoreFloat4x4(&drawConstants->mWorld, dynamicData->world); XMStoreFloat4x4(&drawConstants->mViewProjection, viewProjection); drawConstants->mSurfaceColor = staticData->surfaceColor; drawConstants->mDeepColor = staticData->deepColor; } if( m_BindingMode == BindingMode::Dynamic ) { pVar->Set(mTextureSRVs[staticData->textureIndex]); pCtx->CommitShaderResources(mAsteroidsSRBs[SubsetNum], COMMIT_SHADER_RESOURCES_FLAG_TRANSITION_RESOURCES); } else if( m_BindingMode == BindingMode::Mutable ) { pCtx->CommitShaderResources(mAsteroidsSRBs[drawIdx], COMMIT_SHADER_RESOURCES_FLAG_TRANSITION_RESOURCES); } else if( m_BindingMode == BindingMode::TextureMutable ) { // Resources are transitioned to correct states outside the loop, so no need for // COMMIT_SHADER_RESOURCES_FLAG_TRANSITION_RESOURCES flag pCtx->CommitShaderResources(mAsteroidsSRBs[staticData->textureIndex], 0); } DrawAttribs attribs; attribs.Topology = PRIMITIVE_TOPOLOGY_TRIANGLE_LIST; attribs.IsIndexed = true; attribs.NumIndices = dynamicData->indexCount; attribs.IndexType = VT_UINT16; attribs.FirstIndexLocation = dynamicData->indexStart; attribs.BaseVertex = staticData->vertexStart; pCtx->Draw(attribs); } |
The default mode is Texture Mutable and can be changed by assigning different value to the AsteroidsDE::Asteroids::m_BindingMode variable.