-
Notifications
You must be signed in to change notification settings - Fork 433
EffectPipelineStateDescription
When creating Pipeline State Objects (PSO), DirectX 12 requires the input layout, blend state, depth/stencil state, rasterizer state, and primitive topology along with the RenderTargetState that will be used for rendering. This information is wrapped up by this helper structure which is used by Effects or can be used directly providing your own shaders.
Related tutorial: Simple rendering, 3D shapes
#include "EffectPipelineStateDescription.h"
Typically used during resource creation such as the following, typically used with values from CommonStates:
RenderTargetState rtState(m_deviceResources->GetBackBufferFormat(),
m_deviceResources->GetDepthBufferFormat());
EffectPipelineStateDescription pd(
&InputLayout,
CommonStates::Opaque,
CommonStates::DepthDefault,
CommonStates::CullCounterClockwise,
rtState);
The full parameter list is as follows:
EffectPipelineStateDescription(
const D3D12_INPUT_LAYOUT_DESC* inputLayout,
const D3D12_BLEND_DESC& blend,
const D3D12_DEPTH_STENCIL_DESC& depthStencil,
const D3D12_RASTERIZER_DESC& rasterizer,
const RenderTargetState& renderTarget,
D3D12_PRIMITIVE_TOPOLOGY_TYPE primitiveTopology
= D3D12_PRIMITIVE_TOPOLOGY_TYPE_TRIANGLE,
D3D12_INDEX_BUFFER_STRIP_CUT_VALUE stripCutValue
= D3D12_INDEX_BUFFER_STRIP_CUT_VALUE_DISABLED)
You can set inputLayout to nullptr when the input layout comes from another source (see EffectFactory and Model) or if there's no vertex buffer being bound for rendering.
While generally used by Effects internally, you can use this public method to create your own PSO based on the state description:
ComPtr<ID3D12PipelineState> pso;
// You provide the required root signature for your shaders
ID3D12RootSignature* rootSignature = ...;
D3D12_SHADER_BYTECODE vertexShader = { ... };
D3D12_SHADER_BYTECODE pixelShader= { ... };
pd.CreatePipelineState(
device,
rootSignature,
vertexShader,
pixelShader,
&pso);
The helper method only supports providing the Vertex Shader and Pixel Shader stages as these are the most common ones used by the tool kit. If you need to use other stages like the Geometry Shader, Hull Shader, etc. then use can use ID3D12Device::CreateGraphicsPipelineState
directly and the GetDesc
method to get the basic parameters from the class.
ComPtr<ID3D12PipelineState> pso;
D3D12_GRAPHICS_PIPELINE_STATE_DESC psoDesc = pd.GetDesc();
psoDesc.pRootSignature = rootSignature;
// psoDesc.VS = { ... };
// psoDesc.PS = { ... };
// psoDesc.DS = { ... };
// psoDesc.HS = { ... };
// psoDesc.GS = { ... };
// psoDesc.StreamOutput = { ... };
DX::ThrowIfFailed(device->CreateGraphicsPipelineState(&psoDesc, IID_GRAPHICS_PPV_ARGS(&pso));
For compute shaders, they don't use the data from EffectPipelineStateDescription
:
D3D12_COMPUTE_PIPELINE_STATE_DESC psoDesc = {};
psoDesc.pRootSignature = rootSignature;
psoDesc.CS = { ... };
DX::ThrowIfFailed(device->CreateComputePipelineState(&psoDesc, IID_GRAPHICS_PPV_ARGS(&pso));
For Mesh/Amplifications Shaders (Windows 10 May 2020; Build 19041), you use d3dx12.h
helpers to provide all the state. You can use the data from an EffectPipelineStateDescription
as follows:
#include "d3dx12.h"
ComPtr<ID3D12PipelineState> pso;
D3DX12_MESH_SHADER_PIPELINE_STATE_DESC meshDesc = {};
meshDesc.pRootSignature = rootSignature;
// meshDesc.AS = { ... };
// meshDesc.MS = { ... };
// meshDesc.PS = { ... };
meshDesc.BlendState = pd.blendDesc;
meshDesc.SampleMask = pd.renderTargetState.sampleMask;
meshDesc.RasterizerState = pd.rasterizerDesc;
meshDesc.DepthStencilState = pd.depthStencilDesc;
meshDesc.IBStripCutValue = pd.stripCutValue;
meshDesc.PrimitiveTopologyType = pd.primitiveTopology;
meshDesc.NumRenderTargets = pd.renderTargetState.numRenderTargets;
memcpy_s(meshDesc.RTVFormats, sizeof(meshDesc.RTVFormats), pd.renderTargetState.rtvFormats, sizeof(DXGI_FORMAT) * D3D12_SIMULTANEOUS_RENDER_TARGET_COUNT);
meshDesc.DSVFormat = pd.renderTargetState.dsvFormat;
meshDesc.SampleDesc = pd.renderTargetState.sampleDesc;
meshDesc.NodeMask = pd.renderTargetState.nodeMask;
D3D12_PIPELINE_STATE_STREAM_DESC streamDesc = {};
streamDesc.SizeInBytes = sizeof(meshDesc);
streamDesc.pPipelineStateSubobjectStream = &meshDesc;
DX::ThrowIfFailed(device->CreatePipelineState(&streamDesc, IID_PPV_ARGS(&pso)));
Since the pipeline state description is a large structure, the EffectFactory relies on a hashing function to see if two descriptions are the same.
uint32_t hash = pd.ComputeHash();
This hash is a simple CRC32, and is not cryptographically secure.
Note that the hash currently ignores the input layout.
Creation of resources is fully asynchronous, so you can create many PSOs at the same time.
All content and source code for this package are subject to the terms of the MIT License.
This project has adopted the Microsoft Open Source Code of Conduct. For more information see the Code of Conduct FAQ or contact opencode@microsoft.com with any additional questions or comments.
- Universal Windows Platform apps
- Windows desktop apps
- Windows 11
- Windows 10
- Xbox One
- Xbox Series X|S
- x86
- x64
- ARM64
- Visual Studio 2022
- Visual Studio 2019 (16.11)
- clang/LLVM v12 - v18
- MinGW 12.2, 13.2
- CMake 3.20