HLSL常量缓冲区打包规则

DirectX11 With Windows SDK完整目录

欢迎加入QQ群: 727623616 可以一起探讨DX11,以及有什么问题也可以在这里汇报。

尽管打包规则并不复杂,但是稍不留意就可能会导致因为打包规则的不理解而产生的数据错位问题。

下面会使用大量的例子来进行描述,并对有争议的部分使用图形调试器来反汇编着色器代码加以验证。

1. C++中的结构体数据是以字节流的形式传输给HLSL的

例1.1

若C++结构体和HLSL常量缓冲区如下:

// cpp
struct S1
{
XMFLOAT3 p1;
XMFLOAT3 p2;
}; // HLSL
cbuffer C1
{
float4 v1;
float4 v2;
}

则最终C1两个向量接收到的数据如下:

(p1.x, p1.y, p1.z, p2.x)

(p2.y, p2.z, empty, empty)

2. HLSL常量缓冲区中的向量不允许拆分

例2.1

// cpp
struct S1
{
XMFLOAT3 p1;
XMFLOAT3 p2;
}; // HLSL
cbuffer C1
{
float3 v1;
float4 v2;
}

v1将被单独打包成一个4D向量,确保常量缓冲区的内存按128位对齐。

C1的内存布局为:

(v1.x, v1.y, v1.z, empty)

(v2.x, v2.y, v2.z, v2.w)

这时用S1结构体的数据再传输给C1,结果如下:

(p1.x, p1.y, p1.z, p2.x)

(p2.y, p2.z, empty, empty)

例2.2

// HLSL
cbuffer C1
{
float2 v1;
float4 v2;
float2 v3;
}

v2无法拆分来填充v1的空位,而是单独起一行向量,这样C1的内存布局为:

(v1.x, v1.y, empty, empty)

(v2.x, v2.y, v2.z, v2.w)

(v3.x, v3.y, empty, empty)

3. HLSL常量缓冲区中多个相邻的变量若有空缺则优先打包进同一个4D向量中

例3.1

// HLSL
cbuffer C1
{
float v1;
float2 v2;
float v3;
float2 v4;
float v5;
}

C1的内存布局为:

(v1.x, v2.x, v2.y, v3.x)

(v4.x, v4.y, v5.x, empty)

打包顺序是从最上面的变量开始往下的。

例3.2

// HLSL
cbuffer C1
{
float2 v1;
float v2;
float3 v3;
}

C1的内存布局为:

(v1.x, v1.y, v2.x, empty)

(v3.x, v3.y, v3.z, empty)

4. 对于在常量缓冲区的结构体,也会进行打包操作

通过几个例子来进行观察

例4.1

// HLSL
struct S1
{
float2 p1;
float3 p2;
float p3;
}; cbuffer C1
{
float v1;
S1 v2;
float3 v3;
}

C1的内存布局为:

(v1.x, empty, empty, empty)

(v2.p1.x, v2.p1.y, empty, empty)

(v2.p2.x, v2.p2.y, v2.p2.z, v2.p3.x)

(v3.x, v3.y, v3.z, empty)

例4.2

// HLSL
struct S1
{
float4 p1;
float p2;
}; cbuffer C1
{
S1 v1;
float2 v2;
}

C1的内存布局为:

(v1.p1.x, v1.p1.y, v1.p1.z, v1.p1.w)

(v1.p2.x, v2.x, v2.y, empty)

所以,结构体常量前面的所有常量都会被打包成4D向量,但结构体常量的最后一个成员可能会和后续的常量打包成4D向量。

5. 对于在常量缓冲区的数组,需要特殊对待

数组中的每一个元素都会独自打包,但对于最后一个元素来说如果后续的变量不是数组、结构体且还有空缺,则可以进行打包操作

例5.1

// HLSL
cbuffer C1
{
float v1[4];
}

C1的内存布局为:

(v1[0].x, empty, empty, empty)

(v1[1].x, empty, empty, empty)

(v1[2].x, empty, empty, empty)

(v1[3].x, empty, empty, empty)

