/* * Copyright (C) 2010 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 "WebContext.h" #import "NetworkProcessManager.h" #import "PluginProcessManager.h" #import "SharedWorkerProcessManager.h" #import "WKBrowsingContextControllerInternal.h" #import "WebKitSystemInterface.h" #import "WebProcessCreationParameters.h" #import "WebProcessMessages.h" #import #import #include #import #import #if HAVE(HOSTED_CORE_ANIMATION) && __MAC_OS_X_VERSION_MIN_REQUIRED >= 1070 #import #endif using namespace WebCore; NSString *WebDatabaseDirectoryDefaultsKey = @"WebDatabaseDirectory"; NSString *WebKitLocalCacheDefaultsKey = @"WebKitLocalCache"; NSString *WebStorageDirectoryDefaultsKey = @"WebKitLocalStorageDatabasePathPreferenceKey"; NSString *WebKitKerningAndLigaturesEnabledByDefaultDefaultsKey = @"WebKitKerningAndLigaturesEnabledByDefault"; static NSString *WebKitApplicationDidChangeAccessibilityEnhancedUserInterfaceNotification = @"NSApplicationDidChangeAccessibilityEnhancedUserInterfaceNotification"; // FIXME: - After this "backwards compatibility" radar is removed, this code should be removed to only return an empty String. NSString *WebIconDatabaseDirectoryDefaultsKey = @"WebIconDatabaseDirectoryDefaultsKey"; namespace WebKit { NSString *SchemeForCustomProtocolRegisteredNotificationName = @"WebKitSchemeForCustomProtocolRegisteredNotification"; NSString *SchemeForCustomProtocolUnregisteredNotificationName = @"WebKitSchemeForCustomProtocolUnregisteredNotification"; bool WebContext::s_applicationIsOccluded = false; String WebContext::applicationCacheDirectory() { NSString *appName = [[NSBundle mainBundle] bundleIdentifier]; if (!appName) appName = [[NSProcessInfo processInfo] processName]; ASSERT(appName); NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults]; NSString *cacheDir = [defaults objectForKey:WebKitLocalCacheDefaultsKey]; if (!cacheDir || ![cacheDir isKindOfClass:[NSString class]]) { char cacheDirectory[MAXPATHLEN]; size_t cacheDirectoryLen = confstr(_CS_DARWIN_USER_CACHE_DIR, cacheDirectory, MAXPATHLEN); if (cacheDirectoryLen) cacheDir = [[NSFileManager defaultManager] stringWithFileSystemRepresentation:cacheDirectory length:cacheDirectoryLen - 1]; } return [cacheDir stringByAppendingPathComponent:appName]; } static void registerUserDefaultsIfNeeded() { static bool didRegister; if (didRegister) return; didRegister = true; #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1090 [[NSUserDefaults standardUserDefaults] registerDefaults:[NSDictionary dictionaryWithObject:[NSNumber numberWithBool:YES] forKey:WebKitKerningAndLigaturesEnabledByDefaultDefaultsKey]]; #endif } void WebContext::platformInitializeWebProcess(WebProcessCreationParameters& parameters) { parameters.presenterApplicationPid = getpid(); parameters.parentProcessName = [[NSProcessInfo processInfo] processName]; NSURLCache *urlCache = [NSURLCache sharedURLCache]; parameters.nsURLCacheMemoryCapacity = [urlCache memoryCapacity]; parameters.nsURLCacheDiskCapacity = [urlCache diskCapacity]; registerUserDefaultsIfNeeded(); #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1090 parameters.shouldForceScreenFontSubstitution = [[NSUserDefaults standardUserDefaults] boolForKey:@"NSFontDefaultScreenFontSubstitutionEnabled"]; #endif parameters.shouldEnableKerningAndLigaturesByDefault = [[NSUserDefaults standardUserDefaults] boolForKey:WebKitKerningAndLigaturesEnabledByDefaultDefaultsKey]; #if USE(ACCELERATED_COMPOSITING) && HAVE(HOSTED_CORE_ANIMATION) #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1070 mach_port_t renderServerPort = [[CARemoteLayerServer sharedServer] serverPort]; #else mach_port_t renderServerPort = WKInitializeRenderServer(); #endif if (renderServerPort != MACH_PORT_NULL) parameters.acceleratedCompositingPort = CoreIPC::MachPort(renderServerPort, MACH_MSG_TYPE_COPY_SEND); #endif // FIXME: This should really be configurable; we shouldn't just blindly allow read access to the UI process bundle. parameters.uiProcessBundleResourcePath = [[NSBundle mainBundle] resourcePath]; SandboxExtension::createHandle(parameters.uiProcessBundleResourcePath, SandboxExtension::ReadOnly, parameters.uiProcessBundleResourcePathExtensionHandle); parameters.uiProcessBundleIdentifier = String([[NSBundle mainBundle] bundleIdentifier]); NSArray *schemes = [[WKBrowsingContextController customSchemes] allObjects]; for (size_t i = 0; i < [schemes count]; ++i) parameters.urlSchemesRegisteredForCustomProtocols.append([schemes objectAtIndex:i]); m_customSchemeRegisteredObserver = [[NSNotificationCenter defaultCenter] addObserverForName:WebKit::SchemeForCustomProtocolRegisteredNotificationName object:nil queue:[NSOperationQueue currentQueue] usingBlock:^(NSNotification *notification) { NSString *scheme = [notification object]; ASSERT([scheme isKindOfClass:[NSString class]]); sendToAllProcesses(Messages::WebProcess::RegisterSchemeForCustomProtocol(scheme)); }]; m_customSchemeUnregisteredObserver = [[NSNotificationCenter defaultCenter] addObserverForName:WebKit::SchemeForCustomProtocolUnregisteredNotificationName object:nil queue:[NSOperationQueue currentQueue] usingBlock:^(NSNotification *notification) { NSString *scheme = [notification object]; ASSERT([scheme isKindOfClass:[NSString class]]); sendToAllProcesses(Messages::WebProcess::UnregisterSchemeForCustomProtocol(scheme)); }]; // Listen for enhanced accessibility changes and propagate them to the WebProcess. m_enhancedAccessibilityObserver = [[NSNotificationCenter defaultCenter] addObserverForName:WebKitApplicationDidChangeAccessibilityEnhancedUserInterfaceNotification object:nil queue:[NSOperationQueue currentQueue] usingBlock:^(NSNotification *note) { setEnhancedAccessibility([[[note userInfo] objectForKey:@"AXEnhancedUserInterface"] boolValue]); }]; } void WebContext::platformInvalidateContext() { [[NSNotificationCenter defaultCenter] removeObserver:(id)m_enhancedAccessibilityObserver.get()]; } String WebContext::platformDefaultDiskCacheDirectory() const { RetainPtr cachePath(AdoptNS, (NSString *)WKCopyFoundationCacheDirectory()); if (!cachePath) cachePath = @"~/Library/Caches/com.apple.WebKit2.WebProcess"; return [cachePath.get() stringByStandardizingPath]; } String WebContext::platformDefaultCookieStorageDirectory() const { notImplemented(); return [@"" stringByStandardizingPath]; } String WebContext::platformDefaultDatabaseDirectory() const { NSString *databasesDirectory = [[NSUserDefaults standardUserDefaults] objectForKey:WebDatabaseDirectoryDefaultsKey]; if (!databasesDirectory || ![databasesDirectory isKindOfClass:[NSString class]]) databasesDirectory = @"~/Library/WebKit/Databases"; return [databasesDirectory stringByStandardizingPath]; } String WebContext::platformDefaultIconDatabasePath() const { // FIXME: - After this "backwards compatibility" radar is removed, this code should be removed to only return an empty String. NSString *databasesDirectory = [[NSUserDefaults standardUserDefaults] objectForKey:WebIconDatabaseDirectoryDefaultsKey]; if (!databasesDirectory || ![databasesDirectory isKindOfClass:[NSString class]]) databasesDirectory = @"~/Library/Icons/WebpageIcons.db"; return [databasesDirectory stringByStandardizingPath]; } String WebContext::platformDefaultLocalStorageDirectory() const { NSString *localStorageDirectory = [[NSUserDefaults standardUserDefaults] objectForKey:WebStorageDirectoryDefaultsKey]; if (!localStorageDirectory || ![localStorageDirectory isKindOfClass:[NSString class]]) localStorageDirectory = @"~/Library/WebKit/LocalStorage"; return [localStorageDirectory stringByStandardizingPath]; } bool WebContext::omitPDFSupport() { // Since this is a "secret default" we don't bother registering it. return [[NSUserDefaults standardUserDefaults] boolForKey:@"WebKitOmitPDFSupport"]; } void WebContext::getPasteboardTypes(const String& pasteboardName, Vector& pasteboardTypes) { PlatformPasteboard(pasteboardName).getTypes(pasteboardTypes); } void WebContext::getPasteboardPathnamesForType(const String& pasteboardName, const String& pasteboardType, Vector& pathnames) { PlatformPasteboard(pasteboardName).getPathnamesForType(pathnames, pasteboardType); } void WebContext::getPasteboardStringForType(const String& pasteboardName, const String& pasteboardType, String& string) { string = PlatformPasteboard(pasteboardName).stringForType(pasteboardType); } void WebContext::getPasteboardBufferForType(const String& pasteboardName, const String& pasteboardType, SharedMemory::Handle& handle, uint64_t& size) { RefPtr buffer = PlatformPasteboard(pasteboardName).bufferForType(pasteboardType); if (!buffer) return; size = buffer->size(); RefPtr sharedMemoryBuffer = SharedMemory::create(size); memcpy(sharedMemoryBuffer->data(), buffer->data(), size); sharedMemoryBuffer->createHandle(handle, SharedMemory::ReadOnly); } void WebContext::pasteboardCopy(const String& fromPasteboard, const String& toPasteboard) { PlatformPasteboard(toPasteboard).copy(fromPasteboard); } void WebContext::getPasteboardChangeCount(const String& pasteboardName, uint64_t& changeCount) { changeCount = PlatformPasteboard(pasteboardName).changeCount(); } void WebContext::getPasteboardUniqueName(String& pasteboardName) { pasteboardName = PlatformPasteboard::uniqueName(); } void WebContext::getPasteboardColor(const String& pasteboardName, WebCore::Color& color) { color = PlatformPasteboard(pasteboardName).color(); } void WebContext::getPasteboardURL(const String& pasteboardName, WTF::String& urlString) { urlString = PlatformPasteboard(pasteboardName).url().string(); } void WebContext::addPasteboardTypes(const String& pasteboardName, const Vector& pasteboardTypes) { PlatformPasteboard(pasteboardName).addTypes(pasteboardTypes); } void WebContext::setPasteboardTypes(const String& pasteboardName, const Vector& pasteboardTypes) { PlatformPasteboard(pasteboardName).setTypes(pasteboardTypes); } void WebContext::setPasteboardPathnamesForType(const String& pasteboardName, const String& pasteboardType, const Vector& pathnames) { PlatformPasteboard(pasteboardName).setPathnamesForType(pathnames, pasteboardType); } void WebContext::setPasteboardStringForType(const String& pasteboardName, const String& pasteboardType, const String& string) { PlatformPasteboard(pasteboardName).setStringForType(string, pasteboardType); } void WebContext::setPasteboardBufferForType(const String& pasteboardName, const String& pasteboardType, const SharedMemory::Handle& handle, uint64_t size) { if (handle.isNull()) { PlatformPasteboard(pasteboardName).setBufferForType(0, pasteboardType); return; } RefPtr sharedMemoryBuffer = SharedMemory::create(handle, SharedMemory::ReadOnly); RefPtr buffer = SharedBuffer::create(static_cast(sharedMemoryBuffer->data()), size); PlatformPasteboard(pasteboardName).setBufferForType(buffer, pasteboardType); } void WebContext::applicationBecameVisible(uint32_t, void*, uint32_t, void*, uint32_t) { if (s_applicationIsOccluded) { s_applicationIsOccluded = false; const Vector& contexts = WebContext::allContexts(); for (size_t i = 0, count = contexts.size(); i < count; ++i) contexts[i]->sendToAllProcesses(Messages::WebProcess::SetApplicationIsOccluded(false)); #if ENABLE(PLUGIN_PROCESS) PluginProcessManager::shared().setApplicationIsOccluded(false); #endif #if ENABLE(NETWORK_PROCESS) NetworkProcessManager::shared().setApplicationIsOccluded(false); #endif #if ENABLE(SHARED_WORKER_PROCESS) SharedWorkerProcessManager::shared().setApplicationIsOccluded(false); #endif } } void WebContext::applicationBecameOccluded(uint32_t, void*, uint32_t, void*, uint32_t) { if (!s_applicationIsOccluded) { s_applicationIsOccluded = true; const Vector& contexts = WebContext::allContexts(); for (size_t i = 0, count = contexts.size(); i < count; ++i) contexts[i]->sendToAllProcesses(Messages::WebProcess::SetApplicationIsOccluded(true)); #if ENABLE(PLUGIN_PROCESS) PluginProcessManager::shared().setApplicationIsOccluded(true); #endif #if ENABLE(NETWORK_PROCESS) NetworkProcessManager::shared().setApplicationIsOccluded(true); #endif #if ENABLE(SHARED_WORKER_PROCESS) SharedWorkerProcessManager::shared().setApplicationIsOccluded(true); #endif } } void WebContext::initializeProcessSuppressionSupport() { static bool didInitialize = false; if (didInitialize) return; didInitialize = true; // A temporary default until process suppression is enabled by default, at which point a context setting can be added with the // interpretation that any context disabling process suppression disables it for plugin/network and shared worker processes. bool processSuppressionSupportEnabled = [[NSUserDefaults standardUserDefaults] boolForKey:@"WebKitProcessSuppressionSupportEnabled"]; if (processSuppressionSupportEnabled) registerOcclusionNotificationHandlers(); } void WebContext::registerOcclusionNotificationHandlers() { #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1090 if (!WKRegisterOcclusionNotificationHandler(WKOcclusionNotificationTypeApplicationBecameVisible, applicationBecameVisible)) { WTFLogAlways("Registeration of \"App Became Visible\" notification handler failed.\n"); return; } if (!WKRegisterOcclusionNotificationHandler(WKOcclusionNotificationTypeApplicationBecameOccluded, applicationBecameOccluded)) WTFLogAlways("Registeration of \"App Became Occluded\" notification handler failed.\n"); #endif } } // namespace WebKit