DirectX 11 顶点和像素着色器如何工作

标签 directx shader directx-11 vertex-shader

我正在阅读“Direct3D 11 的实用渲染和计算”一书并查看 DirectX 11 示例:

中国共产党: https://github.com/walbourn/directx-sdk-samples/blob/master/Direct3D11Tutorials/Tutorial02/Tutorial02.cpp

HLSL: https://github.com/walbourn/directx-sdk-samples/blob/master/Direct3D11Tutorials/Tutorial02/Tutorial02.fx

这是什么?

D3D11_INPUT_ELEMENT_DESC layout[] =
 {
  { "POSITION", 0, DXGI_FORMAT_R32G32B32_FLOAT, 0, 0, D3D11_INPUT_PER_VERTEX_DATA, 0 },
 };

更重要的是着色器——它在程序中的什么地方设置了这个“Pos”?

float4 VS( float4 Pos : POSITION ) : SV_POSITION
{
    return Pos;
}

我只能看到您通过以下方式绑定(bind)顶点数据:

// Create vertex buffer
SimpleVertex vertices[] =
{
    XMFLOAT3( 0.0f, 0.5f, 0.5f ),
    XMFLOAT3( 0.5f, -0.5f, 0.5f ),
    XMFLOAT3( -0.5f, -0.5f, 0.5f ),
};
D3D11_BUFFER_DESC bd;
ZeroMemory( &bd, sizeof(bd) );
bd.Usage = D3D11_USAGE_DEFAULT;
bd.ByteWidth = sizeof( SimpleVertex ) * 3;
bd.BindFlags = D3D11_BIND_VERTEX_BUFFER;
bd.CPUAccessFlags = 0;
D3D11_SUBRESOURCE_DATA InitData;
ZeroMemory( &InitData, sizeof(InitData) );
InitData.pSysMem = vertices;
hr = g_pd3dDevice->CreateBuffer( &bd, &InitData, &g_pVertexBuffer );
if( FAILED( hr ) )
    return hr;

// Set vertex buffer
UINT stride = sizeof( SimpleVertex );
UINT offset = 0;
g_pImmediateContext->IASetVertexBuffers( 0, 1, &g_pVertexBuffer, &stride, &offset );

// Set primitive topology
g_pImmediateContext->IASetPrimitiveTopology( D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST );

然后设置着色器并绘制 3 个顶点:

// Render a triangle
g_pImmediateContext->VSSetShader( g_pVertexShader, nullptr, 0 );
g_pImmediateContext->PSSetShader( g_pPixelShader, nullptr, 0 );
g_pImmediateContext->Draw( 3, 0 );

这一切都是有道理的。但我对这个 Pos 输入是什么、它是如何生成的以及为什么使用它感到困惑?此外,还有其他示例做同样的事情......(示例 7)他们使用位置和纹理,但我看不出这些信息来自哪里。例如……

struct VS_INPUT
{
  float4 Pos : POSITION;
  float2 Tex : TEXCOORD0;
};

感谢您的宝贵时间!

最佳答案

D3D11_INPUT_ELEMENT_DESC 的数组确定将从顶点缓冲区读取的数据布局。此数组被传递给 ID3D11Device::CreateInputLayout 调用,以便可以使用它。然后通过调用 ID3D11DeviceContext::IASetInputLayout 在渲染上下文中设置它(未显示,但在您链接的代码中)。

如果您查看 D3D11_INPUT_ELEMENT_DESC 的成员,第一个成员是 SemanticName(设置为 "POSITION"),这是顶点缓冲区的元素与着色器中的输入匹配的方式。它还包含有关数据偏移量和格式的数据(在本例中,偏移量为零,格式 DXGI_FORMAT_R32G32B32_FLOAT 对应于 3x 32 位 float )。您会注意到在着色器中,在 Pos 之后有一个冒号,然后是 "POSITION" - 这是语义,并且与输入布局相匹配。然后着色器知道它应该从偏移零开始读取位置数据,以及该数据的格式。

数据本身是从顶点缓冲区中读取的,使用 ID3D11DeviceContext::IASetVertexBuffers 调用设置。 stride 参数告诉输入汇编程序顶点缓冲区中从一个顶点到下一个顶点的距离,因此它知道从哪里开始读取下一个顶点。传入的 g_pVertexBuffer 是从 vertices 数据初始化的,如您所见,它是一组 3x 32 位 float 。

对于具有额外输入(例如 TEXCOORD0)的着色器,D3D11_INPUT_ELEMENT_DESC 数组中将有更多条目,对应于着色器中使用的语义。

关于DirectX 11 顶点和像素着色器如何工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46492001/

相关文章:

c++ - directX 11 比例纹理 2D

c++ - 在浏览器中运行 DX11

javascript - 提高 threejs/shader 几何体的 UV 线质量

opengl - Cocoa 和 OpenGL,如何使用数组设置 GLSL 顶点属性?

c++ - 结合 C++ 和 C

c++ - 在我的命名空间中隐藏 DirectX 命名空间

c++ - 矩阵乘法优化(使用 Neon simd 与着色器)

c++ - 如何从 Direct3D 11 中的 CPU 访问帧缓冲区?

c++ - 在 ID3DBlob 中初始化 PS 的 DirectX11 CreatePixelShader 中出现访问冲突

c++ - 在 firemonkey 中评估 Canvas 渲染的视频卡性能