/* * Copyright (C) 2014 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ #import "config.h" #import "ViewSnapshotStore.h" #import "WebBackForwardList.h" #import "WebPageProxy.h" #import #import #if PLATFORM(IOS) #import #endif using namespace WebCore; #if USE_IOSURFACE_VIEW_SNAPSHOTS static const size_t maximumSnapshotCacheSize = 400 * (1024 * 1024); #elif USE_RENDER_SERVER_VIEW_SNAPSHOTS // Because render server snapshots are not purgeable, we should keep fewer around. static const size_t maximumSnapshotCacheSize = 50 * (1024 * 1024); #endif namespace WebKit { ViewSnapshotStore::ViewSnapshotStore() : m_snapshotCacheSize(0) { } ViewSnapshotStore::~ViewSnapshotStore() { discardSnapshotImages(); } ViewSnapshotStore& ViewSnapshotStore::shared() { static ViewSnapshotStore& store = *new ViewSnapshotStore; return store; } #if USE_RENDER_SERVER_VIEW_SNAPSHOTS CAContext *ViewSnapshotStore::snapshottingContext() { static CAContext *context; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ NSDictionary *options = @{ kCAContextDisplayName: @"WebKitSnapshotting", kCAContextIgnoresHitTest: @YES, kCAContextDisplayId : @20000 }; context = [[CAContext remoteContextWithOptions:options] retain]; }); return context; } #endif void ViewSnapshotStore::didAddImageToSnapshot(ViewSnapshot& snapshot) { bool isNewEntry = m_snapshotsWithImages.add(&snapshot).isNewEntry; ASSERT_UNUSED(isNewEntry, isNewEntry); m_snapshotCacheSize += snapshot.imageSizeInBytes(); } void ViewSnapshotStore::willRemoveImageFromSnapshot(ViewSnapshot& snapshot) { bool removed = m_snapshotsWithImages.remove(&snapshot); ASSERT_UNUSED(removed, removed); m_snapshotCacheSize -= snapshot.imageSizeInBytes(); } void ViewSnapshotStore::pruneSnapshots(WebPageProxy& webPageProxy) { if (m_snapshotCacheSize <= maximumSnapshotCacheSize) return; ASSERT(!m_snapshotsWithImages.isEmpty()); // FIXME: We have enough information to do smarter-than-LRU eviction (making use of the back-forward lists, etc.) m_snapshotsWithImages.first()->clearImage(); } void ViewSnapshotStore::recordSnapshot(WebPageProxy& webPageProxy) { if (webPageProxy.isShowingNavigationGestureSnapshot()) return; WebBackForwardListItem* item = webPageProxy.backForwardList().currentItem(); if (!item) return; pruneSnapshots(webPageProxy); webPageProxy.willRecordNavigationSnapshot(*item); RefPtr snapshot = webPageProxy.takeViewSnapshot(); if (!snapshot || !snapshot->hasImage()) return; snapshot->setRenderTreeSize(webPageProxy.renderTreeSize()); snapshot->setDeviceScaleFactor(webPageProxy.deviceScaleFactor()); snapshot->setBackgroundColor(webPageProxy.pageExtendedBackgroundColor()); item->setSnapshot(snapshot.release()); } void ViewSnapshotStore::discardSnapshotImages() { while (!m_snapshotsWithImages.isEmpty()) m_snapshotsWithImages.first()->clearImage(); } #if USE_IOSURFACE_VIEW_SNAPSHOTS PassRefPtr ViewSnapshot::create(IOSurface* surface, IntSize size, size_t imageSizeInBytes) { return adoptRef(new ViewSnapshot(surface, size, imageSizeInBytes)); } #elif USE_RENDER_SERVER_VIEW_SNAPSHOTS PassRefPtr ViewSnapshot::create(uint32_t slotID, IntSize size, size_t imageSizeInBytes) { return adoptRef(new ViewSnapshot(slotID, size, imageSizeInBytes)); } #endif #if USE_IOSURFACE_VIEW_SNAPSHOTS ViewSnapshot::ViewSnapshot(IOSurface* surface, IntSize size, size_t imageSizeInBytes) : m_surface(surface) #elif USE_RENDER_SERVER_VIEW_SNAPSHOTS ViewSnapshot::ViewSnapshot(uint32_t slotID, IntSize size, size_t imageSizeInBytes) : m_slotID(slotID) #endif , m_imageSizeInBytes(imageSizeInBytes) , m_size(size) { if (hasImage()) ViewSnapshotStore::shared().didAddImageToSnapshot(*this); } ViewSnapshot::~ViewSnapshot() { clearImage(); } bool ViewSnapshot::hasImage() const { #if USE_IOSURFACE_VIEW_SNAPSHOTS return m_surface; #elif USE_RENDER_SERVER_VIEW_SNAPSHOTS return m_slotID; #endif } void ViewSnapshot::clearImage() { if (!hasImage()) return; ViewSnapshotStore::shared().willRemoveImageFromSnapshot(*this); #if USE_IOSURFACE_VIEW_SNAPSHOTS m_surface = nullptr; #elif USE_RENDER_SERVER_VIEW_SNAPSHOTS [ViewSnapshotStore::snapshottingContext() deleteSlot:m_slotID]; m_slotID = 0; #endif m_imageSizeInBytes = 0; } id ViewSnapshot::asLayerContents() { #if USE_IOSURFACE_VIEW_SNAPSHOTS if (!m_surface) return nullptr; if (m_surface->setIsVolatile(false) != IOSurface::SurfaceState::Valid) { clearImage(); return nullptr; } return (id)m_surface->surface(); #elif USE_RENDER_SERVER_VIEW_SNAPSHOTS return [CAContext objectForSlot:m_slotID]; #endif } } // namespace WebKit