可以看到,一个本应该是16字节的数组变量实际上变成了64字节的4个4D向量,造成内存的大量浪费。如果真的要使用这种数组,下面的声明方式通过强制转换,可以保证没有空间浪费(C++不允许这么做):

// HLSL
cbuffer C1
{
float4 v1;
}
static float packArray[4] = (float[4])v1;

例5.2

// HLSL
cbuffer C1
{
float2 v1[4];
float2 v2;
}

C1的内存布局实际上为:

(v1[0].x, v1[0].y, empty, empty)

(v1[1].x, v1[1].y, empty, empty)

(v1[2].x, v1[2].y, empty, empty)

(v1[3].x, v1[3].y, v2.x, v2.y)

例5.3

// HLSL
struct S1
{
float p1;
int p2;
}; cbuffer C1
{
S1 v1[4];
float v2;
float3 v3;
}

C1的内存布局实际上为:

(v1[0].p1, v1[0].p2, empty, empty)

(v1[1].p1, v1[1].p2, empty, empty)

(v1[2].p1, v1[2].p2, empty, empty)

(v1[3].p1, v1[3].p2, v2.x, empty)

(v3.x, v3.y, v3.z, empty)

例5.4

// HLSL
struct S1
{
float p1;
int p2;
}; cbuffer C1
{
float v1[2];
S1 v2;
}

C1的内存布局为:

(v1[0].x, empty, empty, empty)

(v1[1].x, empty, empty, empty)

(v2.p1, v2.p2, empty, empty)

使用VS的图形调试器来分析HLSL常量缓冲区的内存布局

首先确保着色器、常量缓冲区都已经绑定到渲染管线,如果该常量缓冲区被绑定到像素着色阶段,就应该在图形调试器中对像素着色器代码进行调试。

例4.1的验证

开启反汇编后,找到之前所放的常量缓冲区:

在这些样例中已经确保了常量缓冲区前面的所有值都已经打包好(16字节对齐)。

这里v1的偏移值为2416,然后可以看到结构体对象v2内的p1偏移值为2432,说明v1单独被打包成16字节向量。然后p1无法和p2打包,所以p1单独打包成16字节。

然后p2p3被打包,因为v3的偏移值为2464p2的偏移值为2448

所以内存布局如下:

(v1.x, empty, empty, empty)

(v2.p1.x, v2.p1.y, empty, empty)

(v2.p2.x, v2.p2.y, v2.p2.z, v2.p3.x)

(v3.x, v3.y, v3.z, empty)

例4.2的验证

v1的偏移值为2416p1构成单独的4D向量,p2会和后续的v2打包成新的4D向量,而不是单独打包。

所以内存布局如下:

(v1.p1.x, v1.p1.y, v1.p1.z, v1.p1.w)

(v1.p2.x, v2.x, v2.y, empty)

例5.2的验证

如果数组的每个元素都单独打包的话,理论上这个数组所占字节数为64,但这里数组的最后一个float2和下面的float2打包成一个4D向量。

所以内存布局如下:

(v1[0].x, v1[0].y, empty, empty)

(v1[1].x, v1[1].y, empty, empty)

(v1[2].x, v1[2].y, empty, empty)

(v1[3].x, v1[3].y, v2.x, v2.y)

例5.3的验证

可以看到结构体数组的每个结构体元素都被单独打包成1个4D向量,但数组最后一个元素跟v2打包到一起。

内存布局如下:

(v1[0].p1, v1[0].p2, empty, empty)

(v1[1].p1, v1[1].p2, empty, empty)

(v1[2].p1, v1[2].p2, empty, empty)

(v1[3].p1, v1[3].p2, v2.x, empty)

(v3.x, v3.y, v3.z, empty)

例5.4的验证

因为数组的后面是结构体,而结构体要求前面的所有变量都要先打包好,所以数组的2个元素分别单独打包。

内存布局如下:

(v1[0].x, empty, empty, empty)

(v1[1].x, empty, empty, empty)

(v2.p1, v2.p2, empty, empty)

