【着色器和项目】
使用着色器渲染有两个基本的对象类型你需要去创建:着色器对象和项目对象。着色器对象是包含单个着色器的物体。源码输入着色器对象,
着色器对象被编辑为目标格式(.obj文件)。完成后着色器对象能链接到项目对象上,一个项目可有多个着色器。opengles一个项目中有一个顶点
着色器和一个片段着色器(不能多不能少),然后链接成可执行文件,最后能用来渲染。
为产生链接着色器目标,首先要创建顶点着色器和片段着色器,编译源码创建项目,然后编译链接。
创建着色器:GLuint glCreateShader(GLenum type) // type是GL_VERTEX_SHADER或GL_FRAGMENT_SHADER
删除着色器:void glDeleteShader(GLuint shader)
若着色器正连接着一个项目对象,glDeleteShader不立刻删除着色器,而是设置删除标记,一旦着色器不再链接项目对象,才删除其内存。
提供着色器源码:void glShaderSource(GLuint shader, GLsizei count, const char** string, const GLint *length)
编译着色器:void glCompileShader(GLuint shader)
查询错误:void glGetShaderiv(GLuint shader, GLenum pname, GLint *params)
回忆下加载着色器的代码:
GLuint LoadShader( GLenum type, const char *shaderSrc ) { GLuint shader; GLint compliled; // 创建着色器对象 shader = glCreateShader(type); if (!shader) return 0; // 加载着色器源 glShaderSource(shader, 1, &shaderSrc, NULL); // 编译着色器 glCompileShader(shader); // 测试编译状态 glGetShaderiv(shader, GL_COMPILE_STATUS, &compliled); if (!compliled) { GLint infoLen = 0; glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &infoLen); if (infoLen > 1) { char *infoLog = new char[infoLen]; glGetShaderInfoLog(shader, infoLen, NULL, infoLog); esLogMessage("Error compliling shader:\n%s\n", infoLog); delete infoLog; } glDeleteShader(shader); return 0; } return shader; }
项目 是 你创建的着色器要链接到的对象及链接后可执行的程序。项目调用着色器很简单。
创建项目:GLuint glCreateProgram(void)
删除项目:void glDeleteProgram(GLuint program)
链接顶点着色器和片段着色器:void glAttachShader(GLuint program, GLuint shader)
注意着色器能在任何时候被连接,不需要编译或源码。但要求一个项目只能有一个顶点着色器和一个片段着色器。
也可使用glDetachShader分离一个着色器:void glDetachShader(GLuint program, GLuint shader)
着色器连接后需要链接着色器,使用glLinkProgram: void glLinkProgram(GLuint program)
链接产生最后的可执行程序。链接器将保证每个片段着色器的变量都是顶点着色器产生的(类型相同);链接器也保证所有的
片段着色器uniforms常量和顶点着色器是匹配的;...
链接后使用glGetProgramiv检查链接是否成功:void glGetProgramiv(GLuint program, GLenum pname, GLint *params)
链接成功后准备渲染。要查询项目是否有效,有一些原因会导致项目不能执行,例如程序没有对采样器绑定一个有效的贴图。
检查项目当前的执行状态:void glValidateProgram(GLuint program)
注意:若glValidateProgram只为了调试,将是缓慢的,不能在渲染前随时使用。
设置项目为实际的渲染目标:void glUseProgram(GLuint program)
回忆下项目有关的源码:
// 创建程序对象 programObject = glCreateProgram(); if (!programObject) return 0; glAttachShader(programObject, vertexShader); glAttachShader(programObject, fragmentShader); // 设置顶点着色器vPosition属性,绑定attribute 0 glBindAttribLocation(programObject, 0, "vPosition"); // 链接项目 检查错误 glLinkProgram(programObject); glGetProgramiv(programObject, GL_LINK_STATUS, &linked); if (!linked) { GLint infoLen = 0; glGetProgramiv(programObject, GL_INFO_LOG_LENGTH, &infoLen); if (infoLen > 1) { char *infoLog = new char[infoLen]; glGetProgramInfoLog(programObject, infoLen, NULL, infoLog); esLogMessage("Error linking program:\n%s\n", infoLog); delete infoLog; } glDeleteProgram(programObject); return FALSE; }
链接项目对象时,有些查询要做。首先是找到项目使用的uniforms,输入着色器的只读变量,这些常量集也被项目对象共享。
项目对象里有一系列的常量集合,若常量被顶点着色器和片段着色器共同使用,他们的值在两个着色器里应该是相同的。链接
阶段,链接器将分配常量在项目里的实际地址,那个地址是被应用程序使用和加载的标志。
【获取和使用uniforms】
在项目里查询uniforms,可使用GL_ACTIVE_UNIFORMS参数调用glGetProgramiv,这将告诉你项目里实际的常量数目,
若常量处于active,他正在被项目使用。另外你可声明一些常量但不使用,链接器将会优化掉这些常量,不将他存储到实际使用的
常量列表中。可使用GL_ACTIVE_UNIFORM_MAX_LENGTH参数调用glGetProgramiv,找出在项目里最大的常量列表名字。
在得知实际使用的常量数目和描述的数目,可使用glGetActiveUniform查找常量的细节:
void glGetActiveUniform(GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, char *name)
index是被查询的常量索引,bufSize存储名字特征的矩阵数目..若查询出的是矩阵,size是被项目使用的最大的矩阵元素的数目(+1),
若uniform被查询出不是矩阵,这个值是1.
type是要写的uniform类型:GL_FLOAT,GL_FLOAT_VEC2,GL_FLOAT_MAT4,GL_SAMPLER_2D,GL_SAMPLER_CUBE
使用glGetActiveUniform就能知道所有的uniform的属性,能通过其类型确认它的名字。有了常量的名字后可使用glGetUniformLocation
找到他的地址,这个地址是整形,能被后面的函数使用,如glUniformlf。
GLint glGetUniformLocation(GLuint program, const char *name)
返回常量地址,若常量值在项目中未使用将返回-1.有了常量地址及类型和矩阵尺寸,可用数值装载常量,使用不同函数装载不同常量。
void glUniform1f(GLint location, GLfloat x)
void glUniforml1v(GLint location, GLsizei count, const GLfloat *v)
...
void glUniform2i(GL...)
...
void glUniformMatrix3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
...
装载常量函数大部分是自动完成的,使用哪个函数装载常量取决于glGetActiveUniform函数返回的类型。注意glUniform*调用不使用项目
对象做参数。是因为他调用总是在当前的项目中绑定glUseProgram执行。常量值在项目对象中总是保持一致。即一旦你在项目里设置一个
常量值,这个值将保持不变,甚至你激活另一项目。
// 查询激活的常量 GLint maxUniformLen; GLint numUniforms; char *uniformName; GLint index; glGetProgramiv(progObj, GL_ACTIVE_UNIFORMS, &numUniforms); glGetProgramiv(progObj, GL_ACTIVE_UNIFORM_MAX_LENGTH, &maxUniformLen); uniformName = new char[maxUniformLen]; for (index = 0; index < numUniforms; ++index) { GLint size; GLenum type; GLint location; // 获取信息 glGetActiveUniform(progObj, index, maxUniformLen, NULL, &size, &type, uniformName); // 获取地址 location = glGetUniformLocation(progObj, uniformName); switch (type) { case GL_FLOAT: //.. break; // .. } }【获取和设置属性】
除了查询常量信息,还需要设置顶点属性。能用GL_ACTIVE_ATTRIBUTES查询到一个属性列表,能用glGetActiveAttrib查询一个属性的内容。
那是一个设置顶点矩阵,装载顶点属性值的集合。
设置顶点属性还需要一些基元和顶点着色器的知识,6章介绍。
【着色器编程和着色器二进制码】
。。。P47
opengles2——3.着色器和编程,布布扣,bubuko.com
原文:http://blog.csdn.net/adfansong/article/details/22610941