// Copyright 2011 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. #ifndef CC_LAYER_TREE_HOST_IMPL_H_ #define CC_LAYER_TREE_HOST_IMPL_H_ #include "base/basictypes.h" #include "base/memory/scoped_ptr.h" #include "base/time.h" #include "cc/animation_events.h" #include "cc/animation_registrar.h" #include "cc/cc_export.h" #include "cc/input_handler.h" #include "cc/output_surface_client.h" #include "cc/pinch_zoom_viewport.h" #include "cc/render_pass.h" #include "cc/render_pass_sink.h" #include "cc/renderer.h" #include "cc/tile_manager.h" #include "skia/ext/refptr.h" #include "third_party/skia/include/core/SkColor.h" #include "third_party/skia/include/core/SkPicture.h" #include "ui/gfx/rect.h" namespace cc { class CompletionEvent; class CompositorFrameMetadata; class DebugRectHistory; class FrameRateCounter; class LayerImpl; class LayerTreeHostImplTimeSourceAdapter; class LayerTreeImpl; class PageScaleAnimation; class RenderPassDrawQuad; class ResourceProvider; struct RendererCapabilities; struct RenderingStats; // LayerTreeHost->Proxy callback interface. class LayerTreeHostImplClient { public: virtual void didLoseOutputSurfaceOnImplThread() = 0; virtual void onSwapBuffersCompleteOnImplThread() = 0; virtual void onVSyncParametersChanged(base::TimeTicks timebase, base::TimeDelta interval) = 0; virtual void onCanDrawStateChanged(bool canDraw) = 0; virtual void onHasPendingTreeStateChanged(bool hasPendingTree) = 0; virtual void setNeedsRedrawOnImplThread() = 0; virtual void setNeedsCommitOnImplThread() = 0; virtual void setNeedsManageTilesOnImplThread() = 0; virtual void postAnimationEventsToMainThreadOnImplThread(scoped_ptr, base::Time wallClockTime) = 0; // Returns true if resources were deleted by this call. virtual bool reduceContentsTextureMemoryOnImplThread(size_t limitBytes, int priorityCutoff) = 0; virtual void sendManagedMemoryStats() = 0; }; // LayerTreeHostImpl owns the LayerImpl tree as well as associated rendering state class CC_EXPORT LayerTreeHostImpl : public InputHandlerClient, public RendererClient, public TileManagerClient, public OutputSurfaceClient { typedef std::vector LayerList; public: static scoped_ptr create(const LayerTreeSettings&, LayerTreeHostImplClient*, Proxy*); virtual ~LayerTreeHostImpl(); // InputHandlerClient implementation virtual InputHandlerClient::ScrollStatus scrollBegin(gfx::Point, InputHandlerClient::ScrollInputType) OVERRIDE; virtual bool scrollBy(const gfx::Point&, const gfx::Vector2d&) OVERRIDE; virtual void scrollEnd() OVERRIDE; virtual void pinchGestureBegin() OVERRIDE; virtual void pinchGestureUpdate(float, gfx::Point) OVERRIDE; virtual void pinchGestureEnd() OVERRIDE; virtual void startPageScaleAnimation(gfx::Vector2d targetOffset, bool anchorPoint, float pageScale, base::TimeTicks startTime, base::TimeDelta duration) OVERRIDE; virtual void scheduleAnimation() OVERRIDE; virtual bool haveTouchEventHandlersAt(const gfx::Point&) OVERRIDE; struct CC_EXPORT FrameData : public RenderPassSink { FrameData(); ~FrameData(); std::vector occludingScreenSpaceRects; std::vector nonOccludingScreenSpaceRects; RenderPassList renderPasses; RenderPassIdHashMap renderPassesById; const LayerList* renderSurfaceLayerList; LayerList willDrawLayers; // RenderPassSink implementation. virtual void appendRenderPass(scoped_ptr) OVERRIDE; }; // Virtual for testing. virtual void beginCommit(); virtual void commitComplete(); virtual void animate(base::TimeTicks monotonicTime, base::Time wallClockTime); void manageTiles(); // Returns false if problems occured preparing the frame, and we should try // to avoid displaying the frame. If prepareToDraw is called, // didDrawAllLayers must also be called, regardless of whether drawLayers is // called between the two. virtual bool prepareToDraw(FrameData&); virtual void drawLayers(FrameData&); // Must be called if and only if prepareToDraw was called. void didDrawAllLayers(const FrameData&); // RendererClient implementation virtual const gfx::Size& deviceViewportSize() const OVERRIDE; virtual const LayerTreeSettings& settings() const OVERRIDE; virtual void didLoseOutputSurface() OVERRIDE; virtual void onSwapBuffersComplete() OVERRIDE; virtual void setFullRootLayerDamage() OVERRIDE; virtual void setManagedMemoryPolicy(const ManagedMemoryPolicy& policy) OVERRIDE; virtual void enforceManagedMemoryPolicy(const ManagedMemoryPolicy& policy) OVERRIDE; virtual bool hasImplThread() const OVERRIDE; virtual bool shouldClearRootRenderPass() const OVERRIDE; virtual CompositorFrameMetadata makeCompositorFrameMetadata() const OVERRIDE; // TileManagerClient implementation. virtual void ScheduleManageTiles() OVERRIDE; // OutputSurfaceClient implementation. virtual void OnVSyncParametersChanged(base::TimeTicks timebase, base::TimeDelta interval) OVERRIDE; virtual void OnSendFrameToParentCompositorAck(const CompositorFrameAck&) OVERRIDE; // Called from LayerTreeImpl. void OnCanDrawStateChangedForTree(LayerTreeImpl*); // Implementation bool canDraw(); OutputSurface* outputSurface() const; std::string layerTreeAsText() const; std::string layerTreeAsJson() const; void finishAllRendering(); int sourceAnimationFrameNumber() const; bool initializeRenderer(scoped_ptr); bool isContextLost(); TileManager* tileManager() { return m_tileManager.get(); } Renderer* renderer() { return m_renderer.get(); } const RendererCapabilities& rendererCapabilities() const; bool swapBuffers(); void readback(void* pixels, const gfx::Rect&); LayerTreeImpl* activeTree() { return m_activeTree.get(); } const LayerTreeImpl* activeTree() const { return m_activeTree.get(); } LayerTreeImpl* pendingTree() { return m_pendingTree.get(); } const LayerTreeImpl* pendingTree() const { return m_pendingTree.get(); } void createPendingTree(); void checkForCompletedSetPixels(); virtual void activatePendingTreeIfNeeded(); // Shortcuts to layers on the active tree. LayerImpl* rootLayer() const; LayerImpl* rootScrollLayer() const; LayerImpl* currentlyScrollingLayer() const; bool visible() const { return m_visible; } void setVisible(bool); bool contentsTexturesPurged() const { return m_contentsTexturesPurged; } void setContentsTexturesPurged(); void resetContentsTexturesPurged(); size_t memoryAllocationLimitBytes() const { return m_managedMemoryPolicy.bytesLimitWhenVisible; } void setViewportSize(const gfx::Size& layoutViewportSize, const gfx::Size& deviceViewportSize); const gfx::Size& layoutViewportSize() const { return m_layoutViewportSize; } float deviceScaleFactor() const { return m_deviceScaleFactor; } void setDeviceScaleFactor(float); float pageScaleFactor() const; void setPageScaleFactorAndLimits(float pageScaleFactor, float minPageScaleFactor, float maxPageScaleFactor); scoped_ptr processScrollDeltas(); gfx::Transform implTransform() const; void startPageScaleAnimation(gfx::Vector2d targetOffset, bool useAnchor, float scale, base::TimeDelta duration); bool needsAnimateLayers() const { return !m_animationRegistrar->active_animation_controllers().empty(); } bool needsUpdateDrawProperties() const { return m_needsUpdateDrawProperties; } void setNeedsUpdateDrawProperties() { m_needsUpdateDrawProperties = true; } void setNeedsRedraw(); void renderingStats(RenderingStats*) const; void sendManagedMemoryStats( size_t memoryVisibleBytes, size_t memoryVisibleAndNearbyBytes, size_t memoryUseBytes); FrameRateCounter* fpsCounter() const { return m_fpsCounter.get(); } DebugRectHistory* debugRectHistory() const { return m_debugRectHistory.get(); } ResourceProvider* resourceProvider() const { return m_resourceProvider.get(); } Proxy* proxy() const { return m_proxy; } AnimationRegistrar* animationRegistrar() const { return m_animationRegistrar.get(); } void setDebugState(const LayerTreeDebugState& debugState) { m_debugState = debugState; } const LayerTreeDebugState& debugState() const { return m_debugState; } class CC_EXPORT CullRenderPassesWithCachedTextures { public: bool shouldRemoveRenderPass(const RenderPassDrawQuad&, const FrameData&) const; // Iterates from the root first, in order to remove the surfaces closest // to the root with cached textures, and all surfaces that draw into // them. size_t renderPassListBegin(const RenderPassList& list) const { return list.size() - 1; } size_t renderPassListEnd(const RenderPassList&) const { return 0 - 1; } size_t renderPassListNext(size_t it) const { return it - 1; } CullRenderPassesWithCachedTextures(Renderer& renderer) : m_renderer(renderer) { } private: Renderer& m_renderer; }; class CC_EXPORT CullRenderPassesWithNoQuads { public: bool shouldRemoveRenderPass(const RenderPassDrawQuad&, const FrameData&) const; // Iterates in draw order, so that when a surface is removed, and its // target becomes empty, then its target can be removed also. size_t renderPassListBegin(const RenderPassList&) const { return 0; } size_t renderPassListEnd(const RenderPassList& list) const { return list.size(); } size_t renderPassListNext(size_t it) const { return it + 1; } }; template static void removeRenderPasses(RenderPassCuller, FrameData&); float totalPageScaleFactorForTesting() const { return m_pinchZoomViewport.total_page_scale_factor(); } const PinchZoomViewport& pinchZoomViewport() const { return m_pinchZoomViewport; } skia::RefPtr capturePicture(); bool pinchGestureActive() const { return m_pinchGestureActive; } protected: LayerTreeHostImpl(const LayerTreeSettings&, LayerTreeHostImplClient*, Proxy*); void activatePendingTree(); // Virtual for testing. virtual void animateLayers(base::TimeTicks monotonicTime, base::Time wallClockTime); // Virtual for testing. virtual base::TimeDelta lowFrequencyAnimationInterval() const; const AnimationRegistrar::AnimationControllerMap& activeAnimationControllers() const { return m_animationRegistrar->active_animation_controllers(); } LayerTreeHostImplClient* m_client; Proxy* m_proxy; private: void animatePageScale(base::TimeTicks monotonicTime); void animateScrollbars(base::TimeTicks monotonicTime); void updateDrawProperties(); void computeDoubleTapZoomDeltas(ScrollAndScaleSet* scrollInfo); void computePinchZoomDeltas(ScrollAndScaleSet* scrollInfo); void makeScrollAndScaleSet(ScrollAndScaleSet* scrollInfo, gfx::Vector2d scrollOffset, float pageScale); void setPageScaleDelta(float); void updateMaxScrollOffset(); void trackDamageForAllSurfaces(LayerImpl* rootDrawLayer, const LayerList& renderSurfaceLayerList); // Returns false if the frame should not be displayed. This function should // only be called from prepareToDraw, as didDrawAllLayers must be called // if this helper function is called. bool calculateRenderPasses(FrameData&); void animateLayersRecursive(LayerImpl*, base::TimeTicks monotonicTime, base::Time wallClockTime, AnimationEventsVector*, bool& didAnimate, bool& needsAnimateLayers); void setBackgroundTickingEnabled(bool); gfx::Size contentSize() const; void sendDidLoseOutputSurfaceRecursive(LayerImpl*); void clearRenderSurfaces(); bool ensureRenderSurfaceLayerList(); void clearCurrentlyScrollingLayer(); void animateScrollbarsRecursive(LayerImpl*, base::TimeTicks monotonicTime); void dumpRenderSurfaces(std::string*, int indent, const LayerImpl*) const; static LayerImpl* getNonCompositedContentLayerRecursive(LayerImpl* layer); scoped_ptr m_outputSurface; scoped_ptr m_resourceProvider; scoped_ptr m_renderer; scoped_ptr m_tileManager; scoped_ptr m_pendingTree; scoped_ptr m_activeTree; bool m_scrollDeltaIsInViewportSpace; LayerTreeSettings m_settings; LayerTreeDebugState m_debugState; gfx::Size m_layoutViewportSize; gfx::Size m_deviceViewportSize; float m_deviceScaleFactor; bool m_visible; bool m_contentsTexturesPurged; ManagedMemoryPolicy m_managedMemoryPolicy; bool m_needsUpdateDrawProperties; bool m_pinchGestureActive; gfx::Point m_previousPinchAnchor; scoped_ptr m_pageScaleAnimation; // This is used for ticking animations slowly when hidden. scoped_ptr m_timeSourceClientAdapter; PinchZoomViewport m_pinchZoomViewport; scoped_ptr m_fpsCounter; scoped_ptr m_debugRectHistory; int64 m_numImplThreadScrolls; int64 m_numMainThreadScrolls; int64 m_cumulativeNumLayersDrawn; int64 m_cumulativeNumMissingTiles; size_t m_lastSentMemoryVisibleBytes; size_t m_lastSentMemoryVisibleAndNearbyBytes; size_t m_lastSentMemoryUseBytes; scoped_ptr m_animationRegistrar; DISALLOW_COPY_AND_ASSIGN(LayerTreeHostImpl); }; } // namespace cc #endif // CC_LAYER_TREE_HOST_IMPL_H_