本文主要是介绍DirectX11笔记(五)--Direct3D渲染1--VERTICES AND INPUT LAYOUTS,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
- 前言
- 顶点结构与描述
- 创建并绑定输入布局
前言
在前几篇文章中我们主要学习了概念上的渲染管线, 那时我们更加注重说明渲染管线的含义和作用, 现在我们会转换方向, 开始学习使用API来配置渲染管线, 定义shader, 并把几何体提交到渲染管线渲染.
我们将重点学习以下几点:
- 定义, 存储, 渲染图形数据的接口.
- 编写基础的顶点shader和像素shader.
- 学习如何使用渲染状态来配置渲染管线.
- 了解 Effect 框架如何将 shader 和渲染状态封装到一个Render Technique里, 并学习如何把效果框架作为一个shader生成器来使用.
顶点结构与描述
之前我们提到过, 顶点可以是包含除了位置以外的很多数据, 所以为了定义我们自己的顶点, 可以借助结构体来保存我们的顶点信息. 比如下面这两种:
struct Vertex1
{XMFLOAT3 Pos; // 位置XMFLOAT4 Color; // 颜色
};
struct Vertex2
{XMFLOAT3 Pos; // 位置XMFLOAT3 Normal; // 法线XMFLOAT2 Tex0; // 纹理坐标1XMFLOAT2 Tex1; // 纹理坐标2
};
当我们确定顶点格式之后, 我们会通过一个D3D11_INPUT_ELEMENT_DESC类型的数组来描述每一个分量, 这样才能让显卡知道我们传给他的数字要代表怎样的含义. 这个数组也就是输入布局(Input Layout)的描述.
typedef struct D3D11_INPUT_ELEMENT_DESC {LPCSTR SemanticName;UINT SemanticIndex;DXGI_FORMAT Format;UINT InputSlot;UINT AlignedByteOffset;D3D11_INPUT_CLASSIFICATION InputSlotClass;UINT InstanceDataStepRate;
} D3D11_INPUT_ELEMENT_DESC;
- SemanticName和SemanticIndex: 作用如图所示, 他们与顶点shader的输入标记相映射, index的作用主要是在由相同含义的成员时不必重新定义一个名字, 比如上面顶点的两个纹理就可以用index来区分.
- Format: 用来告诉显卡顶点结构中的数字的格式, 是一维的标量还是三维的向量, 是浮点数还是整数.
- InputSlot: 指定元素来源的slot的索引, D3D支持16条slot, 你可以在这16条slot中填充顶点数据, D3D会整合所有的slot来装配一个顶点, 但是在例子中我们都只会使用一条slot.
- AlignedByteOffset: 对于一条slot来说, 他是从顶点结构第一元素开始计算的byte偏移, 还是之前Vertex2的例子, Pos的偏移是0, Pos是一个三维浮点数向量, 每一个float是32bits, 也就是4bytes, 所以Normal的偏移量就是12.
- InputSlotClass和InstanceDataStepRate: 例子中暂时都会指定为 D3D11_INPUT_PER_VERTEX_DATA , 其他的格式会在以后更高级的应用场景里说明使用.
//Vertex1和Vertex2的输入描述
D3D11_INPUT_ELEMENT_DESC desc1[] =
{{"POSITION", 0, DXGI_FORMAT_R32G32B32_FLOAT, 0, 0,D3D11_INPUT_PER_VERTEX_DATA, 0},{"COLOR", 0, DXGI_FORMAT_R32G32B32A32_FLOAT, 0, 12,D3D11_INPUT_PER_VERTEX_DATA, 0}
};
D3D11_INPUT_ELEMENT_DESC desc2[] =
{{"POSITION", 0, DXGI_FORMAT_R32G32B32_FLOAT, 0, 0,D3D11_INPUT_PER_VERTEX_DATA, 0},{"NORMAL", 0, DXGI_FORMAT_R32G32B32_FLOAT, 0, 12,D3D11_INPUT_PER_VERTEX_DATA, 0},{"TEXCOORD", 0, DXGI_FORMAT_R32G32_FLOAT, 0, 24,D3D11_INPUT_PER_VERTEX_DATA, 0},{"TEXCOORD", 1, DXGI_FORMAT_R32G32_FLOAT, 0, 32,D3D11_INPUT_PER_VERTEX_DATA, 0}
};
创建并绑定输入布局
当定义了输入元素的描述之后, 我们就可以通过 ID3D11Device::CreateInputLayout 方法来创建输入布局( ID3D11InputLayout )了.
HRESULT ID3D11Device::CreateInputLayout(const D3D11_INPUT_ELEMENT_DESC *pInputElementDescs,UINT NumElements,const void *pShaderBytecodeWithInputSignature,SIZE_T BytecodeLength,ID3D11InputLayout **ppInputLayout);
- pInputElementDescs: 之前定义的D3D11_INPUT_ELEMENT_DESC数组.
- NumElements: D3D11_INPUT_ELEMENT_DESC数组的长度.
- pShaderBytecodeWithInputSignature: 一个带有顶点shader输入签名的shader字节码的指针.
- BytecodeLength: 前一个参数指明的顶点shader的bytes大小.
这里进一步解释一下第三个参数. 顶点shader将一个名为 Input Signature 的列表作为输入参数. 用户自定义定点格式元素在传递给顶点shader时需要一一对应. 通过这个签名D3D可以验证这个对应关系并在创建的时候就完成顶点结构到shader输入的映射.
// 一种没有对应的定点结构和shader输入
VertexOut VS(float3 Pos : POSITION,float4 Color : COLOR,float3 Normal : NORMAL) { }struct Vertex
{XMFLOAT3 Pos;XMFLOAT4 Color;
};
接下来说明如何使用 ID3D11Device::CreateInputLayout , 这里用到了一些还没讨论过的东西, 但是最终我们的目的是拿到之前所说的输入签名.
ID3DX11Effect* mFX;
ID3DX11EffectTechnique* mTech;
ID3D11InputLayout* mInputLayout;
// 创建 effect
mTech = mFX->GetTechniqueByName("Tech");
D3DX11_PASS_DESC passDesc;
mTech->GetPassByIndex(0)->GetDesc(&passDesc);
HR(md3dDevice->CreateInputLayout(vertexDesc, 4,passDesc.pIAInputSignature,passDesc.IAInputSignatureSize,&mInputLayout));
这时InputLayout已经创建出来了, 但是我们还需要将他与我们用来渲染的设备进行绑定, 这里用到的是:
md3dImmediateContext->IASetInputLayout(mInputLayout);
需要注意一点是: 输入布局与设备绑定之后直到你自己手动修改它之前都不会发生改变, 当你有不止一个输入布局时需要自己重新设置.
这篇关于DirectX11笔记(五)--Direct3D渲染1--VERTICES AND INPUT LAYOUTS的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!