记一次打包问题引发的错误

在一次进行图形调试的时候,发现原本设置平行光和点光灯的数目为1,到了图形调试器却变成了点光灯和聚光灯的数目为1

C++代码如下:


struct CBNeverChange
{
DirectionalLight dirLight[10]; // 已按16字节对齐
PointLight pointLight[10]; // 已按16字节对齐
SpotLight spotLight[10]; // 已按16字节对齐
int numDirLight;
int numPointLight;
int numSpotLight;
int pad;
}; // ... mCBNeverChange.numDirLight = 1;
mCBNeverChange.numPointLight = 1;
mCBNeverChange.numSpotLight = 0;

HLSL代码如下:

cbuffer CBNeverChange : register(b3)
{
DirectionalLight gDirLight[10];
PointLight gPointLight[10];
SpotLight gSpotLight[10];
int gNumDirLight;
int gNumPointLight;
int gNumSpotLight;
}

在图形调试器查看C++提供的字节数据,可以看到最后四个32位的传入是没有问题的

经过一番折腾,翻到像素着色器的反编译,发现里面有常量缓冲区数据偏移信息:

仔细比对的话可以发现从gNumDirLight开始的字节偏移量出现了不是我想要的结果,本应该是2400的值,结果却是2396,导致原本赋给gNumDirLightgNumPointLight为1的值,却赋给了gNumPointLightgNumSpotLight。这也是我为什么要写出这篇文章的原因。

常量缓冲区声明技巧

首先重新总结之前的打包规则:

1. C++中的结构体数据是以字节流的形式传输给HLSL的;

2. HLSL常量缓冲区中的向量不允许拆分;

3. HLSL常量缓冲区中多个相邻的变量若有空缺则优先打包进同一个4D向量中;

4. HLSL常量缓冲区中,结构体常量前面的所有常量都会被打包成4D向量,内部也进行打包操作,但结构体的最后一个成员可能会和后续的常量打包成4D向量;

5. 数组中的每一个元素都会独自打包,但对于最后一个元素来说如果后续的变量不是数组、结构体且还有空缺,则可以进行打包操作。

所以避免出现潜在问题的办法如下:

1. 若要使用数组,数组的类型最好能按16字节对齐

2. 结构体的总大小也需要按16字节对齐。

DirectX11 With Windows SDK完整目录

欢迎加入QQ群: 727623616 可以一起探讨DX11,以及有什么问题也可以在这里汇报。

