gpt4 book ai didi

c++ - 如何更快地绘制许多带纹理的四边形,并保留 glScissor(或类似的东西)?

转载 作者:太空宇宙 更新时间:2023-11-04 11:22:41 25 4
gpt4 key购买 nike

我正在使用 OpenGL 4 和 C++11。

目前,我使用带有单独 VBO 和 IBO 的单独 VAO 对 glDrawElements 进行了一大堆单独调用。

我这样做是因为每个纹理坐标都会发生变化,而我的顶点数据具有纹理坐标。我知道这个顶点数据中有一些冗余的位置信息;但是,它始终为 -1,-1,1,1,因为我在顶点着色器中使用平移矩阵和缩放矩阵来定位和缩放顶点数据。

VAO、VBO、IBO、位置和比例矩阵以及纹理 ID 存储在一个对象中。每个四边形一个对象。

目前,一些绘图会像这样出现:

  1. 通过 (glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_INT,0)) 绘制四边形对象。绑定(bind)的 VBO 只是 -1,-1,1,1 而 IBO 画了一个四边形。绑定(bind)的 VBO 包含一个公共(public)纹理的纹理坐标(相同的纹理用于纹理所有绘制的四边形)。着色器上的矩阵变换对其进行定位。
  2. 用另一个四边形对象重复
  3. glEnable(GL_SCISSOR_TEST) 被调用,预览四边形的位置信息用于调用 glScissor
  4. 绘制下一个四边形对象;实际上只显示了从前一个四边形可见的部分。
  5. 再画一个四边形对象

我现在获得的性能是可以接受的,但我希望它更快,因为我只触及了我的想法的表面。所以我正在考虑优化。到目前为止,我已经读到我应该:

  1. 从我的顶点数据中删除位置信息,只保留纹理坐标。而是在绘制四边形开始时绑定(bind)一个位置 VBO,以便所有四边形都使用它。

    但我不确定这将如何运作?因为我一次只能激活一个 VBO。

    然后我是否必须在绘制每个四边形之前调用 glBufferSubData 并更新纹理坐标?这是更好的性能还是更差的性能(为每个对象调用 glBindVertexArray 或调用 glBufferSubData?)

    我是否仍会将位置和比例作为矩阵传递给着色器,我是否会借此机会更新顶点的位置信息以及纹理坐标?哪个更快?

  2. 创建一个带或不带 IBO 的大型 VBO,并更新其中每个四边形的位置顶点数据(而不是使用变换和缩放矩阵)。这似乎很难管理。

    即使我确实做到了;我只会有一个 glDraw 调用;听起来很快。这是真的?与多次调用相比,单个 glBindVertexArray 调用对性能有何影响?

    我不认为有任何方法可以使用此方法来实现像我现在正在调用的 glScissor 这样的东西?

  3. 我读过的另一个选项是实例化。因此,无论我需要多少次,我都会绘制四边形;这意味着我会向着色器传递一个平移矩阵数组和一个纹理坐标数组?

    这样会不会快很多?

    我想我可以通过传递一个额外的 bool 值数组来做类似 glScissor 测试的事情,该数组定义当前四边形是否应该只在前一个四边形的范围内绘制。但是,我认为这意味着对于每个 gl_InstanceID,我都必须遍历所有以前的实例来寻找 true 和 false 值,这看起来会很慢。

我试图通过不单独实现所有这些来节省时间。希望专家可以指出哪个可能更好。如果有人有更好的主意,请告诉我。

最佳答案

  1. 您可以将多个 VBO 附加到不同的属性!

以下序列将 2 个 vbo 绑定(bind)到属性 0 和 1,请注意 glBindBuffer() 临时绑定(bind)缓冲区,实际的 VBO 分配到属性是在 glVertexAttribPointer() 期间进行的。

glBindBuffer(GL_ARRAY_BUFFER,buf1);
glVertexAttribPointer(0, ...);
glEnableVertexAttribArray(0);

glBindBuffer(GL_ARRAY_BUFFER,buf2);
glVertexAttribPointer(1, ...);
glEnableVertexAttribArray(1);

提供四边形位置和大小的最快方法是使用纹理并在顶点着色器中对其进行采样。当然,您至少需要 RGBA(x、y、宽度、高度)16 位/ channel 纹理。但是随后您可以使用 glTexSubImage2D() 更新四边形位置,或者您甚至可以通过 FBO 渲染它们。

除此之外的一切都会执行得更慢,当然,如果您愿意,我们可以详细说明如何在 vbos 中使用制服、属性或使用不启用数组的属性。

放在一起:

  • 使用单个 vbo,在其中存储 quad id (int) + 你的纹理数据
  • 准备 x,y,w,h 纹理,定义从 quad id 到此纹理 texcoord 的映射,即:u=quad_id&0xFF,v=(quad_id>>8)(对于纹理 256x256 最大 65536 个四边形)
  • 使用顶点着色器从该纹理中采样位移和大小(对于存储在属性中的给定 quad_id(或使用 vertex_ID/4 或 vertex_ID/6)
  • 填充vbo和纹理
  • 用单个 drawarrays 绘制元素绘制所有东西

关于c++ - 如何更快地绘制许多带纹理的四边形,并保留 glScissor(或类似的东西)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27739780/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com