Fast Blit with OpenGL ES 2.0 and texture uploading
OpenGL ES 2.0 is fully shader based, which means you can’t draw any geometry without having the appropriate shaders loaded and bound. This means there is more setup code required to render than there was in OpenGL ES 1.1 with fixed pipeline.
Setting up the OpenGL ES 2.0 pipeline.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
void glBlitter::initializeGL ()
{
GLuint fragmentShader;
GLuint vertexShader;
GLint linked;
glDepthFunc(GL_ALWAYS);
glDisable(GL_DEPTH_TEST);
glDisable(GL_STENCIL_TEST);
glDisable(GL_CULL_FACE);
//Vertex shader
QString srcVertShader =
"attribute vec4 a_position; \n"
"attribute vec2 a_texCoord; \n"
"varying vec2 v_texCoord; \n"
"void main() \n"
"{ \n"
" gl_Position = a_position; \n"
" v_texCoord = a_texCoord; \n"
"} \n";
//fragment shader
QString srcFragShader =
"precision mediump float; \n"
"varying vec2 v_texCoord; \n"
"uniform sampler2D s_texture; \n"
"void main() \n"
"{ \n"
" gl_FragColor = texture2D( s_texture, v_texCoord );\n"
"} \n";
// Create the program object
m_program = glCreateProgram();
if(!m_program)
return;
// Load the shaders
vertexShader = loadShader(qPrintable(srcVertShader), GL_VERTEX_SHADER);
fragmentShader = loadShader(qPrintable(srcFragShader), GL_FRAGMENT_SHADER);
glAttachShader(m_program, vertexShader);
glAttachShader(m_program, fragmentShader);
// Link the program
glLinkProgram(m_program);
// Check the link status
glGetProgramiv(m_program, GL_LINK_STATUS, &linked);
if(!linked){
GLint infoLen = 0;
glGetProgramiv(m_program, GL_INFO_LOG_LENGTH, &infoLen);
if(infoLen > 1){
char* infoLog = (char*)malloc(sizeof(char) * infoLen);
glGetProgramInfoLog(m_program, infoLen, NULL, infoLog);
qDebug() << infoLog;
free(infoLog);
}
glDeleteProgram(m_program);
return;
}
m_posLoc = glGetAttribLocation ( m_program, "a_position" );
m_texLoc = glGetAttribLocation ( m_program, "a_texCoord" );
// Get the sampler location
m_samplerLoc = glGetUniformLocation ( m_program, "s_texture" );
//Init textures to plot in the paintgl function
InitTextures();
glClearColor(.3,.4,.6,1);
}
The actual drawing is done in the QGLWidget’s paintGL function. The OpenGL ES 2.0 guild line says that you should use upload textures with sizes of power of two. However you don’t have blit the whole texture. You can create a texCoords matrix for using only a part of the uploaded texture. There’s a really good tutorial for texturemapping in iPhone Development Blog. It’s for iPhone and it’s for OpenGL ES 1.1 or so, but the vector math in binding the textures etc. is still valid with OpenGL ES 2.0.
Here’s my example of painGL and texture blitting.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
void glBlitter::paintGL()
{
glClear( GL_COLOR_BUFFER_BIT );
// Use the program object
glUseProgram ( m_program );
glVertexAttribPointer( m_posLoc, 2, GL_FLOAT, GL_FALSE, 2 * sizeof(GLfloat), vertexCoords );
glVertexAttribPointer( m_texLoc, 2, GL_FLOAT, GL_FALSE, 2*sizeof(GLfloat), texCoords );
glEnableVertexAttribArray( m_posLoc );
glEnableVertexAttribArray( m_texLoc );
glBindTexture(GL_TEXTURE_2D, m_textures[0] );
//copy the emulator's virtual framebuffer into a texture
copyScreen( m_ScreenPtr, BaseAddress );
//upload the texture
glTexSubImage2D( GL_TEXTURE_2D,0,
0,0, 256,256,
GL_RGB,GL_UNSIGNED_SHORT_5_6_5, m_ScreenPtr );
//sampler texture unit to 0
glUniform1i( m_samplerLoc, 0 );
glDrawElements( GL_TRIANGLES, 6, GL_UNSIGNED_SHORT, indices );
}
You can also download the whole glBlitter.cpp from here: glBlitter.cpp