OpenGL中的Uniform block size 的大小限制

本文探讨了如何在提升陨石实例数量到2000时遇到UBO大小限制,通过计算发现单个UBO大小限制在65536字节。作者介绍了如何确定硬件限制,并分析了UBO在实例化大量矩阵时的局限性,提出了使用UBO的挑战及替代方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我们绘制了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 的方式限制问题还是很大的。

OpenGL UBO(Uniform Buffer Object)是一种高效的缓存机制,用于存储渲染管线中的uniform变量。这些变量可以是顶点着色器、片元着色器或计算着色器中的uniform变量。 以下是OpenGL UBO的使用方法: 1. 创建UBO对象 使用以下代码创建UBO对象: GLuint ubo; glGenBuffers(1, &ubo); 2. 绑定UBO对象 使用以下代码将UBO对象绑定到OpenGL上下文中: glBindBuffer(GL_UNIFORM_BUFFER, ubo); 3. 分配UBO对象内存 使用以下代码分配UBO对象所需的内存: glBufferData(GL_UNIFORM_BUFFER, size, data, GL_STATIC_DRAW); 其中,size是UBO对象所需的内存大小,data是指向要存储在UBO对象中的数据的指针。 4. 绑定UBO对象到uniform变量 使用以下代码将UBO对象绑定到uniform变量: GLuint blockIndex = glGetUniformBlockIndex(program, "uniformBlockName"); glUniformBlockBinding(program, blockIndex, bindingPoint); glBindBufferBase(GL_UNIFORM_BUFFER, bindingPoint, ubo); 其中,program是要绑定的着色器程序对象,"uniformBlockName"是要绑定的uniform块的名称,bindingPoint是UBO对象的绑定点。 5. 更新UBO对象中的数据 使用以下代码更新UBO对象中的数据: glBindBuffer(GL_UNIFORM_BUFFER, ubo); glBufferSubData(GL_UNIFORM_BUFFER, offset, size, data); 其中,offset是要更新的数据在UBO对象中的偏移量,size是要更新的数据大小,data是指向要更新的数据的指针。 6. 渲染 在渲染对象之前,使用以下代码将UBO对象绑定到OpenGL上下文中: glBindBuffer(GL_UNIFORM_BUFFER, ubo); 然后,就可以在着色器程序中使用uniform变量了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值