Chromium Code Reviews| Index: content/browser/compositor/browser_compositor_ca_layer_tree_mac.mm |
| diff --git a/content/browser/compositor/browser_compositor_ca_layer_tree_mac.mm b/content/browser/compositor/browser_compositor_ca_layer_tree_mac.mm |
| index 3abee8cd7630ff56ae0a427617a7313555cb26a0..50fd27ad3f16ef5884b93f6b535b5a1f31d64b9a 100644 |
| --- a/content/browser/compositor/browser_compositor_ca_layer_tree_mac.mm |
| +++ b/content/browser/compositor/browser_compositor_ca_layer_tree_mac.mm |
| @@ -6,14 +6,13 @@ |
| #include <map> |
| +#include "cc/output/software_frame_data.h" |
| #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" |
| @@ -22,18 +21,30 @@ |
| namespace content { |
| namespace { |
| -typedef std::map<gfx::AcceleratedWidget,BrowserCompositorCALayerTreeMac*> |
| - WidgetToInternalsMap; |
| -base::LazyInstance<WidgetToInternalsMap> g_widget_to_internals_map; |
| +typedef std::map<gfx::AcceleratedWidget,AcceleratedWidgetMac*> |
| + WidgetToHelperMap; |
| +base::LazyInstance<WidgetToHelperMap> g_widget_to_helper_map; |
| + |
| +AcceleratedWidgetMac* GetHelperFromAcceleratedWidget( |
| + gfx::AcceleratedWidget widget) { |
| + WidgetToHelperMap::const_iterator found = |
| + g_widget_to_helper_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_helper_map.Pointer()->end()) |
| + return NULL; |
| + return found->second; |
| +} |
| } |
| //////////////////////////////////////////////////////////////////////////////// |
| -// BrowserCompositorCALayerTreeMac |
| +// AcceleratedWidgetMac |
| -BrowserCompositorCALayerTreeMac::BrowserCompositorCALayerTreeMac() |
| - : view_(NULL), |
| - accelerated_output_surface_id_(0) { |
| +AcceleratedWidgetMac::AcceleratedWidgetMac() |
| + : client_(NULL), |
| + has_unacknowledged_accelerated_frame_(false) { |
| // Disable the fade-in animation as the layers are added. |
| ScopedCAActionDisabler disabler; |
| @@ -52,43 +63,35 @@ BrowserCompositorCALayerTreeMac::BrowserCompositorCALayerTreeMac() |
| last_sequence_number += 1; |
| native_widget_ = reinterpret_cast<gfx::AcceleratedWidget>( |
| last_sequence_number); |
| - g_widget_to_internals_map.Pointer()->insert( |
| + g_widget_to_helper_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); |
| } |
| -BrowserCompositorCALayerTreeMac::~BrowserCompositorCALayerTreeMac() { |
| - DCHECK(!view_); |
| - g_widget_to_internals_map.Pointer()->erase(native_widget_); |
| +AcceleratedWidgetMac::~AcceleratedWidgetMac() { |
| + DCHECK(!client_); |
| + g_widget_to_helper_map.Pointer()->erase(native_widget_); |
| } |
| -void BrowserCompositorCALayerTreeMac::SetView( |
| - BrowserCompositorViewMac* view) { |
| - // Disable the fade-in animation as the view is added. |
| +void AcceleratedWidgetMac::SetClient( |
| + AcceleratedWidgetMacClient* client) { |
|
tapted
2014/11/20 05:38:53
nit: pull up
ccameron
2014/11/20 21:33:50
Done.
|
| + // Disable the fade-in animation as the client is added. |
| ScopedCAActionDisabler disabler; |
| - DCHECK(view && !view_); |
| - view_ = view; |
| - compositor_->SetRootLayer(view_->ui_root_layer()); |
| + DCHECK(client && !client_); |
| + client_ = client; |
| - CALayer* background_layer = [view_->native_view() layer]; |
| + CALayer* background_layer = |
| + [client_->AcceleratedWidgetClientGetNSView() layer]; |
| DCHECK(background_layer); |
| [flipped_layer_ setBounds:[background_layer bounds]]; |
| [background_layer addSublayer:flipped_layer_]; |
| - compositor_->SetVisible(true); |
| } |
| -void BrowserCompositorCALayerTreeMac::ResetView() { |
| - if (!view_) |
| +void AcceleratedWidgetMac::ResetClient() { |
| + if (!client_) |
| return; |
| - // Disable the fade-out animation as the view is removed. |
| + // Disable the fade-out animation as the client is removed. |
| ScopedCAActionDisabler disabler; |
| [flipped_layer_ removeFromSuperlayer]; |
| @@ -96,53 +99,50 @@ void BrowserCompositorCALayerTreeMac::ResetView() { |
| 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); |
| - view_ = NULL; |
| + client_ = NULL; |
| } |
| -bool BrowserCompositorCALayerTreeMac::HasFrameOfSize( |
| +bool AcceleratedWidgetMac::HasFrameOfSize( |
| const gfx::Size& dip_size) const { |
| return last_swap_size_dip_ == dip_size; |
| } |
| -int BrowserCompositorCALayerTreeMac::GetRendererID() const { |
| +int AcceleratedWidgetMac::GetRendererID() const { |
| if (io_surface_layer_) |
| return [io_surface_layer_ rendererID]; |
| return 0; |
| } |
| -bool BrowserCompositorCALayerTreeMac::IsRendererThrottlingDisabled() const { |
| - if (view_) |
| - return view_->client()->BrowserCompositorViewShouldAckImmediately(); |
| +bool AcceleratedWidgetMac::IsRendererThrottlingDisabled() const { |
| + if (client_) |
| + return client_->AcceleratedWidgetShouldIgnoreBackpressure(); |
| return false; |
| } |
| -void BrowserCompositorCALayerTreeMac::BeginPumpingFrames() { |
| +void AcceleratedWidgetMac::BeginPumpingFrames() { |
| [io_surface_layer_ beginPumpingFrames]; |
| } |
| -void BrowserCompositorCALayerTreeMac::EndPumpingFrames() { |
| +void AcceleratedWidgetMac::EndPumpingFrames() { |
| [io_surface_layer_ endPumpingFrames]; |
| } |
| -void BrowserCompositorCALayerTreeMac::GotAcceleratedFrame( |
| - uint64 surface_handle, int output_surface_id, |
| +void AcceleratedWidgetMac::GotAcceleratedFrame( |
| + uint64 surface_handle, |
| const std::vector<ui::LatencyInfo>& latency_info, |
| - gfx::Size pixel_size, float scale_factor) { |
| + gfx::Size pixel_size, float scale_factor, |
| + const base::Closure& drawn_callback) { |
| // Record the surface and latency info to use when acknowledging this frame. |
| - DCHECK(!accelerated_output_surface_id_); |
| - accelerated_output_surface_id_ = output_surface_id; |
| + DCHECK(!has_unacknowledged_accelerated_frame_); |
| + has_unacknowledged_accelerated_frame_ = true; |
| + accelerated_frame_drawn_callback_ = drawn_callback; |
| accelerated_latency_info_.insert(accelerated_latency_info_.end(), |
| latency_info.begin(), latency_info.end()); |
| - // If there is no view and therefore no superview to draw into, early-out. |
| - if (!view_) { |
| - IOSurfaceLayerDidDrawFrame(); |
| + // If there is no client and therefore no superview to draw into, early-out. |
| + if (!client_) { |
| + AcknowledgeAcceleratedFrame(); |
| return; |
| } |
| @@ -167,7 +167,7 @@ void BrowserCompositorCALayerTreeMac::GotAcceleratedFrame( |
| } |
| } |
| -void BrowserCompositorCALayerTreeMac::GotAcceleratedCAContextFrame( |
| +void AcceleratedWidgetMac::GotAcceleratedCAContextFrame( |
| CAContextID ca_context_id, |
| gfx::Size pixel_size, |
| float scale_factor) { |
| @@ -188,7 +188,7 @@ void BrowserCompositorCALayerTreeMac::GotAcceleratedCAContextFrame( |
| // Acknowledge the frame to unblock the compositor immediately (the GPU |
| // process will do any required throttling). |
| - IOSurfaceLayerDidDrawFrame(); |
| + AcknowledgeAcceleratedFrame(); |
| // If this replacing a same-type layer, remove it now that the new layer is |
| // in the hierarchy. |
| @@ -200,7 +200,7 @@ void BrowserCompositorCALayerTreeMac::GotAcceleratedCAContextFrame( |
| DestroySoftwareLayer(); |
| } |
| -void BrowserCompositorCALayerTreeMac::GotAcceleratedIOSurfaceFrame( |
| +void AcceleratedWidgetMac::GotAcceleratedIOSurfaceFrame( |
| IOSurfaceID io_surface_id, |
| gfx::Size pixel_size, |
| float scale_factor) { |
| @@ -270,11 +270,11 @@ void BrowserCompositorCALayerTreeMac::GotAcceleratedIOSurfaceFrame( |
| DestroySoftwareLayer(); |
| } |
| -void BrowserCompositorCALayerTreeMac::GotSoftwareFrame( |
| +void AcceleratedWidgetMac::GotSoftwareFrame( |
| cc::SoftwareFrameData* frame_data, |
| float scale_factor, |
| SkCanvas* canvas) { |
| - if (!frame_data || !canvas || !view_) |
| + if (!frame_data || !canvas || !client_) |
| return; |
| // Disable the fade-in or fade-out effect if we create or remove layers. |
| @@ -302,7 +302,7 @@ void BrowserCompositorCALayerTreeMac::GotSoftwareFrame( |
| DestroyIOSurfaceLayer(io_surface_layer_); |
| } |
| -void BrowserCompositorCALayerTreeMac::DestroyCAContextLayer( |
| +void AcceleratedWidgetMac::DestroyCAContextLayer( |
| base::scoped_nsobject<CALayerHost> ca_context_layer) { |
| if (!ca_context_layer) |
| return; |
| @@ -311,7 +311,7 @@ void BrowserCompositorCALayerTreeMac::DestroyCAContextLayer( |
| ca_context_layer_.reset(); |
| } |
| -void BrowserCompositorCALayerTreeMac::DestroyIOSurfaceLayer( |
| +void AcceleratedWidgetMac::DestroyIOSurfaceLayer( |
| base::scoped_nsobject<IOSurfaceLayer> io_surface_layer) { |
| if (!io_surface_layer) |
| return; |
| @@ -321,86 +321,75 @@ void BrowserCompositorCALayerTreeMac::DestroyIOSurfaceLayer( |
| io_surface_layer_.reset(); |
| } |
| -void BrowserCompositorCALayerTreeMac::DestroySoftwareLayer() { |
| +void AcceleratedWidgetMac::DestroySoftwareLayer() { |
| if (!software_layer_) |
| return; |
| [software_layer_ removeFromSuperlayer]; |
| software_layer_.reset(); |
| } |
| -bool BrowserCompositorCALayerTreeMac::IOSurfaceLayerShouldAckImmediately() |
| +bool AcceleratedWidgetMac::IOSurfaceLayerShouldAckImmediately() |
| const { |
| - // If there is no view then the accelerated layer is not in the hierarchy |
| - // and will never draw. |
| - if (!view_) |
| + // If there is no client then the accelerated layer is not in the view |
| + // hierarchy and will never draw. |
| + if (!client_) |
| return true; |
| - return view_->client()->BrowserCompositorViewShouldAckImmediately(); |
| + return client_->AcceleratedWidgetShouldIgnoreBackpressure(); |
| } |
| -void BrowserCompositorCALayerTreeMac::IOSurfaceLayerDidDrawFrame() { |
| - if (accelerated_output_surface_id_) { |
| - content::ImageTransportFactory::GetInstance()->OnSurfaceDisplayed( |
| - accelerated_output_surface_id_); |
| - accelerated_output_surface_id_ = 0; |
| - } |
| - |
| - if (view_) { |
| - view_->client()->BrowserCompositorViewFrameSwapped( |
| - accelerated_latency_info_); |
| - } |
| +void AcceleratedWidgetMac::IOSurfaceLayerDidDrawFrame() { |
| + AcknowledgeAcceleratedFrame(); |
| +} |
| +void AcceleratedWidgetMac::AcknowledgeAcceleratedFrame() { |
| + if (!has_unacknowledged_accelerated_frame_) |
| + return; |
| + accelerated_frame_drawn_callback_.Run(); |
| + if (client_) |
| + client_->AcceleratedWidgetSwapCompleted(accelerated_latency_info_); |
| accelerated_latency_info_.clear(); |
| + has_unacknowledged_accelerated_frame_ = false; |
| } |
| -void BrowserCompositorCALayerTreeMac::IOSurfaceLayerHitError() { |
| +void AcceleratedWidgetMac::IOSurfaceLayerHitError() { |
| // Perform all acks that would have been done if the frame had succeeded, to |
| // un-block the compositor and renderer. |
| - IOSurfaceLayerDidDrawFrame(); |
| + AcknowledgeAcceleratedFrame(); |
| // Poison the context being used and request a mulligan. |
| [io_surface_layer_ poisonContextAndSharegroup]; |
| - compositor_->ScheduleFullRedraw(); |
| -} |
| -// static |
| -BrowserCompositorCALayerTreeMac* BrowserCompositorCALayerTreeMac:: |
| - 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; |
| + if (client_) |
| + client_->AcceleratedWidgetHitError(); |
| } |
| -void BrowserCompositorCALayerTreeMacGotAcceleratedFrame( |
| - gfx::AcceleratedWidget widget, |
| - uint64 surface_handle, int surface_id, |
| +void AcceleratedWidgetMacGotAcceleratedFrame( |
| + gfx::AcceleratedWidget widget, uint64 surface_handle, |
| const std::vector<ui::LatencyInfo>& latency_info, |
| gfx::Size pixel_size, float scale_factor, |
| + const base::Closure& drawn_callback, |
| bool* disable_throttling, int* renderer_id) { |
| - BrowserCompositorCALayerTreeMac* ca_layer_tree = |
| - BrowserCompositorCALayerTreeMac::FromAcceleratedWidget(widget); |
| - if (ca_layer_tree) { |
| - ca_layer_tree->GotAcceleratedFrame( |
| - surface_handle, surface_id, latency_info, pixel_size, scale_factor); |
| - *disable_throttling = ca_layer_tree->IsRendererThrottlingDisabled(); |
| - *renderer_id = ca_layer_tree->GetRendererID(); |
| + AcceleratedWidgetMac* accelerated_widget_mac = |
| + GetHelperFromAcceleratedWidget(widget); |
| + if (accelerated_widget_mac) { |
| + accelerated_widget_mac->GotAcceleratedFrame( |
| + surface_handle, latency_info, pixel_size, scale_factor, drawn_callback); |
| + *disable_throttling = |
| + accelerated_widget_mac->IsRendererThrottlingDisabled(); |
| + *renderer_id = accelerated_widget_mac->GetRendererID(); |
| } else { |
| *disable_throttling = false; |
| *renderer_id = 0; |
| } |
| } |
| -void BrowserCompositorCALayerTreeMacGotSoftwareFrame( |
| +void AcceleratedWidgetMacGotSoftwareFrame( |
| gfx::AcceleratedWidget widget, |
| cc::SoftwareFrameData* frame_data, float scale_factor, SkCanvas* canvas) { |
| - BrowserCompositorCALayerTreeMac* ca_layer_tree = |
| - BrowserCompositorCALayerTreeMac::FromAcceleratedWidget(widget); |
| - if (ca_layer_tree) |
| - ca_layer_tree->GotSoftwareFrame(frame_data, scale_factor, canvas); |
| + AcceleratedWidgetMac* accelerated_widget_mac = |
| + GetHelperFromAcceleratedWidget(widget); |
| + if (accelerated_widget_mac) |
| + accelerated_widget_mac->GotSoftwareFrame(frame_data, scale_factor, canvas); |
| } |
| } // namespace content |