顶点属性数据可以用一个顶点数组对每个顶点指定,也可以将一个常量值用于一个图元的所有顶点
OpenGLES支持最少16个顶点属性。准确查询顶点数量方法如下:
GLint maxVertexAttribs;
glGetIntegerv(GL_MAX_VERTEX_ATTRIBS, &maxVertexAttribs);
一、指定顶点数组
1、常量顶点属性:
常量顶点属性对于一个图元的所有顶点都相同,所以对一个图元的所有顶点只需指定一个值。函数如下:
void glVertexAttrib1f(GLuint index, GLfloat x);
void glVertexAttrib2f(GLuint index, GLfloat x, GLfloat y);
void glVertexAttrib3f(GLuint index, GLfloat x, GLfloat y, GLfloat z);
void glVertexAttrib4f(GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w);
void glVertexAttrib1fv(GLuint index, GLfloat x);
void glVertexAttrib2fv(GLuint index, GLfloat x, GLfloat y);
void glVertexAttrib3fv(GLuint index, GLfloat x, GLfloat y, GLfloat z);
void glVertexAttrib4fv(GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w);
glVertexAttrib* 命令用于加载index 指定的通用顶点属性。
2、顶点数组
顶点数组指定每个顶点的属性,是保存应用程序地址空间的缓冲区。他们作为顶点缓冲对象的基础,提供指定顶点属性数据的一个高效、灵活的手段。
void glVertexAttribPointer(GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, const void *ptr);
void glVertexAttribIPointer(GLuint index, GLint size, GLenum type, GLsizei stride, const void *ptr);
index - 指定通用顶点属性索引。
size - 顶点数组中为索引引用的顶点属性所指定的分量数量。
type - 数据格式,两条函数均有的值:GL_BYTE,GL_UNSIGNED_BYTE,GL_SHORT,GL_UNSIGNED_SHORT,GL_INT,GL_UNSIGNED_INT,glVertexAttribPointer还包括GL_HALF_FLOAT,GL_FLOAT,GL_FIXED,GL_INT_2_10_10_10_REV,GL_UNSIGNED_INT_2_10_10_10_REV
normalized - 用于表示非浮点数据格式类型在转换为浮点值时是否应该规范化。对于glVertexAttribIPointer来说,这些值被当作整数对待
stride - 每个顶点由size指定顶点属性分量顺序存储。stride指定顶点索引 表示顶点数据之间的位移,大于0,表示使用该值作为下一个索引表示的顶点数据的跨距。
ptr - 如果使用客户端顶点数据,则保存顶点属性数据的缓冲区的指针,如果使用顶点缓冲区对象,则表示该缓冲区内的偏移量。
分配和存储顶点属性数据有两种常用的方法:
在一个缓冲区中存储顶点属性 —— 这种方法称为结构数组。结构表示顶点的所有属性,每个顶点有一个属性的数组。结构数组的分配方法通常是最高效的。
在单独的缓冲区中保存每个顶点属性 —— 这个方法称为数组结构。
顶点属性使用哪种数据格式
glVertexAttribPointer 中用type参数指定的顶点属性数据格式不仅影响顶点属性数据的图形内存存储需求,而且影响整体性能。OpenGLES 3.0开始支持GL_HALF_FLOAT 的16位浮点顶点格式,建议尽可能使用GL_HALF_FLOAT,即纹理、坐标、法线、副法线、切向量等使用GL_HALF_FLOAT,颜色存储为GL_UNSIGNED_BYTE,每个顶点颜色具有4个分量。
glVertexAttribPointer中规范化标志是如何工作的?
在用于顶点着色器之前,顶点属性在内部保存为单精度浮点数。如果数据类型表示顶点属性不是浮点数,顶点属性将在用于顶点着色器之前转换为单精度浮点数。规范化标志控制非浮点数顶点属性数据到单精度浮点值的转换,标志为假时,顶点数据被肢解转换为浮点数,如果为真且数据类型为GL_BYTE、GL_SHORT、GL_FIXED,则顶点数据被映射到[-1.0, 1.0]范围内,如果数据类型为GL_UNSIGNED_BYTE或GL_UNSIGNED_SHORT,则被映射到[0.0, 1.0]范围内。如果想要按照整数的形式访问整数型顶点,应该使用glVertexAttribIPointer。
常量顶点属性和顶点数组之间选择
glEnableVertexAttribArray 和 glDisableVertexAttribArray 命令分别用于启用和禁用通用顶点属性数组。
二、在顶点着色器中声明顶点属性变量
属性变量也可以选择包含一个布局限定符,提供属性索引。示例:
layout(location = 0) in vec4 a_postion;
layout(location = 1) in vec2 a_texcoord;
layout(location = 2) in vec3 a_normal;
顶点着色器中声明为顶点属性的变量是只读变量,不能修改。属性可以在顶点着色器内部声明,不使用不会被认为是活动属性,着色器属性数量不能大于GL_MAX_VERTEX_ATTRIBS,否则无法被链接。
将顶点属性绑定到顶点着色器中的属性变量
顶点属性由in限定符指定,活动属性数量可以用glGetProgramiv查询,程序中活动属性列表可以用glGetActiveAttrib查询。
指定通用顶点属性和绑定到顶点着色器中的属性名称:
OpenGLES3.0中,可以使用3种方法将通用顶点属性索引映射到顶点着色器中的一个属性变量名称。
1、索引在源码中用layout(location = N)声明指定
2、OpenGLES3.0将通用顶点属性索引绑定到属性名称
3、应用程序可以将顶点属性索引绑定到属性名称
使用glBindAttribLocation命令可以将通用顶点属性索引绑定到顶点着色器中的一个属性变量。这种绑定在下一个程序链接时生效——不会改变当前链接的程序中使用的绑定。
另外,也可以使用glGetAttribLocation命令查询分配的绑定,需要在返回program定义的程序独享最后一次链接时绑定到属性变量的通用属性索引。
三、顶点缓冲区对象
顶点数组指定的顶点数据保存在客户内存中。在进行glDrawArrays活着glDrawElements等绘图调用时,这些数据必须同客户内存复制到图形内存。
没必要每次绘图时都复制顶点数据,而是在图形内存中缓存这些数据,这样可以显著改善渲染性能,也可以降低内存带宽和电力消耗需求。这就是顶点缓冲区对象发挥作用的地方。
不紧紧是顶点数据,描述图元顶点索引、作为glDrawElements参数传递的元素索引也可以缓存。
GL_ARRAY_BUFFER 指定的数组缓冲区对象用于创建保存顶点数据的缓冲区对象。
GL_ELEMENT_ARRAY_BUFFER 指定的元素数组缓冲区对象用于保存图元索引的缓冲区对象。
创建和绑定顶点缓冲区对象例子:
void initVertexBufferObjects(vertex_t *vertexBuffer,
GLushort *indices,
GLuint numVertices,
GLuint numIndices,
GLuint *vboIds) {
// 获取vboIds中两个未用的缓冲区对象名称。然后vboIds返回的未使用的缓冲区对象名称用于创建一个数组缓冲区对象
// 和一个元素数组缓冲区对象。数组缓冲区对象用于保存一个或多个图元的顶点属性数据。
glGenBuffers(2, vboIds);
glBindBuffer(GL_ARRAY_BUFFER, vboIds[0]);
glBufferData(GL_ARRAY_BUFFER, numVertices *sizeof(vertex_t),
vertexBuffer, GL_STATIC_DRAW);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, vboIds[1]);
glBufferData(GL_ELEMENT_ARRAY_BUFFER, numIndices *sizeof(GLushort),
indices, GL_STATIC_DRAW);
}
glGenBuffers(GLsizei n, GLuint * buffers) :分配n个缓冲区对象名称,并在buffers中返回它们。
glBindBuffer 用于指定当前缓冲区对象。第一次通过调用glBindBuffer绑定缓冲区对象名称时,缓冲区对象可以默认状态分配;如果分配成功,则分配的对象绑定微目标的当前缓冲区对象。
glBufferData 用于创建和初始化顶点数组或元素数组