我们绘制了1000个陨石,如果我们将数量提升到 2000 个,会怎么样?
在编译 shader 的时候就会报错:
那要怎么样才能知道 UBO 最大的大小限制呢?
获取 单个UBO大小 的最大限制
使用 glGetIntegerv 接口即可,传入:GL_MAX_UNIFORM_BLOCK_SIZE
// GL_MAX_UNIFORM_BLOCK_SIZE
GLint maxUniformBlockSize;
glGetIntegerv(GL_MAX_UNIFORM_BLOCK_SIZE, &maxUniformBlockSize);
std::cout
<< "Maximun number of Uniform Block Size : "
<< maxUniformBlockSize
<< std::endl;
输出:Maximun number of Uniform Block Size : 65536
注意不同的硬件中,这个限制大小是不同的,这里是我的笔记本上的测试结果
这个:65536
大小的单位是:字节
uniform Instancing_UBO {
mat4 instancingMat[1000]; // instancing 矩阵
};
从上面 shader 代码可以看到 UBO 中有一个 mat4 的数组成员,数组大小为1000,一个 mat4 就是相当于 4个 vec4
每个 vec4 也就是 4个 float
所以一个 mat4 相当于 float4x4(DX的 shader : HLSL 数据类型),也就是 4x4=16 个 float
一个 float = 4 bytes,所以 mat4 = 16 * 4 个 bytes = 64 个 bytes
那么 1000 个 mat4 就是 1000 * 64 = 64000 bytes ,已经相当接近 65536 了,
那么我们可以通过 MaxUBOSize : 65536 除以单个 mat4 需要的字节(64字节),就可以知道我们的硬件最大可以给单个 UBO 分配多大的大小:
65536 / 64 = 1024,所以我的硬件是只支持单个 UBO 最多 1024个 mat4 的大小,也就是 64 KB 的大小
OK,我测试后,1024 个实例化是没有 shader 编译错误问题的
那么为了测试,我改为 1025 个后,也出现了上面的大小限制的问题了
上面的示例中,只是在 UBO 中使用了一个 mat4 的属性,如果还有其他的颜色,或是其他的属性,那么 UBO 的大小限制很快就成为最大的问题的
那么 UBO 的方式限制这么大,我们在一些场景中的绘制实例数量会大大超过这个数量,有些甚至到达 10万个,或以上
除了 UBO 大小之外,不知道大家没有发现,这种方式,还需要对 glsl Shader 层的 UBO指定固定大小
使用起来很不方便,这个数量很大时候取决于应用层的实例数量而动态变化大小的,虽然可以将 UBO 设置大些,但 UBO 大小本身有限制,而且不同硬件限制大小还不一样,如,我这个笔记本,1024 个 mat4 的大小(64KB)就到顶了
所以使用 UBO 的方式限制问题还是很大的。