// 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/surface/accelerated_surface_mac.h"
#include "base/logging.h"
#include "base/mac/scoped_cftyperef.h"
#include "ui/gfx/rect.h"
#include "ui/gl/gl_bindings.h"
#include "ui/gl/gl_context.h"
#include "ui/gl/gl_implementation.h"
#include "ui/gl/gl_surface.h"
#include "ui/gl/io_surface_support_mac.h"
#include "ui/gl/scoped_make_current.h"
AcceleratedSurface::AcceleratedSurface()
: io_surface_id_(0),
allocate_fbo_(false),
texture_(0),
fbo_(0) {
}
AcceleratedSurface::~AcceleratedSurface() {}
bool AcceleratedSurface::Initialize(
gfx::GLContext* share_context,
bool allocate_fbo,
gfx::GpuPreference gpu_preference) {
allocate_fbo_ = allocate_fbo;
// Ensure GL is initialized before trying to create an offscreen GL context.
if (!gfx::GLSurface::InitializeOneOff())
return false;
// Drawing to IOSurfaces via OpenGL only works with Apple's GL and
// not with the OSMesa software renderer.
if (gfx::GetGLImplementation() != gfx::kGLImplementationDesktopGL &&
gfx::GetGLImplementation() != gfx::kGLImplementationAppleGL)
return false;
gl_surface_ = gfx::GLSurface::CreateOffscreenGLSurface(gfx::Size(1, 1));
if (!gl_surface_.get()) {
Destroy();
return false;
}
gfx::GLShareGroup* share_group =
share_context ? share_context->share_group() : NULL;
gl_context_ = gfx::GLContext::CreateGLContext(
share_group,
gl_surface_.get(),
gpu_preference);
if (!gl_context_.get()) {
Destroy();
return false;
}
// Now we're ready to handle SetSurfaceSize calls, which will
// allocate and/or reallocate the IOSurface and associated offscreen
// OpenGL structures for rendering.
return true;
}
void AcceleratedSurface::Destroy() {
// The FBO and texture objects will be destroyed when the OpenGL context,
// and any other contexts sharing resources with it, is. We don't want to
// make the context current one last time here just in order to delete
// these objects.
gl_context_ = NULL;
gl_surface_ = NULL;
}
// Call after making changes to the surface which require a visual update.
// Makes the rendering show up in other processes.
void AcceleratedSurface::SwapBuffers() {
if (io_surface_.get() != NULL) {
if (allocate_fbo_) {
// Bind and unbind the framebuffer to make changes to the
// IOSurface show up in the other process.
glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0);
glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, fbo_);
glFlush();
} else {
// Copy the current framebuffer's contents into our "live" texture.
// Note that the current GL context might not be ours at this point!
// This is deliberate, so that surrounding code using GL can produce
// rendering results consumed by the AcceleratedSurface.
// Need to save and restore OpenGL state around this call.
GLint current_texture = 0;
GLenum target_binding = GL_TEXTURE_BINDING_RECTANGLE_ARB;
GLenum target = GL_TEXTURE_RECTANGLE_ARB;
glGetIntegerv(target_binding, ¤t_texture);
glBindTexture(target, texture_);
glCopyTexSubImage2D(target, 0,
0, 0,
0, 0,
real_surface_size_.width(),
real_surface_size_.height());
glBindTexture(target, current_texture);
// This flush is absolutely essential -- it guarantees that the
// rendering results are seen by the other process.
glFlush();
}
}
}
static void AddBooleanValue(CFMutableDictionaryRef dictionary,
const CFStringRef key,
bool value) {
CFDictionaryAddValue(dictionary, key,
(value ? kCFBooleanTrue : kCFBooleanFalse));
}
static void AddIntegerValue(CFMutableDictionaryRef dictionary,
const CFStringRef key,
int32 value) {
base::ScopedCFTypeRef<CFNumberRef> number(
CFNumberCreate(NULL, kCFNumberSInt32Type, &value));
CFDictionaryAddValue(dictionary, key, number.get());
}
// Creates a new OpenGL texture object bound to the given texture target.
// Caller owns the returned texture.
static GLuint CreateTexture(GLenum target) {
GLuint texture = 0;
glGenTextures(1, &texture);
glBindTexture(target, texture);
glTexParameteri(target, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
glTexParameteri(target, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
glTexParameteri(target, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
glTexParameteri(target, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
return texture;
}
void AcceleratedSurface::AllocateRenderBuffers(GLenum target,
const gfx::Size& size) {
if (!texture_) {
// Generate the texture object.
texture_ = CreateTexture(target);
// Generate and bind the framebuffer object.
glGenFramebuffersEXT(1, &fbo_);
glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, fbo_);
}
// Make sure that subsequent set-up code affects the render texture.
glBindTexture(target, texture_);
}
bool AcceleratedSurface::SetupFrameBufferObject(GLenum target) {
glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, fbo_);
GLenum fbo_status;
glFramebufferTexture2DEXT(GL_FRAMEBUFFER_EXT,
GL_COLOR_ATTACHMENT0_EXT,
target,
texture_,
0);
fbo_status = glCheckFramebufferStatusEXT(GL_FRAMEBUFFER_EXT);
return fbo_status == GL_FRAMEBUFFER_COMPLETE_EXT;
}
gfx::Size AcceleratedSurface::ClampToValidDimensions(const gfx::Size& size) {
return gfx::Size(std::max(size.width(), 1), std::max(size.height(), 1));
}
bool AcceleratedSurface::MakeCurrent() {
if (!gl_context_.get())
return false;
return gl_context_->MakeCurrent(gl_surface_.get());
}
void AcceleratedSurface::Clear(const gfx::Rect& rect) {
DCHECK(gl_context_->IsCurrent(gl_surface_.get()));
glClearColor(0, 0, 0, 0);
glViewport(0, 0, rect.width(), rect.height());
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
glOrtho(0, rect.width(), 0, rect.height(), -1, 1);
glClear(GL_COLOR_BUFFER_BIT);
}
uint32 AcceleratedSurface::SetSurfaceSize(const gfx::Size& size) {
if (surface_size_ == size) {
// Return 0 to indicate to the caller that no new backing store
// allocation occurred.
return 0;
}
// Only support IO surfaces if the GL implementation is the native desktop GL.
// IO surfaces will not work with, for example, OSMesa software renderer
// GL contexts.
if (gfx::GetGLImplementation() != gfx::kGLImplementationDesktopGL)
return 0;
IOSurfaceSupport* io_surface_support = IOSurfaceSupport::Initialize();
if (!io_surface_support)
return 0;
ui::ScopedMakeCurrent make_current(gl_context_.get(), gl_surface_.get());
if (!make_current.Succeeded())
return 0;
gfx::Size clamped_size = ClampToValidDimensions(size);
// GL_TEXTURE_RECTANGLE_ARB is the best supported render target on
// Mac OS X and is required for IOSurface interoperability.
GLenum target = GL_TEXTURE_RECTANGLE_ARB;
if (allocate_fbo_) {
AllocateRenderBuffers(target, clamped_size);
} else if (!texture_) {
// Generate the texture object.
texture_ = CreateTexture(target);
}
// Allocate a new IOSurface, which is the GPU resource that can be
// shared across processes.
base::ScopedCFTypeRef<CFMutableDictionaryRef> properties;
properties.reset(CFDictionaryCreateMutable(kCFAllocatorDefault,
0,
&kCFTypeDictionaryKeyCallBacks,
&kCFTypeDictionaryValueCallBacks));
AddIntegerValue(properties,
io_surface_support->GetKIOSurfaceWidth(),
clamped_size.width());
AddIntegerValue(properties,
io_surface_support->GetKIOSurfaceHeight(),
clamped_size.height());
AddIntegerValue(properties,
io_surface_support->GetKIOSurfaceBytesPerElement(), 4);
AddBooleanValue(properties,
io_surface_support->GetKIOSurfaceIsGlobal(), true);
// I believe we should be able to unreference the IOSurfaces without
// synchronizing with the browser process because they are
// ultimately reference counted by the operating system.
io_surface_.reset(io_surface_support->IOSurfaceCreate(properties));
// Don't think we need to identify a plane.
GLuint plane = 0;
CGLError error = io_surface_support->CGLTexImageIOSurface2D(
static_cast<CGLContextObj>(gl_context_->GetHandle()),
target,
GL_RGBA,
clamped_size.width(),
clamped_size.height(),
GL_BGRA,
GL_UNSIGNED_INT_8_8_8_8_REV,
io_surface_.get(),
plane);
if (error != kCGLNoError) {
DLOG(ERROR) << "CGL error " << error << " during CGLTexImageIOSurface2D";
}
if (allocate_fbo_) {
// Set up the frame buffer object.
if (!SetupFrameBufferObject(target)) {
DLOG(ERROR) << "Failed to set up frame buffer object";
}
}
surface_size_ = size;
real_surface_size_ = clamped_size;
// Now send back an identifier for the IOSurface. We originally
// intended to send back a mach port from IOSurfaceCreateMachPort
// but it looks like Chrome IPC would need to be modified to
// properly send mach ports between processes. For the time being we
// make our IOSurfaces global and send back their identifiers. On
// the browser process side the identifier is reconstituted into an
// IOSurface for on-screen rendering.
io_surface_id_ = io_surface_support->IOSurfaceGetID(io_surface_);
return io_surface_id_;
}
uint32 AcceleratedSurface::GetSurfaceId() {
return io_surface_id_;
}