OpenGL ES 2.0 和 3.0区别

打印 上一主题 下一主题

主题 876|帖子 876|积分 2630

目录
零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 基础
零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 特效
零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 转场
零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 函数
零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES GPUImage 使用
零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES GLSL 编程
一.嵌入式设备的 OpenGL ES 版本

OpenGL ES 1.x 支持 初代 iPhone 和 Android;
OpenGL ES 2.0 支持 Android 2.2 以后的平台,支持 iPad , iPhone3GS 和后续版本,以及 iPodTouch3 代和后续版本。
OpenGL ES 3.0 支持 Android 4.3 以后的平台。支持 iPhone 5s ,iPad Air ,iPad mini 2 及后续版本。
二.兼容性

OpenGL ES 3.0 是向下兼容 OpenGL ES 2.0 的。也就是说使用 2.0 编写的应用程序是可以在 3.0 中继续使用的。

三.着色器脚本

1.OpenGL ES shader 2.0
  1. //顶点着色器
  2. attribute vec4 aPosition;                            // 应用程序传入顶点着色器的顶点位置
  3. attribute vec2 aTextureCoord;                        // 应用程序传入顶点着色器的顶点纹理坐标
  4. varying vec2 vTextureCoord;                          // 用于传递给片元着色器的顶点纹理数据
  5. void main()
  6. {
  7.     gl_Position = aPosition;                         // 此次绘制此顶点位置
  8.     vTextureCoord = aTextureCoord;                   // 将接收的纹理坐标传递给片元着色器
  9. }
  10. //片元着色器
  11. precision mediump float;                           // 设置工作精度
  12. varying vec2 vTextureCoord;                        // 接收从顶点着色器过来的纹理坐标
  13. uniform sampler2D sTexture;                        // 纹理采样器,代表一幅纹理
  14. void main()
  15. {
  16.     gl_FragColor = texture2D(sTexture, vTextureCoord);// 进行纹理采样
  17. }
复制代码
2.OpenGL ES shader 3.0
  1. //顶点着色器
  2. #version es 300
  3. uniform mat4 u_matViewProj;
  4. layout(location = 0) in vec4 a_position;
  5. layout(location = 1) in vec3 a_color;
  6. out vec3 v_color;
  7. void main() {
  8.     gl_Position = u_matViewProj * a_position;
  9.     v_color = a_color;
  10. }
  11. //片元着色器
  12. #version es 300
  13. precision mediump float;
  14. in vec3 v_color; // input form vertex shader
  15. layout(location = 0) out vec4 o_fragColor;
  16. void main() {
  17.     o_fragColor = vec4(v_color, 1.0);
  18. }
复制代码
3.版本声明

在 OpenGL ES 3.0 中,顶点着色器和片段着色器的第一行必须声明着色器版本,否则编译报错:
  1. ERROR: 0:1: '' : syntax error: #version directive must occur in a shader before anything
复制代码
OpenGL ES 3.0 中,可以必须声明着色器版本:
  1. #version es 300
复制代码
OpenGL ES 2.0 中,可以不用声明着色器版本,默认为:
  1. #version es 100
复制代码
备注: 以往 2.0 刚刚出来可编程的图形管线,所以版本声明为 #version 100 es ,后来为了使版本号相匹配,OpenGL ES 3.0 的 shader 版本直接从 1.0 跳到了 3.0 。
  1. #version 300 es 指定使用OpenGL3.0
  2. #version 100 es 指定使用OpenGL2.0 (不指定version 默认为OpenGL2.0)
复制代码
4. 默认精度修饰符 precision

在顶点语言中有如下预定义的全局默认精度语句:
  1. precision highp float;
  2. precision highp int;
  3. precision lowp sampler2D;
  4. precision lowp samplerCube;
复制代码
在片元语言中有如下预定义的全局默认精度语句:
  1. precision mediump int;
  2. precision lowp sampler2D;
  3. precision lowp samplerCube;
复制代码
片元语言没有默认的浮点数精度修饰符。因此,对于浮点数,浮点数向量和矩阵变量声明,要么声明必须包含一个精度修饰符,要不默认的精度修饰符在之前已经被声明过了
  1. precision highp float;
