Changeset 156291 in webkit
- Timestamp:
- Sep 23, 2013, 1:43:17 PM (12 years ago)
- Location:
- trunk/Source/WebCore
- Files:
-
- 10 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/Source/WebCore/ChangeLog
r156289 r156291 1 2013-09-23 Simon Fraser <simon.fraser@apple.com> 2 3 Rapidly loading pages can result in !needsLayout() assertions in FrameView::paintContents() and RenderView::paint() 4 https://bugs.webkit.org/show_bug.cgi?id=121725 5 <rdar://problem/13982161> ASSERT(!needsLayout()) in WebCore::RenderView::paint (121196) 6 7 Reviewed by Dean Jackson. 8 9 Layout has to be up-to-date before we paint, and when composited we rely on a 10 CFRunLoopObserver owned by LayerFlushScheduler to enforce this. The contract is 11 that changes to GraphicsLayers accumulate in those layers, and are all pushed to 12 CALayers only inside of flushCompositingState(). CALayers must not be touched 13 directly outside of flushCompositingState(), since this may trigger a CA commit 14 and resultant painting without us having updated layout. 15 16 In addition, we rely on the CA commits only happening after our runloop observer 17 has fired in order for LayerFlushScheduler's suspend/resume to work (these are used 18 for freezing the layer tree during page transitions). 19 20 However, TileController was violating this contract. It called revalidateTiles() 21 at random times, and on a timer. This would result in CA commits happening when 22 layer flushing was suspended at the LayerFlushScheduler level, and thus before 23 we'd had a chance to do layout. 24 25 Fix by only ever revalidatingTiles inside of flushCompositingState(). Calls 26 that used to result in revalidateTiles() at other times now just set a bit 27 on GraphicsLayerCA to indicate that revalidation is required, and this triggers 28 a later flush like any other GraphicsLayerCA change. 29 30 There is one exception to the revalidate-inside-flush rule: when purging 31 files for background tabs on a timer, we are only dealing with an unparented 32 layer tree, so CALayer manipulation is safe at any time. 33 34 Very timing dependent, so hard to test. 35 36 * platform/graphics/TiledBacking.h: Expose revalidateTiles() 37 * platform/graphics/avfoundation/cf/MediaPlayerPrivateAVFoundationCF.cpp: Remove 38 platformCALayerDidCreateTiles() override. 39 * platform/graphics/ca/GraphicsLayerCA.h: Keep track of m_isCommittingChanges 40 so we can assert. Add TilingAreaChanged flag. 41 * platform/graphics/ca/GraphicsLayerCA.cpp: 42 (WebCore::GraphicsLayerCA::GraphicsLayerCA): 43 (WebCore::GraphicsLayerCA::recursiveCommitChanges): noteSublayersChanged() is 44 used inside of a commit in some cases to trigger a parent layer to update 45 children after the children have been processed. To avoid trying to pointlessly 46 schedule the flusher inside a flush, and for strong assertions, use a new flag 47 that indicates that flushing is not required in this situation. 48 Set the m_isCommittingChanges flag around committing state. 49 (WebCore::GraphicsLayerCA::platformCALayerSetNeedsToRevalidateTiles): We used 50 to use platformCALayerDidCreateTiles() to ensure that we would flush after adding 51 new tiles to avoid garbage flashes. Now that we only ever create tiles inside 52 a flush, this is no longer necessary. 53 The new platformCALayerSetNeedsToRevalidateTiles() function just sets the 54 TilingAreaChanged flag and triggers a flush. 55 (WebCore::GraphicsLayerCA::commitLayerChangesBeforeSublayers): After updating 56 visible rect, we look for the TilingAreaChanged flag to tell us whether to 57 revalidate TiledBacking tiles. 58 (WebCore::GraphicsLayerCA::ensureStructuralLayer): Use DontScheduleFlush when 59 telling a parent layer that it needs to reconsider sublayers. 60 (WebCore::GraphicsLayerCA::updateTiles): The caller of revalidateTiles(). 61 (WebCore::GraphicsLayerCA::updateContentsRects): DontScheduleFlush again. 62 (WebCore::GraphicsLayerCA::noteSublayersChanged): Pass scheduleFlush along. 63 (WebCore::GraphicsLayerCA::noteLayerPropertyChanged): Handle scheduleFlush. 64 * platform/graphics/ca/PlatformCALayerClient.h: 65 (WebCore::PlatformCALayerClient::platformCALayerSetNeedsToRevalidateTiles): Added 66 so that TileController can call back out to GraphicsLayerCA. 67 (WebCore::PlatformCALayerClient::isCommittingChanges): 68 * platform/graphics/ca/mac/TileController.h: Need to keep track of secondary 69 coverage rects in m_secondaryTileCoverageRects. 70 * platform/graphics/ca/mac/TileController.mm: 71 (WebCore::TileController::tileCacheLayerBoundsChanged): Just trigger a tile revalidate, 72 which will happen later in this same commit. 73 (WebCore::TileController::setScale): This is called inside the commit. Does an extra 74 revalidate, which is needed to prune tiles. This could be optimized in future. 75 (WebCore::TileController::setVisibleRect): Assert that we're inside a commit, 76 and trigger a revalidate. 77 (WebCore::TileController::setExposedRect): Trigger a revalidate. 78 (WebCore::TileController::setClipsToExposedRect): Ditto. 79 (WebCore::TileController::prepopulateRect): Save the rect and trigger a revalidate. 80 (WebCore::TileController::setIsInWindow): If we're in the window, trigger a revalidate 81 otherwise schedule the timer. 82 (WebCore::TileController::revalidateTiles): Wrapper for the real revalidateTiles(). 83 (WebCore::TileController::tileRevalidationTimerFired): If we're in the window, 84 schedule a revalidate. Otherwise our layer tree is disconnected, and it's OK to call revalidateTiles(). 85 (WebCore::TileController::setNeedsRevalidateTiles): Call through the client to GraphicsLayerCA 86 which will schedule a layer tree flush. 87 (WebCore::TileController::revalidateTiles): 88 (WebCore::TileController::ensureTilesForRect): Move code from prepopulateRect 89 into here; ensureTilesForRect() for each secondary rect, then clear the rects. 90 No need to call platformCALayerDidCreateTiles() at the end. 91 * platform/graphics/win/MediaPlayerPrivateQuickTimeVisualContext.cpp: Remove 92 platformCALayerDidCreateTiles() override. 93 * rendering/RenderLayerCompositor.cpp: 94 (WebCore::RenderLayerCompositor::scheduleLayerFlush): This should only be 95 called when we're not inside a flush, otherwise it will fail to schedule 96 another one. 97 1 98 2013-09-23 Sam Weinig <sam@webkit.org> 2 99 -
trunk/Source/WebCore/platform/graphics/TiledBacking.h
r147058 r156291 71 71 virtual IntSize tileSize() const = 0; 72 72 73 virtual void revalidateTiles() = 0; 73 74 virtual void forceRepaint() = 0; 74 75 -
trunk/Source/WebCore/platform/graphics/avfoundation/cf/MediaPlayerPrivateAVFoundationCF.cpp
r155127 r156291 194 194 virtual bool platformCALayerDrawsContent() const { return false; } 195 195 virtual void platformCALayerLayerDidDisplay(PlatformLayer*) { } 196 virtual void platformCALayerDidCreateTiles(const Vector<FloatRect>&) { }197 196 virtual float platformCALayerDeviceScaleFactor() { return 1; } 198 197 -
trunk/Source/WebCore/platform/graphics/ca/GraphicsLayerCA.cpp
r155998 r156291 45 45 #include <limits.h> 46 46 #include <wtf/CurrentTime.h> 47 #include <wtf/TemporaryChange.h> 47 48 #include <wtf/text/WTFString.h> 48 49 … … 288 289 , m_rootRelativeScaleFactor(1) 289 290 , m_uncommittedChanges(0) 291 , m_isCommittingChanges(false) 290 292 { 291 293 PlatformCALayer::LayerType layerType = PlatformCALayer::LayerTypeWebLayer; … … 1104 1106 m_visibleTileWashLayer->setBorderWidth(8); 1105 1107 m_visibleTileWashLayer->setBackgroundColor(washFillColor); 1106 noteSublayersChanged( );1108 noteSublayersChanged(DontScheduleFlush); 1107 1109 } 1108 1110 … … 1124 1126 1125 1127 TransformationMatrix transformFromRoot = rootRelativeTransformForScaling; 1126 commitLayerChangesBeforeSublayers(childCommitState, pageScaleFactor, baseRelativePosition, oldVisibleRect, &transformFromRoot); 1128 { 1129 TemporaryChange<bool> committingChangesChange(m_isCommittingChanges, true); 1130 commitLayerChangesBeforeSublayers(childCommitState, pageScaleFactor, baseRelativePosition, oldVisibleRect, &transformFromRoot); 1131 } 1127 1132 1128 1133 if (isRunningTransformAnimation()) { … … 1150 1155 static_cast<GraphicsLayerCA*>(m_maskLayer)->commitLayerChangesAfterSublayers(childCommitState); 1151 1156 1152 commitLayerChangesAfterSublayers(childCommitState); 1157 { 1158 TemporaryChange<bool> committingChangesChange(m_isCommittingChanges, true); 1159 commitLayerChangesAfterSublayers(childCommitState); 1160 } 1153 1161 1154 1162 if (affectedByTransformAnimation && client() && m_layer->layerType() == PlatformCALayer::LayerTypeTiledBackingLayer) … … 1174 1182 } 1175 1183 1176 void GraphicsLayerCA::platformCALayerDidCreateTiles(const Vector<FloatRect>& dirtyRects) 1177 { 1178 ASSERT(m_layer->usesTiledBackingLayer()); 1179 1180 for (size_t i = 0; i < dirtyRects.size(); ++i) 1181 setNeedsDisplayInRect(dirtyRects[i]); 1182 1183 noteLayerPropertyChanged(TilesAdded); 1184 void GraphicsLayerCA::platformCALayerSetNeedsToRevalidateTiles() 1185 { 1186 noteLayerPropertyChanged(TilingAreaChanged, m_isCommittingChanges ? DontScheduleFlush : ScheduleFlush); 1184 1187 } 1185 1188 … … 1273 1276 if (m_uncommittedChanges & VisibleRectChanged) 1274 1277 updateVisibleRect(oldVisibleRect); 1278 1279 if (m_uncommittedChanges & TilingAreaChanged) // Needs to happen after VisibleRectChanged, ContentsScaleChanged 1280 updateTiles(); 1275 1281 1276 1282 if (m_uncommittedChanges & DirtyRectsChanged) … … 1621 1627 // We've changed the layer that our parent added to its sublayer list, so tell it to update 1622 1628 // sublayers again in its commitLayerChangesAfterSublayers(). 1623 static_cast<GraphicsLayerCA*>(parent())->noteSublayersChanged( );1629 static_cast<GraphicsLayerCA*>(parent())->noteSublayersChanged(DontScheduleFlush); 1624 1630 1625 1631 // Set properties of m_layer to their default values, since these are expressed on on the structural layer. … … 1756 1762 } 1757 1763 1764 void GraphicsLayerCA::updateTiles() 1765 { 1766 if (!m_layer->usesTiledBackingLayer()) 1767 return; 1768 1769 tiledBacking()->revalidateTiles(); 1770 } 1771 1758 1772 void GraphicsLayerCA::updateBackgroundColor() 1759 1773 { … … 1882 1896 1883 1897 if (gainedOrLostClippingLayer) 1884 noteSublayersChanged( );1898 noteSublayersChanged(DontScheduleFlush); 1885 1899 1886 1900 m_contentsLayer->setPosition(contentOrigin); … … 3265 3279 } 3266 3280 3267 void GraphicsLayerCA::noteSublayersChanged( )3268 { 3269 noteLayerPropertyChanged(ChildrenChanged );3281 void GraphicsLayerCA::noteSublayersChanged(ScheduleFlushOrNot scheduleFlush) 3282 { 3283 noteLayerPropertyChanged(ChildrenChanged, scheduleFlush); 3270 3284 propagateLayerChangeToReplicas(); 3271 3285 } … … 3277 3291 } 3278 3292 3279 void GraphicsLayerCA::noteLayerPropertyChanged(LayerChangeFlags flags )3293 void GraphicsLayerCA::noteLayerPropertyChanged(LayerChangeFlags flags, ScheduleFlushOrNot scheduleFlush) 3280 3294 { 3281 3295 bool hadUncommittedChanges = !!m_uncommittedChanges; … … 3284 3298 m_uncommittedChanges |= flags; 3285 3299 3286 bool needsFlush = !hadUncommittedChanges || oldCanThrottleLayerFlush != canThrottleLayerFlush(); 3287 if (needsFlush && m_client) 3288 m_client->notifyFlushRequired(this); 3300 if (scheduleFlush == ScheduleFlush) { 3301 bool needsFlush = !hadUncommittedChanges || oldCanThrottleLayerFlush != canThrottleLayerFlush(); 3302 if (needsFlush && m_client) 3303 m_client->notifyFlushRequired(this); 3304 } 3289 3305 } 3290 3306 -
trunk/Source/WebCore/platform/graphics/ca/GraphicsLayerCA.h
r155998 r156291 173 173 virtual bool platformCALayerDrawsContent() const { return drawsContent(); } 174 174 virtual void platformCALayerLayerDidDisplay(PlatformLayer* layer) { return layerDidDisplay(layer); } 175 virtual void platformCALayer DidCreateTiles(const Vector<FloatRect>& dirtyRects) OVERRIDE;175 virtual void platformCALayerSetNeedsToRevalidateTiles() OVERRIDE; 176 176 virtual float platformCALayerDeviceScaleFactor() OVERRIDE; 177 virtual bool isCommittingChanges() const OVERRIDE { return m_isCommittingChanges; } 177 178 178 179 virtual double backingStoreMemoryEstimate() const; … … 356 357 void updateDebugBorder(); 357 358 void updateVisibleRect(const FloatRect& oldVisibleRect); 359 void updateTiles(); 358 360 void updateContentsScale(float pageScaleFactor); 359 361 … … 411 413 VisibleRectChanged = 1 << 26, 412 414 FiltersChanged = 1 << 27, 413 TilesAdded = 1 < 28, 414 DebugIndicatorsChanged = 1 << 29 415 TilingAreaChanged = 1 << 28, 416 TilesAdded = 1 < 29, 417 DebugIndicatorsChanged = 1 << 30 415 418 }; 416 419 typedef unsigned LayerChangeFlags; 417 void noteLayerPropertyChanged(LayerChangeFlags flags); 418 void noteSublayersChanged(); 420 enum ScheduleFlushOrNot { ScheduleFlush, DontScheduleFlush }; 421 void noteLayerPropertyChanged(LayerChangeFlags, ScheduleFlushOrNot = ScheduleFlush); 422 void noteSublayersChanged(ScheduleFlushOrNot = ScheduleFlush); 419 423 void noteChangesForScaleSensitiveProperties(); 420 424 … … 506 510 507 511 LayerChangeFlags m_uncommittedChanges; 512 bool m_isCommittingChanges; 508 513 }; 509 514 -
trunk/Source/WebCore/platform/graphics/ca/PlatformCALayerClient.h
r130676 r156291 60 60 virtual void platformCALayerLayerDidDisplay(PlatformLayer*) = 0; 61 61 62 virtual void platformCALayer DidCreateTiles(const Vector<FloatRect>& dirtyRects) = 0;62 virtual void platformCALayerSetNeedsToRevalidateTiles() { } 63 63 virtual float platformCALayerDeviceScaleFactor() = 0; 64 65 virtual bool isCommittingChanges() const { return false; } 64 66 65 67 protected: -
trunk/Source/WebCore/platform/graphics/ca/mac/TileController.h
r154415 r156291 116 116 virtual void setTileCoverage(TileCoverage) OVERRIDE; 117 117 virtual TileCoverage tileCoverage() const OVERRIDE { return m_tileCoverage; } 118 virtual void revalidateTiles() OVERRIDE; 118 119 virtual void forceRepaint() OVERRIDE; 119 120 virtual IntSize tileSize() const OVERRIDE { return m_tileSize; } … … 146 147 typedef unsigned TileValidationPolicyFlags; 147 148 148 void revalidateTiles(TileValidationPolicyFlags foregroundValidationPolicy = 0, TileValidationPolicyFlags backgroundValidationPolicy = 0); 149 void setNeedsRevalidateTiles(); 150 void revalidateTiles(TileValidationPolicyFlags foregroundValidationPolicy, TileValidationPolicyFlags backgroundValidationPolicy); 149 151 enum class CoverageType { PrimaryTiles, SecondaryTiles }; 150 152 … … 180 182 FloatRect m_exposedRect; // The exposed area of containing platform views. 181 183 IntRect m_boundsAtLastRevalidate; 184 185 Vector<FloatRect> m_secondaryTileCoverageRects; 182 186 183 187 typedef HashMap<TileIndex, TileInfo> TileMap; -
trunk/Source/WebCore/platform/graphics/ca/mac/TileController.mm
r154415 r156291 136 136 void TileController::tileCacheLayerBoundsChanged() 137 137 { 138 if (m_tiles.isEmpty()) { 139 // We must revalidate immediately instead of using a timer when there are 140 // no tiles to avoid a flash when transitioning from one page to another. 141 revalidateTiles(); 142 return; 143 } 144 145 scheduleTileRevalidation(0); 138 ASSERT(PlatformCALayer::platformCALayer(m_tileCacheLayer)->owner()->isCommittingChanges()); 139 setNeedsRevalidateTiles(); 146 140 } 147 141 … … 233 227 { 234 228 PlatformCALayer* platformLayer = PlatformCALayer::platformCALayer(m_tileCacheLayer); 229 ASSERT(platformLayer->owner()->isCommittingChanges()); 230 235 231 float deviceScaleFactor = platformLayer->owner()->platformCALayerDeviceScaleFactor(); 236 232 … … 248 244 [m_tileContainerLayer.get() setTransform:CATransform3DMakeScale(1 / m_scale, 1 / m_scale, 1)]; 249 245 246 // FIXME: we may revalidateTiles twice in this commit. 250 247 revalidateTiles(PruneSecondaryTiles, PruneSecondaryTiles); 251 248 … … 260 257 dirtyRects.append(scaledTileRect); 261 258 } 262 263 platformLayer->owner()->platformCALayerDidCreateTiles(dirtyRects);264 259 } 265 260 … … 292 287 void TileController::setVisibleRect(const FloatRect& visibleRect) 293 288 { 289 ASSERT(PlatformCALayer::platformCALayer(m_tileCacheLayer)->owner()->isCommittingChanges()); 294 290 if (m_visibleRect == visibleRect) 295 291 return; 296 292 297 293 m_visibleRect = visibleRect; 298 revalidateTiles();294 setNeedsRevalidateTiles(); 299 295 } 300 296 … … 334 330 335 331 m_exposedRect = exposedRect; 336 revalidateTiles();332 setNeedsRevalidateTiles(); 337 333 } 338 334 … … 346 342 // Going from not clipping to clipping, we don't need to revalidate right away. 347 343 if (clipsToExposedRect) 348 revalidateTiles();344 setNeedsRevalidateTiles(); 349 345 } 350 346 … … 357 353 if (m_primaryTileCoverageRect.contains(rectInTileCoords)) 358 354 return; 359 360 ensureTilesForRect(rect, CoverageType::SecondaryTiles); 361 362 if (m_tiledScrollingIndicatorLayer) 363 updateTileCoverageMap(); 355 356 m_secondaryTileCoverageRects.append(rect); 357 setNeedsRevalidateTiles(); 364 358 } 365 359 … … 372 366 373 367 if (m_isInWindow) 374 revalidateTiles();368 setNeedsRevalidateTiles(); 375 369 else { 376 370 const double tileRevalidationTimeout = 4; … … 385 379 386 380 m_tileCoverage = coverage; 387 scheduleTileRevalidation(0); 381 setNeedsRevalidateTiles(); 382 } 383 384 void TileController::revalidateTiles() 385 { 386 ASSERT(PlatformCALayer::platformCALayer(m_tileCacheLayer)->owner()->isCommittingChanges()); 387 revalidateTiles(0, 0); 388 388 } 389 389 … … 511 511 void TileController::tileRevalidationTimerFired(Timer<TileController>*) 512 512 { 513 if (m_isInWindow) { 514 setNeedsRevalidateTiles(); 515 return; 516 } 517 513 518 TileValidationPolicyFlags foregroundValidationPolicy = m_aggressivelyRetainsTiles ? 0 : PruneSecondaryTiles; 514 519 TileValidationPolicyFlags backgroundValidationPolicy = foregroundValidationPolicy | UnparentAllTiles; … … 603 608 LayerPool::sharedPool()->addLayer(tileInfo.layer); 604 609 } 610 } 611 612 void TileController::setNeedsRevalidateTiles() 613 { 614 PlatformCALayer* platformLayer = PlatformCALayer::platformCALayer(m_tileCacheLayer); 615 platformLayer->owner()->platformCALayerSetNeedsToRevalidateTiles(); 605 616 } 606 617 … … 672 683 scheduleCohortRemoval(); 673 684 } 674 685 675 686 // Ensure primary tile coverage tiles. 676 687 m_primaryTileCoverageRect = ensureTilesForRect(tileCoverageRect, CoverageType::PrimaryTiles); … … 679 690 removeAllSecondaryTiles(); 680 691 m_cohortList.clear(); 692 } else { 693 for (size_t i = 0; i < m_secondaryTileCoverageRects.size(); ++i) 694 ensureTilesForRect(m_secondaryTileCoverageRects[i], CoverageType::SecondaryTiles); 695 m_secondaryTileCoverageRects.clear(); 681 696 } 682 697 … … 834 849 if (tilesInCohort) 835 850 startedNewCohort(currCohort); 836 837 // This will ensure we flush compositing state and do layout in this run loop iteration.838 if (!dirtyRects.isEmpty())839 platformLayer->owner()->platformCALayerDidCreateTiles(dirtyRects);840 851 841 852 return coverageRect; -
trunk/Source/WebCore/platform/graphics/win/MediaPlayerPrivateQuickTimeVisualContext.cpp
r155130 r156291 109 109 virtual bool platformCALayerDrawsContent() const { return false; } 110 110 virtual void platformCALayerLayerDidDisplay(PlatformLayer*) { } 111 virtual void platformCALayerDidCreateTiles(const Vector<FloatRect>&) { }112 111 virtual float platformCALayerDeviceScaleFactor() { return 1; } 113 112 -
trunk/Source/WebCore/rendering/RenderLayerCompositor.cpp
r156201 r156291 356 356 void RenderLayerCompositor::scheduleLayerFlush(bool canThrottle) 357 357 { 358 ASSERT(!m_flushingLayers); 359 358 360 if (canThrottle && isThrottlingLayerFlushes()) { 359 361 m_hasPendingLayerFlush = true;
Note:
See TracChangeset
for help on using the changeset viewer.