179 lines
4.7 KiB
C++
179 lines
4.7 KiB
C++
/*
|
|
* Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
#include "webrtc/test/linux/glx_renderer.h"
|
|
|
|
#include <assert.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <X11/Xatom.h>
|
|
#include <X11/Xlib.h>
|
|
|
|
#include "webrtc/common_video/libyuv/include/webrtc_libyuv.h"
|
|
|
|
namespace webrtc {
|
|
namespace test {
|
|
|
|
GlxRenderer::GlxRenderer(size_t width, size_t height)
|
|
: width_(width),
|
|
height_(height),
|
|
display_(NULL),
|
|
context_(NULL) {
|
|
assert(width > 0);
|
|
assert(height > 0);
|
|
}
|
|
|
|
GlxRenderer::~GlxRenderer() { Destroy(); }
|
|
|
|
bool GlxRenderer::Init(const char* window_title) {
|
|
if ((display_ = XOpenDisplay(NULL)) == NULL) {
|
|
Destroy();
|
|
return false;
|
|
}
|
|
|
|
int screen = DefaultScreen(display_);
|
|
|
|
XVisualInfo* vi;
|
|
int attr_list[] = { GLX_DOUBLEBUFFER, GLX_RGBA, GLX_RED_SIZE, 4,
|
|
GLX_GREEN_SIZE, 4, GLX_BLUE_SIZE, 4, GLX_DEPTH_SIZE, 16,
|
|
None, };
|
|
|
|
if ((vi = glXChooseVisual(display_, screen, attr_list)) == NULL) {
|
|
Destroy();
|
|
return false;
|
|
}
|
|
|
|
context_ = glXCreateContext(display_, vi, 0, true);
|
|
if (context_ == NULL) {
|
|
Destroy();
|
|
return false;
|
|
}
|
|
|
|
XSetWindowAttributes window_attributes;
|
|
window_attributes.colormap = XCreateColormap(
|
|
display_, RootWindow(display_, vi->screen), vi->visual, AllocNone);
|
|
window_attributes.border_pixel = 0;
|
|
window_attributes.event_mask = StructureNotifyMask | ExposureMask;
|
|
window_ = XCreateWindow(display_, RootWindow(display_, vi->screen), 0, 0,
|
|
width_, height_, 0, vi->depth, InputOutput,
|
|
vi->visual, CWBorderPixel | CWColormap | CWEventMask,
|
|
&window_attributes);
|
|
XFree(vi);
|
|
|
|
XSetStandardProperties(display_, window_, window_title, window_title, None,
|
|
NULL, 0, NULL);
|
|
|
|
Atom wm_delete = XInternAtom(display_, "WM_DELETE_WINDOW", True);
|
|
if (wm_delete != None) {
|
|
XSetWMProtocols(display_, window_, &wm_delete, 1);
|
|
}
|
|
|
|
XMapRaised(display_, window_);
|
|
|
|
if (!glXMakeCurrent(display_, window_, context_)) {
|
|
Destroy();
|
|
return false;
|
|
}
|
|
GlRenderer::Init();
|
|
if (!glXMakeCurrent(display_, None, NULL)) {
|
|
Destroy();
|
|
return false;
|
|
}
|
|
|
|
Resize(width_, height_);
|
|
return true;
|
|
}
|
|
|
|
void GlxRenderer::Destroy() {
|
|
if (context_ != NULL) {
|
|
glXMakeCurrent(display_, window_, context_);
|
|
GlRenderer::Destroy();
|
|
glXMakeCurrent(display_, None, NULL);
|
|
glXDestroyContext(display_, context_);
|
|
context_ = NULL;
|
|
}
|
|
|
|
if (display_ != NULL) {
|
|
XCloseDisplay(display_);
|
|
display_ = NULL;
|
|
}
|
|
}
|
|
|
|
GlxRenderer* GlxRenderer::Create(const char* window_title, size_t width,
|
|
size_t height) {
|
|
GlxRenderer* glx_renderer = new GlxRenderer(width, height);
|
|
if (!glx_renderer->Init(window_title)) {
|
|
// TODO(pbos): Add GLX-failed warning here?
|
|
delete glx_renderer;
|
|
return NULL;
|
|
}
|
|
return glx_renderer;
|
|
}
|
|
|
|
void GlxRenderer::Resize(size_t width, size_t height) {
|
|
width_ = width;
|
|
height_ = height;
|
|
if (!glXMakeCurrent(display_, window_, context_)) {
|
|
abort();
|
|
}
|
|
GlRenderer::ResizeViewport(width_, height_);
|
|
if (!glXMakeCurrent(display_, None, NULL)) {
|
|
abort();
|
|
}
|
|
|
|
XSizeHints* size_hints = XAllocSizeHints();
|
|
if (size_hints == NULL) {
|
|
abort();
|
|
}
|
|
size_hints->flags = PAspect;
|
|
size_hints->min_aspect.x = size_hints->max_aspect.x = width_;
|
|
size_hints->min_aspect.y = size_hints->max_aspect.y = height_;
|
|
XSetWMNormalHints(display_, window_, size_hints);
|
|
XFree(size_hints);
|
|
|
|
XWindowChanges wc;
|
|
wc.width = static_cast<int>(width);
|
|
wc.height = static_cast<int>(height);
|
|
XConfigureWindow(display_, window_, CWWidth | CWHeight, &wc);
|
|
}
|
|
|
|
void GlxRenderer::OnFrame(const webrtc::VideoFrame& frame) {
|
|
if (static_cast<size_t>(frame.width()) != width_ ||
|
|
static_cast<size_t>(frame.height()) != height_) {
|
|
Resize(static_cast<size_t>(frame.width()),
|
|
static_cast<size_t>(frame.height()));
|
|
}
|
|
|
|
XEvent event;
|
|
if (!glXMakeCurrent(display_, window_, context_)) {
|
|
abort();
|
|
}
|
|
while (XPending(display_)) {
|
|
XNextEvent(display_, &event);
|
|
switch (event.type) {
|
|
case ConfigureNotify:
|
|
GlRenderer::ResizeViewport(event.xconfigure.width,
|
|
event.xconfigure.height);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
GlRenderer::OnFrame(frame);
|
|
glXSwapBuffers(display_, window_);
|
|
|
|
if (!glXMakeCurrent(display_, None, NULL)) {
|
|
abort();
|
|
}
|
|
}
|
|
} // test
|
|
} // webrtc
|