DirectX11--深入理解HLSL常量缓冲区打包规则的更多相关文章

  1. DirectX11 With Windows SDK--03 索引缓冲区、常量缓冲区

    前言 一个立方体有8个顶点,然而绘制一个立方体需要画12个三角形,如果按照前面的方法绘制的话,则需要提供36个顶点,而且这里面的顶点数据会重复4次甚至5次.这样的绘制方法会占用大量的内存空间. 接下来 ...

  2. DirectX11--深入理解与使用缓冲区资源

    前言 在Direct3D 11中,缓冲区属于其中一种资源类型,它在内存上的布局是一维线性的.根据HLSL支持的类型以及C++的使用情况,缓冲区可以分为下面这些类型: 顶点缓冲区(Vertex Buff ...

  3. Direct3D 11 Tutorial 7:Texture Mapping and Constant Buffers_Direct3D 11 教程7:纹理映射和常量缓冲区

    概述 在上一个教程中,我们为项目引入了照明. 现在我们将通过向我们的立方体添加纹理来构建它. 此外,我们将介绍常量缓冲区的概念,并解释如何使用缓冲区通过最小化带宽使用来加速处理. 本教程的目的是修改中 ...

  4. 【转】Directx11 HelloWorld之HLSL的Effect框架的使用

    最近尝试用了下Directx下的Effect框架,作为一初学者初学者,说下为什么我们要使用Effect框架及其好处吧. 首先Effect最大好处的就是简单,使得编写Shader绘制的程序工作量大大下降 ...

  5. Java语言基本语法(一)————关键字&标识符(Java语言标识符命名规范&Java语言的包名、类名、接口名、变量名、函数名、常量名命名规则 )

    一.关键字 关键字的定义和特点 定义:被Java语言赋予特殊含义,用做专门用途的字符串(单词). 特点:关键字中所有字母均为小写 下面列举一些常用的关键字. 用于定义数据类型的关键字:byte.sho ...

  6. 理解SVG的图形填充规则

    SVG的图形填充规则通过fill-rule属性来指定. 有效值:   nonzero | evenodd | inherit 默认值:   nonzero fill-rule属性用于指定使用哪一种算法 ...

  7. Java中包、类、方法、属性、常量的命名规则

    1:包(package):用于将完成不同功能的类分门别类,放在不同的目录(包)下,包的命名规则:将公司域名反转作为包名.比如www.baidu.com 对于包名:每个字母都需要小写.比如:com.ba ...

  8. 理解JAVA常量池

    下面是一些String相关的常见问题: String中的final用法和理解final StringBuffer a = new StringBuffer("111");final ...

  9. 从Java到C++——常量的使用规则

    常量是一种标识符,它的值在执行期间恒定不变.C语言用 #define来定义常量(称为宏常量). C++ 语言除了 #define外还能够用const来定义常量(称为const常量). 一.为什么须要常 ...

随机推荐

  1. 进程命令(taskkill)

    taskkill 命令: // 描述: 结束一个或多个任务或流程. // 语法: taskkill [/s <computer> [/u [<Domain>\]<User ...

  2. spingboot一键部署到阿里云(Cloud Toolkit工具)

    一般做法 一键部署工具   前些天在完成一个项目时候需要将springboot项目部署到服务器上, 以下是两种做法 前面介绍的是一般做法: 后面将介绍省去这些步骤的一键部署工具Cloud Toolki ...

  3. C#动态调用webService出现 基础连接已经关闭: 未能为 SSL/TLS 安全通道建立信任关系。

    这里因为的原因是https请求要检查证书,有些证书不正确的,网页不会正常展示内容,而会返回链接不安全,是否继续.不安全的链接是否继续. 详情参考: C#动态调用webService出现 基础连接已经关 ...

  4. 距离放弃python又近了一大步,而然只是第四天

    今天是周末后的第一天,周末四处浪浪浪,所以在周一的时候就要狠狠的复习之前的东西了,之后从第一天的计算机基础开始复习,具体内容请翻阅前三篇随笔,主要是要仔细看看,怕学了后面的忘了前面的,今天引进的第一个 ...

  5. Linux:Day12(下) 进程、任务计划

    vmstat命令: vmstat [options] [delay [ count]] procs: r:等待运行的进程的个数: b:处于不可中断睡眠态的进程个数:(被阻塞的队列的长度): memor ...

  6. MongoDB 用MongoTemplate查询指定时间范围的数据

    mongoDB大于小于符号对应: > 大于 $gt< 小于 $lt>= 大于等于 $gte<= 小于等于 $lte 要查询同一个时间多个约束可能出现的error: org.sp ...

  7. SQL 增删改语句

    阅读目录 一:插入数据 二:更新数据 三:删除数据 回到顶部 一:插入数据 把数据插入表中的最简单方法是使用基本的 INSERT 语法.它的要求是需要我们指定表名和插入到新行中的值. 1.1 插入完整 ...

  8. Django学习笔记之表单验证

    表单概述 HTML中的表单 单纯从前端的html来说,表单是用来提交数据给服务器的,不管后台的服务器用的是Django还是PHP语言还是其他语言.只要把input标签放在form标签中,然后再添加一个 ...

  9. Python--day02(编程语言、运行python代码、变量)

    day01主要内容回顾 1.进制转换: 二进制: 1111  0101 1010 十六进制          f        5      a 2.内存分布:堆区 和 栈区 外来人只能访问栈区的数据 ...

  10. Web并发页面访问量统计实现

    Web并发页面访问量统计实现 - huangshulang1234的博客 - CSDN博客https://blog.csdn.net/huangshulang1234/article/details/ ...