复制代码
4.输入输出

OpenGL ES 3.0 中新增了** in,out,inout **关键字,用来取代 **attribute 和 varying **关键字。
同时 gl_FragColorgl_FragData 也删除了,片段着色器可以使用 out 声明字段输出。
5.变量赋值

OpenGL ES 3.0 中可以直接使用 layout 对指定位置的变量赋值。例如:
  1. shader脚本中
  2. layout (location = 1) uniform float a;
复制代码
代码中,直接写上对应的 layout 的值就可以赋值
  1. GLES30.glUniform1f(1, 1f);
复制代码
而 OpenGL ES 2.0 中必须使用如下形式赋值:
  1. GLES20.glUniform1f(GLES20.glGetAttribLocation(program, "a"), 1f)
复制代码
四.关于顶点缓冲区对象 VBO 与顶点数组对象 VAO

VAO (顶点数组对象:Vertex Array Object)是指顶点数组对象,主要用于管理 VBO 或 EBO ,减少 glBindBuffer 、glEnableVertexAttribArray、 glVertexAttribPointer 这些调用操作,高效地实现在顶点数组配置之间切换。
**VBO(顶点缓冲区对象: Vertex Buffer Object)是指把顶点数据保存在显存中,绘制时直接从显存中取数据,减少了数据传输的开销,**因为顶点数据多了,就是坐标的数据多了很多的很多组,切换的时候很麻烦,就出现了这个 VAO,绑定对应的顶点数据
OpenGL 2.0 有 VBO,没有 VAO,VAO 是 OpenGL 3.0 才开始支持的,并且在 OpenGL 3.0 中,强制要求绑定一个 VAO 才能开始绘制。
例如:在 OpenGL 3.0 中,不使用 VAO ,调用完 glVertexAttribPointer, glGetError 报错 GL_INVALID_OPERATION
  1. int pos_location = glGetAttribLocation(ProgramId, "position");
  2. glVertexAttribPointer(pos_location, 2, GL_FLOAT, GL_FALSE, 0, kVertices);
复制代码
查看 OpenGL 官方文档 http://docs.gl/gl3/glVertexAttribPointer
GL_INVALID_OPERATION is generated in the core context if there is no Vertex Array Object bound
在 OpenGL 3.0 中,强制要求绑定一个 VAO 才能开始绘制。因此,需要在程序初始化的时候,创建并绑定一个 VAO
  1. GLuint vao;
  2. glGenVertexArrays(1, &vao);
  3. glBindVertexArray(vao);
复制代码
此外,OpenGL 3.0 中也不允许在 glVertexAttribPointer 直接传数组了,因此要把顶点先传入 vbo 中
  1. GLuint vbo;
  2. glGenBuffers(1, &vbo);
  3. glBindBuffer(GL_ARRAY_BUFFER, vbo);
  4. glBufferData(GL_ARRAY_BUFFER, sizeof(kVertices), kVertices, GL_STATIC_DRAW);
复制代码
绑定 VBO 之后,glVertexAttribPointer 后面的 pointer 参数就要填 0 了。
  1. glVertexAttribPointer(pos_location, 2, GL_FLOAT, GL_FALSE, 0, 0);
复制代码
五.PBO

OpenGL 2.0 不支持 PBO ,3.0 支持 PBO , PBO 设计的目的就是快速地向显卡传输数据,或者从显卡读取数据,我们可以使用它更加高效的读取屏幕数据。
单个 PBO 读取屏幕数据效率大概和 glReadPixels 差不多,双 PBO 交换读取效率会很高。

原因是使用 PBO 时,屏幕上的数据不是读取到内存,而是从显卡读到 PBO 中,或者如果内部机制是读取到内存中,但这也是由 DMA 控制器来完成的,而不是 cpu 指令来做的,再加上两个 PBO 交换使用,所以读取效率很高。

PBO 快速地从内存 CPU 向显卡 GPU 传输数据 —> GL_PIXEL_PACK_BUFFER
**PBO 快速地从显卡 GPU 读取数据到内存 CPU** —> GL_PIXEL_UNPACK_BUFFER
六.猜你喜欢

本文由博客 - 猿说编程 猿说编程 发布!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

tsx81428

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表