blob: 913b4b093be83ff8b165874af1f50ad4e3dac7c1 [file] [log] [blame]
// Copyright 2014 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 "content/browser/compositor/browser_compositor_view_private_mac.h"
#include <map>
#include "base/debug/trace_event.h"
#include "base/lazy_instance.h"
#include "base/message_loop/message_loop.h"
#include "content/browser/compositor/gpu_process_transport_factory.h"
#include "content/browser/compositor/io_surface_layer_mac.h"
#include "content/browser/compositor/software_layer_mac.h"
#include "content/browser/renderer_host/dip_util.h"
#include "content/browser/renderer_host/render_widget_resize_helper.h"
#include "content/common/gpu/surface_handle_types_mac.h"
#include "content/public/browser/context_factory.h"
#include "ui/base/cocoa/animation_utils.h"
#include "ui/gl/scoped_cgl.h"
namespace content {
namespace {
typedef std::map<gfx::AcceleratedWidget,BrowserCompositorViewMacInternal*>
WidgetToInternalsMap;
base::LazyInstance<WidgetToInternalsMap> g_widget_to_internals_map;
}
////////////////////////////////////////////////////////////////////////////////
// BrowserCompositorViewMacInternal
BrowserCompositorViewMacInternal::BrowserCompositorViewMacInternal()
: client_(NULL),
accelerated_output_surface_id_(0) {
// Disable the fade-in animation as the layers are added.
ScopedCAActionDisabler disabler;
// Add a flipped transparent layer as a child, so that we don't need to
// fiddle with the position of sub-layers -- they will always be at the
// origin.
flipped_layer_.reset([[CALayer alloc] init]);
[flipped_layer_ setGeometryFlipped:YES];
[flipped_layer_ setAnchorPoint:CGPointMake(0, 0)];
[flipped_layer_
setAutoresizingMask:kCALayerWidthSizable|kCALayerHeightSizable];
// Use a sequence number as the accelerated widget handle that we can use
// to look up the internals structure.
static uintptr_t last_sequence_number = 0;
last_sequence_number += 1;
native_widget_ = reinterpret_cast<gfx::AcceleratedWidget>(
last_sequence_number);
g_widget_to_internals_map.Pointer()->insert(
std::make_pair(native_widget_, this));
// Create a compositor to draw the contents of this view.
compositor_.reset(new ui::Compositor(
native_widget_,
content::GetContextFactory(),
RenderWidgetResizeHelper::Get()->task_runner()));
compositor_->SetVisible(false);
}
BrowserCompositorViewMacInternal::~BrowserCompositorViewMacInternal() {
DCHECK(!client_);
g_widget_to_internals_map.Pointer()->erase(native_widget_);
}
void BrowserCompositorViewMacInternal::SetClient(
BrowserCompositorViewMacClient* client) {
// Disable the fade-in animation as the view is added.
ScopedCAActionDisabler disabler;
DCHECK(client && !client_);
client_ = client;
compositor_->SetRootLayer(client_->BrowserCompositorRootLayer());
CALayer* background_layer = [client_->BrowserCompositorSuperview() layer];
DCHECK(background_layer);
[flipped_layer_ setBounds:[background_layer bounds]];
[background_layer addSublayer:flipped_layer_];
compositor_->SetVisible(true);
}
void BrowserCompositorViewMacInternal::ResetClient() {
if (!client_)
return;
// Disable the fade-out animation as the view is removed.
ScopedCAActionDisabler disabler;
[flipped_layer_ removeFromSuperlayer];
DestroyIOSurfaceLayer(io_surface_layer_);
DestroyCAContextLayer(ca_context_layer_);
DestroySoftwareLayer();
accelerated_output_surface_id_ = 0;
last_swap_size_dip_ = gfx::Size();
compositor_->SetVisible(false);
compositor_->SetScaleAndSize(1.0, gfx::Size(0, 0));
compositor_->SetRootLayer(NULL);
client_ = NULL;
}
bool BrowserCompositorViewMacInternal::HasFrameOfSize(
const gfx::Size& dip_size) const {
return last_swap_size_dip_ == dip_size;
}
int BrowserCompositorViewMacInternal::GetRendererID() const {
if (io_surface_layer_)
return [io_surface_layer_ rendererID];
return 0;
}
void BrowserCompositorViewMacInternal::BeginPumpingFrames() {
[io_surface_layer_ beginPumpingFrames];
}
void BrowserCompositorViewMacInternal::EndPumpingFrames() {
[io_surface_layer_ endPumpingFrames];
}
void BrowserCompositorViewMacInternal::GotAcceleratedFrame(
uint64 surface_handle, int output_surface_id,
const std::vector<ui::LatencyInfo>& latency_info,
gfx::Size pixel_size, float scale_factor) {
// Record the surface and latency info to use when acknowledging this frame.
DCHECK(!accelerated_output_surface_id_);
accelerated_output_surface_id_ = output_surface_id;
accelerated_latency_info_.insert(accelerated_latency_info_.end(),
latency_info.begin(), latency_info.end());
// If there is no client and therefore no superview to draw into, early-out.
if (!client_) {
IOSurfaceLayerDidDrawFrame();
return;
}
// Disable the fade-in or fade-out effect if we create or remove layers.
ScopedCAActionDisabler disabler;
last_swap_size_dip_ = ConvertSizeToDIP(scale_factor, pixel_size);
switch (GetSurfaceHandleType(surface_handle)) {
case kSurfaceHandleTypeIOSurface: {
IOSurfaceID io_surface_id = IOSurfaceIDFromSurfaceHandle(surface_handle);
GotAcceleratedIOSurfaceFrame(io_surface_id, pixel_size, scale_factor);
break;
}
case kSurfaceHandleTypeCAContext: {
CAContextID ca_context_id = CAContextIDFromSurfaceHandle(surface_handle);
GotAcceleratedCAContextFrame(ca_context_id, pixel_size, scale_factor);
break;
}
default:
LOG(ERROR) << "Unrecognized accelerated frame type.";
return;
}
}
void BrowserCompositorViewMacInternal::GotAcceleratedCAContextFrame(
CAContextID ca_context_id,
gfx::Size pixel_size,
float scale_factor) {
// In the layer is replaced, keep the old one around until after the new one
// is installed to avoid flashes.
base::scoped_nsobject<CALayerHost> old_ca_context_layer =
ca_context_layer_;
// Create the layer to host the layer exported by the GPU process with this
// particular CAContext ID.
if ([ca_context_layer_ contextId] != ca_context_id) {
ca_context_layer_.reset([[CALayerHost alloc] init]);
[ca_context_layer_ setContextId:ca_context_id];
[ca_context_layer_
setAutoresizingMask:kCALayerMaxXMargin|kCALayerMaxYMargin];
[flipped_layer_ addSublayer:ca_context_layer_];
}
// Acknowledge the frame to unblock the compositor immediately (the GPU
// process will do any required throttling).
IOSurfaceLayerDidDrawFrame();
// If this replacing a same-type layer, remove it now that the new layer is
// in the hierarchy.
if (old_ca_context_layer != ca_context_layer_)
DestroyCAContextLayer(old_ca_context_layer);
// Remove any different-type layers that this is replacing.
DestroyIOSurfaceLayer(io_surface_layer_);
DestroySoftwareLayer();
}
void BrowserCompositorViewMacInternal::GotAcceleratedIOSurfaceFrame(
IOSurfaceID io_surface_id,
gfx::Size pixel_size,
float scale_factor) {
// In the layer is replaced, keep the old one around until after the new one
// is installed to avoid flashes.
base::scoped_nsobject<IOSurfaceLayer> old_io_surface_layer =
io_surface_layer_;
// Create or re-create an IOSurface layer if needed. If there already exists
// a layer but it has the wrong scale factor or it was poisoned, re-create the
// layer.
bool needs_new_layer =
!io_surface_layer_ ||
[io_surface_layer_ hasBeenPoisoned] ||
[io_surface_layer_ scaleFactor] != scale_factor;
if (needs_new_layer) {
io_surface_layer_.reset(
[[IOSurfaceLayer alloc] initWithClient:this
withScaleFactor:scale_factor]);
if (io_surface_layer_)
[flipped_layer_ addSublayer:io_surface_layer_];
else
LOG(ERROR) << "Failed to create IOSurfaceLayer";
}
// Open the provided IOSurface.
if (io_surface_layer_) {
bool result = [io_surface_layer_ gotFrameWithIOSurface:io_surface_id
withPixelSize:pixel_size
withScaleFactor:scale_factor];
if (!result) {
DestroyIOSurfaceLayer(io_surface_layer_);
LOG(ERROR) << "Failed open IOSurface in IOSurfaceLayer";
}
}
// Give a final complaint if anything with the layer's creation went wrong.
// This frame will appear blank, the compositor will try to create another,
// and maybe that will go better.
if (!io_surface_layer_) {
LOG(ERROR) << "IOSurfaceLayer is nil, tab will be blank";
IOSurfaceLayerHitError();
}
// Make the CALayer draw and set its size appropriately.
if (io_surface_layer_) {
[io_surface_layer_ gotNewFrame];
// Set the bounds of the accelerated layer to match the size of the frame.
// If the bounds changed, force the content to be displayed immediately.
CGRect new_layer_bounds = CGRectMake(
0, 0, last_swap_size_dip_.width(), last_swap_size_dip_.height());
bool bounds_changed = !CGRectEqualToRect(
new_layer_bounds, [io_surface_layer_ bounds]);
[io_surface_layer_ setBounds:new_layer_bounds];
if (bounds_changed)
[io_surface_layer_ setNeedsDisplayAndDisplayAndAck];
}
// If this replacing a same-type layer, remove it now that the new layer is
// in the hierarchy.
if (old_io_surface_layer != io_surface_layer_)
DestroyIOSurfaceLayer(old_io_surface_layer);
// Remove any different-type layers that this is replacing.
DestroyCAContextLayer(ca_context_layer_);
DestroySoftwareLayer();
}
void BrowserCompositorViewMacInternal::GotSoftwareFrame(
cc::SoftwareFrameData* frame_data,
float scale_factor,
SkCanvas* canvas) {
if (!frame_data || !canvas || !client_)
return;
// Disable the fade-in or fade-out effect if we create or remove layers.
ScopedCAActionDisabler disabler;
// If there is not a layer for software frames, create one.
if (!software_layer_) {
software_layer_.reset([[SoftwareLayer alloc] init]);
[flipped_layer_ addSublayer:software_layer_];
}
// Set the software layer to draw the provided canvas.
SkImageInfo info;
size_t row_bytes;
const void* pixels = canvas->peekPixels(&info, &row_bytes);
gfx::Size pixel_size(info.fWidth, info.fHeight);
[software_layer_ setContentsToData:pixels
withRowBytes:row_bytes
withPixelSize:pixel_size
withScaleFactor:scale_factor];
last_swap_size_dip_ = ConvertSizeToDIP(scale_factor, pixel_size);
// Remove any different-type layers that this is replacing.
DestroyCAContextLayer(ca_context_layer_);
DestroyIOSurfaceLayer(io_surface_layer_);
}
void BrowserCompositorViewMacInternal::DestroyCAContextLayer(
base::scoped_nsobject<CALayerHost> ca_context_layer) {
if (!ca_context_layer)
return;
[ca_context_layer removeFromSuperlayer];
if (ca_context_layer == ca_context_layer_)
ca_context_layer_.reset();
}
void BrowserCompositorViewMacInternal::DestroyIOSurfaceLayer(
base::scoped_nsobject<IOSurfaceLayer> io_surface_layer) {
if (!io_surface_layer)
return;
[io_surface_layer resetClient];
[io_surface_layer removeFromSuperlayer];
if (io_surface_layer == io_surface_layer_)
io_surface_layer_.reset();
}
void BrowserCompositorViewMacInternal::DestroySoftwareLayer() {
if (!software_layer_)
return;
[software_layer_ removeFromSuperlayer];
software_layer_.reset();
}
bool BrowserCompositorViewMacInternal::IOSurfaceLayerShouldAckImmediately()
const {
// If there is no client then the accelerated layer is not in the hierarchy
// and will never draw.
if (!client_)
return true;
return client_->BrowserCompositorViewShouldAckImmediately();
}
void BrowserCompositorViewMacInternal::IOSurfaceLayerDidDrawFrame() {
if (accelerated_output_surface_id_) {
content::ImageTransportFactory::GetInstance()->OnSurfaceDisplayed(
accelerated_output_surface_id_);
accelerated_output_surface_id_ = 0;
}
if (client_)
client_->BrowserCompositorViewFrameSwapped(accelerated_latency_info_);
accelerated_latency_info_.clear();
}
void BrowserCompositorViewMacInternal::IOSurfaceLayerHitError() {
// Perform all acks that would have been done if the frame had succeeded, to
// un-block the compositor and renderer.
IOSurfaceLayerDidDrawFrame();
// Poison the context being used and request a mulligan.
[io_surface_layer_ poisonContextAndSharegroup];
compositor_->ScheduleFullRedraw();
}
// static
BrowserCompositorViewMacInternal* BrowserCompositorViewMacInternal::
FromAcceleratedWidget(gfx::AcceleratedWidget widget) {
WidgetToInternalsMap::const_iterator found =
g_widget_to_internals_map.Pointer()->find(widget);
// This can end up being accessed after the underlying widget has been
// destroyed, but while the ui::Compositor is still being destroyed.
// Return NULL in these cases.
if (found == g_widget_to_internals_map.Pointer()->end())
return NULL;
return found->second;
}
} // namespace content