| /* |
| * QEMU opengl shader helper functions |
| * |
| * Copyright (c) 2014 Red Hat |
| * |
| * Authors: |
| * Gerd Hoffmann <kraxel@redhat.com> |
| * |
| * Permission is hereby granted, free of charge, to any person obtaining a copy |
| * of this software and associated documentation files (the "Software"), to deal |
| * in the Software without restriction, including without limitation the rights |
| * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
| * copies of the Software, and to permit persons to whom the Software is |
| * furnished to do so, subject to the following conditions: |
| * |
| * The above copyright notice and this permission notice shall be included in |
| * all copies or substantial portions of the Software. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
| * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
| * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
| * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
| * THE SOFTWARE. |
| */ |
| #include "qemu/osdep.h" |
| #include "ui/shader.h" |
| |
| #include "ui/shader/texture-blit-vert.h" |
| #include "ui/shader/texture-blit-flip-vert.h" |
| #include "ui/shader/texture-blit-frag.h" |
| |
| struct QemuGLShader { |
| GLint texture_blit_prog; |
| GLint texture_blit_flip_prog; |
| GLint texture_blit_vao; |
| }; |
| |
| /* ---------------------------------------------------------------------- */ |
| |
| static GLuint qemu_gl_init_texture_blit(GLint texture_blit_prog) |
| { |
| static const GLfloat in_position[] = { |
| -1, -1, |
| 1, -1, |
| -1, 1, |
| 1, 1, |
| }; |
| GLint l_position; |
| GLuint vao, buffer; |
| |
| glGenVertexArrays(1, &vao); |
| glBindVertexArray(vao); |
| |
| /* this is the VBO that holds the vertex data */ |
| glGenBuffers(1, &buffer); |
| glBindBuffer(GL_ARRAY_BUFFER, buffer); |
| glBufferData(GL_ARRAY_BUFFER, sizeof(in_position), in_position, |
| GL_STATIC_DRAW); |
| |
| l_position = glGetAttribLocation(texture_blit_prog, "in_position"); |
| glVertexAttribPointer(l_position, 2, GL_FLOAT, GL_FALSE, 0, 0); |
| glEnableVertexAttribArray(l_position); |
| |
| glBindBuffer(GL_ARRAY_BUFFER, 0); |
| glBindVertexArray(0); |
| |
| return vao; |
| } |
| |
| void qemu_gl_run_texture_blit(QemuGLShader *gls, bool flip) |
| { |
| glUseProgram(flip |
| ? gls->texture_blit_flip_prog |
| : gls->texture_blit_prog); |
| glBindVertexArray(gls->texture_blit_vao); |
| glDrawArrays(GL_TRIANGLE_STRIP, 0, 4); |
| } |
| |
| /* ---------------------------------------------------------------------- */ |
| |
| static GLuint qemu_gl_create_compile_shader(GLenum type, const GLchar *src) |
| { |
| GLuint shader; |
| GLint status, length; |
| char *errmsg; |
| |
| shader = glCreateShader(type); |
| glShaderSource(shader, 1, &src, 0); |
| glCompileShader(shader); |
| |
| glGetShaderiv(shader, GL_COMPILE_STATUS, &status); |
| if (!status) { |
| glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &length); |
| errmsg = g_malloc(length); |
| glGetShaderInfoLog(shader, length, &length, errmsg); |
| fprintf(stderr, "%s: compile %s error\n%s\n", __func__, |
| (type == GL_VERTEX_SHADER) ? "vertex" : "fragment", |
| errmsg); |
| g_free(errmsg); |
| return 0; |
| } |
| return shader; |
| } |
| |
| static GLuint qemu_gl_create_link_program(GLuint vert, GLuint frag) |
| { |
| GLuint program; |
| GLint status, length; |
| char *errmsg; |
| |
| program = glCreateProgram(); |
| glAttachShader(program, vert); |
| glAttachShader(program, frag); |
| glLinkProgram(program); |
| |
| glGetProgramiv(program, GL_LINK_STATUS, &status); |
| if (!status) { |
| glGetProgramiv(program, GL_INFO_LOG_LENGTH, &length); |
| errmsg = g_malloc(length); |
| glGetProgramInfoLog(program, length, &length, errmsg); |
| fprintf(stderr, "%s: link program: %s\n", __func__, errmsg); |
| g_free(errmsg); |
| return 0; |
| } |
| return program; |
| } |
| |
| static GLuint qemu_gl_create_compile_link_program(const GLchar *vert_src, |
| const GLchar *frag_src) |
| { |
| GLuint vert_shader, frag_shader, program = 0; |
| |
| vert_shader = qemu_gl_create_compile_shader(GL_VERTEX_SHADER, vert_src); |
| frag_shader = qemu_gl_create_compile_shader(GL_FRAGMENT_SHADER, frag_src); |
| if (!vert_shader || !frag_shader) { |
| goto end; |
| } |
| |
| program = qemu_gl_create_link_program(vert_shader, frag_shader); |
| |
| end: |
| glDeleteShader(vert_shader); |
| glDeleteShader(frag_shader); |
| |
| return program; |
| } |
| |
| /* ---------------------------------------------------------------------- */ |
| |
| QemuGLShader *qemu_gl_init_shader(void) |
| { |
| QemuGLShader *gls = g_new0(QemuGLShader, 1); |
| |
| gls->texture_blit_prog = qemu_gl_create_compile_link_program |
| (texture_blit_vert_src, texture_blit_frag_src); |
| gls->texture_blit_flip_prog = qemu_gl_create_compile_link_program |
| (texture_blit_flip_vert_src, texture_blit_frag_src); |
| if (!gls->texture_blit_prog || !gls->texture_blit_flip_prog) { |
| exit(1); |
| } |
| |
| gls->texture_blit_vao = |
| qemu_gl_init_texture_blit(gls->texture_blit_prog); |
| |
| return gls; |
| } |
| |
| void qemu_gl_fini_shader(QemuGLShader *gls) |
| { |
| if (!gls) { |
| return; |
| } |
| glDeleteProgram(gls->texture_blit_prog); |
| glDeleteProgram(gls->texture_blit_flip_prog); |
| glDeleteProgram(gls->texture_blit_vao); |
| g_free(gls); |
| } |