| Index: media/tools/player_x11/gles_video_renderer.cc
|
| diff --git a/media/tools/player_x11/gles_video_renderer.cc b/media/tools/player_x11/gles_video_renderer.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..601be1d64308569e5f8dc4097e30b7aa307d31c2
|
| --- /dev/null
|
| +++ b/media/tools/player_x11/gles_video_renderer.cc
|
| @@ -0,0 +1,344 @@
|
| +// Copyright (c) 2010 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +#include "media/tools/player_x11/gles_video_renderer.h"
|
| +
|
| +#include <dlfcn.h>
|
| +#include <X11/Xutil.h>
|
| +#include <X11/extensions/Xrender.h>
|
| +#include <X11/extensions/Xcomposite.h>
|
| +
|
| +#include "media/base/buffers.h"
|
| +#include "media/base/yuv_convert.h"
|
| +
|
| +GlesVideoRenderer* GlesVideoRenderer::instance_ = NULL;
|
| +
|
| +GlesVideoRenderer::GlesVideoRenderer(Display* display, Window window)
|
| + : display_(display),
|
| + window_(window),
|
| + new_frame_(false),
|
| + egl_display_(NULL),
|
| + egl_surface_(NULL),
|
| + egl_context_(NULL) {
|
| +}
|
| +
|
| +GlesVideoRenderer::~GlesVideoRenderer() {
|
| +}
|
| +
|
| +// static
|
| +bool GlesVideoRenderer::IsMediaFormatSupported(
|
| + const media::MediaFormat& media_format) {
|
| + int width = 0;
|
| + int height = 0;
|
| + return ParseMediaFormat(media_format, &width, &height);
|
| +}
|
| +
|
| +void GlesVideoRenderer::OnStop() {
|
| + eglMakeCurrent(egl_display_, EGL_NO_SURFACE,
|
| + EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
| + eglDestroyContext(egl_display_, egl_context_);
|
| + eglDestroySurface(egl_display_, egl_surface_);
|
| +}
|
| +
|
| +// Matrix used for the YUV to RGB conversion.
|
| +static const float kYUV2RGB[9] = {
|
| + 1.f, 0.f, 1.403f,
|
| + 1.f, -.344f, -.714f,
|
| + 1.f, 1.772f, 0.f,
|
| +};
|
| +
|
| +// Vertices for a full screen quad.
|
| +static const float kVertices[8] = {
|
| + -1.f, 1.f,
|
| + -1.f, -1.f,
|
| + 1.f, 1.f,
|
| + 1.f, -1.f,
|
| +};
|
| +
|
| +// Texture Coordinates mapping the entire texture.
|
| +static const float kTextureCoords[8] = {
|
| + 0, 0,
|
| + 0, 1,
|
| + 1, 0,
|
| + 1, 1,
|
| +};
|
| +
|
| +// Pass-through vertex shader.
|
| +static const char kVertexShader[] =
|
| + "precision highp float; precision highp int;\n"
|
| + "varying vec2 interp_tc;\n"
|
| + "\n"
|
| + "attribute vec4 in_pos;\n"
|
| + "attribute vec2 in_tc;\n"
|
| + "\n"
|
| + "void main() {\n"
|
| + " interp_tc = in_tc;\n"
|
| + " gl_Position = in_pos;\n"
|
| + "}\n";
|
| +
|
| +// YUV to RGB pixel shader. Loads a pixel from each plane and pass through the
|
| +// matrix.
|
| +static const char kFragmentShader[] =
|
| + "precision mediump float; precision mediump int;\n"
|
| + "varying vec2 interp_tc;\n"
|
| + "\n"
|
| + "uniform sampler2D y_tex;\n"
|
| + "uniform sampler2D u_tex;\n"
|
| + "uniform sampler2D v_tex;\n"
|
| + "uniform mat3 yuv2rgb;\n"
|
| + "\n"
|
| + "void main() {\n"
|
| + " float y = texture2D(y_tex, interp_tc).x;\n"
|
| + " float u = texture2D(u_tex, interp_tc).r - .5;\n"
|
| + " float v = texture2D(v_tex, interp_tc).r - .5;\n"
|
| + " vec3 rgb = yuv2rgb * vec3(y, u, v);\n"
|
| + " gl_FragColor = vec4(rgb, 1);\n"
|
| + "}\n";
|
| +
|
| +// Buffer size for compile errors.
|
| +static const unsigned int kErrorSize = 4096;
|
| +
|
| +bool GlesVideoRenderer::OnInitialize(media::VideoDecoder* decoder) {
|
| + if (!ParseMediaFormat(decoder->media_format(), &width_, &height_))
|
| + return false;
|
| +
|
| + LOG(INFO) << "Initializing GLES Renderer...";
|
| +
|
| + // Resize the window to fit that of the video.
|
| + XResizeWindow(display_, window_, width_, height_);
|
| +
|
| + egl_display_ = eglGetDisplay(display_);
|
| + if (eglGetError() != EGL_SUCCESS) {
|
| + DLOG(ERROR) << "eglGetDisplay failed.";
|
| + return false;
|
| + }
|
| +
|
| + EGLint major;
|
| + EGLint minor;
|
| + if (!eglInitialize(egl_display_, &major, &minor)) {
|
| + DLOG(ERROR) << "eglInitialize failed.";
|
| + return false;
|
| + }
|
| + DLOG(INFO) << "EGL vendor:" << eglQueryString(egl_display_, EGL_VENDOR);
|
| + DLOG(INFO) << "EGL version:" << eglQueryString(egl_display_, EGL_VERSION);
|
| + DLOG(INFO) << "EGL extensions:"
|
| + << eglQueryString(egl_display_, EGL_EXTENSIONS);
|
| + DLOG(INFO) << "EGL client apis:"
|
| + << eglQueryString(egl_display_, EGL_CLIENT_APIS);
|
| +
|
| + EGLint attribs[] = {
|
| + EGL_RED_SIZE, 5,
|
| + EGL_GREEN_SIZE, 6,
|
| + EGL_BLUE_SIZE, 5,
|
| + EGL_DEPTH_SIZE, 16,
|
| + EGL_STENCIL_SIZE, 0,
|
| + EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
|
| + EGL_NONE
|
| + };
|
| +
|
| + EGLint num_configs = -1;
|
| + if (!eglGetConfigs(egl_display_, NULL, 0, &num_configs)) {
|
| + DLOG(ERROR) << "eglGetConfigs failed.";
|
| + return false;
|
| + }
|
| +
|
| + EGLConfig config;
|
| + if (!eglChooseConfig(egl_display_, attribs, &config, 1, &num_configs)) {
|
| + DLOG(ERROR) << "eglChooseConfig failed.";
|
| + return false;
|
| + }
|
| +
|
| + EGLint red_size, green_size, blue_size, alpha_size, depth_size, stencil_size;
|
| + eglGetConfigAttrib(egl_display_, config, EGL_RED_SIZE, &red_size);
|
| + eglGetConfigAttrib(egl_display_, config, EGL_GREEN_SIZE, &green_size);
|
| + eglGetConfigAttrib(egl_display_, config, EGL_BLUE_SIZE, &blue_size);
|
| + eglGetConfigAttrib(egl_display_, config, EGL_ALPHA_SIZE, &alpha_size);
|
| + eglGetConfigAttrib(egl_display_, config, EGL_DEPTH_SIZE, &depth_size);
|
| + eglGetConfigAttrib(egl_display_, config, EGL_STENCIL_SIZE, &stencil_size);
|
| + DLOG(INFO) << "R,G,B,A: " << red_size << "," << green_size
|
| + << "," << blue_size << "," << alpha_size << " bits";
|
| + DLOG(INFO) << "Depth: " << depth_size << " bits, Stencil:" << stencil_size
|
| + << "bits";
|
| +
|
| + egl_surface_ = eglCreateWindowSurface(egl_display_, config, window_, NULL);
|
| + if (!egl_surface_) {
|
| + DLOG(ERROR) << "eglCreateWindowSurface failed.";
|
| + return false;
|
| + }
|
| +
|
| + egl_context_ = eglCreateContext(egl_display_, config, NULL, NULL);
|
| + if (!egl_context_) {
|
| + DLOG(ERROR) << "eglCreateContext failed.";
|
| + eglDestroySurface(egl_display_, egl_surface_);
|
| + return false;
|
| + }
|
| +
|
| + if (eglMakeCurrent(egl_display_, egl_surface_,
|
| + egl_surface_, egl_context_) == EGL_FALSE) {
|
| + eglDestroyContext(egl_display_, egl_context_);
|
| + eglDestroySurface(egl_display_, egl_surface_);
|
| + egl_display_ = NULL;
|
| + egl_surface_ = NULL;
|
| + egl_context_ = NULL;
|
| + return false;
|
| + }
|
| +
|
| + EGLint width;
|
| + EGLint height;
|
| + eglQuerySurface(egl_display_, egl_surface_, EGL_WIDTH, &width);
|
| + eglQuerySurface(egl_display_, egl_surface_, EGL_HEIGHT, &height);
|
| + glViewport(0, 0, width, height);
|
| +
|
| + glViewport(0, 0, width_, height_);
|
| +
|
| + // Create 3 textures, one for each plane, and bind them to different
|
| + // texture units.
|
| + glGenTextures(media::VideoSurface::kNumYUVPlanes, textures_);
|
| + glActiveTexture(GL_TEXTURE0);
|
| + glBindTexture(GL_TEXTURE_2D, textures_[0]);
|
| + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
| + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
| + glEnable(GL_TEXTURE_2D);
|
| +
|
| + glActiveTexture(GL_TEXTURE1);
|
| + glBindTexture(GL_TEXTURE_2D, textures_[1]);
|
| + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
| + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
| + glEnable(GL_TEXTURE_2D);
|
| +
|
| + glActiveTexture(GL_TEXTURE2);
|
| + glBindTexture(GL_TEXTURE_2D, textures_[2]);
|
| + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
| + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
| + glEnable(GL_TEXTURE_2D);
|
| +
|
| + GLuint program_ = glCreateProgram();
|
| +
|
| + // Create our YUV->RGB shader.
|
| + GLuint vertex_shader_ = glCreateShader(GL_VERTEX_SHADER);
|
| + const char* vs_source = kVertexShader;
|
| + int vs_size = sizeof(kVertexShader);
|
| + glShaderSource(vertex_shader_, 1, &vs_source, &vs_size);
|
| + glCompileShader(vertex_shader_);
|
| + int result = GL_FALSE;
|
| + glGetShaderiv(vertex_shader_, GL_COMPILE_STATUS, &result);
|
| + if (!result) {
|
| + char log[kErrorSize];
|
| + int len;
|
| + glGetShaderInfoLog(vertex_shader_, kErrorSize - 1, &len, log);
|
| + log[kErrorSize - 1] = 0;
|
| + LOG(FATAL) << log;
|
| + }
|
| + glAttachShader(program_, vertex_shader_);
|
| +
|
| + GLuint fragment_shader_ = glCreateShader(GL_FRAGMENT_SHADER);
|
| + const char* ps_source = kFragmentShader;
|
| + int ps_size = sizeof(kFragmentShader);
|
| + glShaderSource(fragment_shader_, 1, &ps_source, &ps_size);
|
| + glCompileShader(fragment_shader_);
|
| + result = GL_FALSE;
|
| + glGetShaderiv(fragment_shader_, GL_COMPILE_STATUS, &result);
|
| + if (!result) {
|
| + char log[kErrorSize];
|
| + int len;
|
| + glGetShaderInfoLog(fragment_shader_, kErrorSize - 1, &len, log);
|
| + log[kErrorSize - 1] = 0;
|
| + LOG(FATAL) << log;
|
| + }
|
| + glAttachShader(program_, fragment_shader_);
|
| +
|
| + glLinkProgram(program_);
|
| + result = GL_FALSE;
|
| + glGetProgramiv(program_, GL_LINK_STATUS, &result);
|
| + if (!result) {
|
| + char log[kErrorSize];
|
| + int len;
|
| + glGetProgramInfoLog(program_, kErrorSize - 1, &len, log);
|
| + log[kErrorSize - 1] = 0;
|
| + LOG(FATAL) << log;
|
| + }
|
| + glUseProgram(program_);
|
| +
|
| + // Bind parameters.
|
| + glUniform1i(glGetUniformLocation(program_, "y_tex"), 0);
|
| + glUniform1i(glGetUniformLocation(program_, "u_tex"), 1);
|
| + glUniform1i(glGetUniformLocation(program_, "v_tex"), 2);
|
| + int yuv2rgb_location = glGetUniformLocation(program_, "yuv2rgb");
|
| + glUniformMatrix3fv(yuv2rgb_location, 1, GL_TRUE, kYUV2RGB);
|
| +
|
| + int pos_location = glGetAttribLocation(program_, "in_pos");
|
| + glEnableVertexAttribArray(pos_location);
|
| + glVertexAttribPointer(pos_location, 2, GL_FLOAT, GL_FALSE, 0, kVertices);
|
| +
|
| + int tc_location = glGetAttribLocation(program_, "in_tc");
|
| + glEnableVertexAttribArray(tc_location);
|
| + glVertexAttribPointer(tc_location, 2, GL_FLOAT, GL_FALSE, 0,
|
| + kTextureCoords);
|
| +
|
| + // We are getting called on a thread. Release the context so that it can be
|
| + // made current on the main thread.
|
| + eglMakeCurrent(egl_display_, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
| +
|
| + // Save this instance.
|
| + DCHECK(!instance_);
|
| + instance_ = this;
|
| + return true;
|
| +}
|
| +
|
| +void GlesVideoRenderer::OnFrameAvailable() {
|
| + AutoLock auto_lock(lock_);
|
| + new_frame_ = true;
|
| +}
|
| +
|
| +void GlesVideoRenderer::Paint() {
|
| + // Use |new_frame_| to prevent overdraw since Paint() is called more
|
| + // often than needed. It is OK to lock only this flag and we don't
|
| + // want to lock the whole function because this method takes a long
|
| + // time to complete.
|
| + {
|
| + AutoLock auto_lock(lock_);
|
| + if (!new_frame_)
|
| + return;
|
| + new_frame_ = false;
|
| + }
|
| +
|
| + scoped_refptr<media::VideoFrame> video_frame;
|
| + GetCurrentFrame(&video_frame);
|
| +
|
| + if (!video_frame)
|
| + return;
|
| +
|
| + // Convert YUV frame to RGB.
|
| + media::VideoSurface frame_in;
|
| + if (video_frame->Lock(&frame_in)) {
|
| + DCHECK(frame_in.format == media::VideoSurface::YV12 ||
|
| + frame_in.format == media::VideoSurface::YV16);
|
| + DCHECK(frame_in.strides[media::VideoSurface::kUPlane] ==
|
| + frame_in.strides[media::VideoSurface::kVPlane]);
|
| + DCHECK(frame_in.planes == media::VideoSurface::kNumYUVPlanes);
|
| +
|
| + if (eglGetCurrentContext() != egl_context_) {
|
| + eglMakeCurrent(egl_display_, egl_surface_,
|
| + egl_surface_, egl_context_);
|
| + }
|
| + for (unsigned int i = 0; i < media::VideoSurface::kNumYUVPlanes; ++i) {
|
| + unsigned int width = (i == media::VideoSurface::kYPlane) ?
|
| + frame_in.width : frame_in.width / 2;
|
| + unsigned int height = (i == media::VideoSurface::kYPlane ||
|
| + frame_in.format == media::VideoSurface::YV16) ?
|
| + frame_in.height : frame_in.height / 2;
|
| + glActiveTexture(GL_TEXTURE0 + i);
|
| + // No GL_UNPACK_ROW_LENGTH in GLES2.
|
| + // glPixelStorei(GL_UNPACK_ROW_LENGTH, frame_in.strides[i]);
|
| + glTexImage2D(GL_TEXTURE_2D, 0, GL_LUMINANCE, width, height, 0,
|
| + GL_LUMINANCE, GL_UNSIGNED_BYTE, frame_in.data[i]);
|
| + }
|
| + video_frame->Unlock();
|
| + } else {
|
| + NOTREACHED();
|
| + }
|
| +
|
| + glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
|
| + eglSwapBuffers(egl_display_, egl_surface_);
|
| +}
|
|
|