// Copyright (c) 2012 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 "ui/gl/gl_context_osmesa.h"
#include <GL/osmesa.h>
#include "base/logging.h"
#include "ui/gfx/size.h"
#include "ui/gl/gl_bindings.h"
#include "ui/gl/gl_surface.h"
namespace gfx {
GLContextOSMesa::GLContextOSMesa(GLShareGroup* share_group)
: GLContextReal(share_group),
context_(NULL) {
}
bool GLContextOSMesa::Initialize(GLSurface* compatible_surface,
GpuPreference gpu_preference) {
DCHECK(!context_);
OSMesaContext share_handle = static_cast<OSMesaContext>(
share_group() ? share_group()->GetHandle() : NULL);
GLuint format = compatible_surface->GetFormat();
DCHECK_NE(format, (unsigned)0);
context_ = OSMesaCreateContextExt(format,
0, // depth bits
0, // stencil bits
0, // accum bits
share_handle);
if (!context_) {
LOG(ERROR) << "OSMesaCreateContextExt failed.";
return false;
}
return true;
}
void GLContextOSMesa::Destroy() {
if (context_) {
OSMesaDestroyContext(static_cast<OSMesaContext>(context_));
context_ = NULL;
}
}
bool GLContextOSMesa::MakeCurrent(GLSurface* surface) {
DCHECK(context_);
gfx::Size size = surface->GetSize();
ScopedReleaseCurrent release_current;
if (!OSMesaMakeCurrent(context_,
surface->GetHandle(),
GL_UNSIGNED_BYTE,
size.width(),
size.height())) {
LOG(ERROR) << "OSMesaMakeCurrent failed.";
Destroy();
return false;
}
// Set this as soon as the context is current, since we might call into GL.
SetRealGLApi();
// Row 0 is at the top.
OSMesaPixelStore(OSMESA_Y_UP, 0);
SetCurrent(surface);
if (!InitializeDynamicBindings()) {
return false;
}
if (!surface->OnMakeCurrent(this)) {
LOG(ERROR) << "Could not make current.";
return false;
}
release_current.Cancel();
return true;
}
void GLContextOSMesa::ReleaseCurrent(GLSurface* surface) {
if (!IsCurrent(surface))
return;
SetCurrent(NULL);
// TODO: Calling with NULL here does not work to release the context.
OSMesaMakeCurrent(NULL, NULL, GL_UNSIGNED_BYTE, 0, 0);
}
bool GLContextOSMesa::IsCurrent(GLSurface* surface) {
DCHECK(context_);
bool native_context_is_current =
context_ == OSMesaGetCurrentContext();
// If our context is current then our notion of which GLContext is
// current must be correct. On the other hand, third-party code
// using OpenGL might change the current context.
DCHECK(!native_context_is_current || (GetRealCurrent() == this));
if (!native_context_is_current)
return false;
if (surface) {
GLint width;
GLint height;
GLint format;
void* buffer = NULL;
OSMesaGetColorBuffer(context_, &width, &height, &format, &buffer);
if (buffer != surface->GetHandle())
return false;
}
return true;
}
void* GLContextOSMesa::GetHandle() {
return context_;
}
void GLContextOSMesa::SetSwapInterval(int interval) {
DCHECK(IsCurrent(NULL));
}
GLContextOSMesa::~GLContextOSMesa() {
Destroy();
}
} // namespace gfx