/* * Copyright (C) 2010 Apple Inc. All rights reserved. * Copyright (C) 2011 Google 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 "PluginProcess.h" #if ENABLE(PLUGIN_PROCESS) #import "ArgumentCoders.h" #import "NetscapePlugin.h" #import "PluginProcessCreationParameters.h" #import "PluginProcessProxyMessages.h" #import "PluginProcessShim.h" #import "PluginSandboxProfile.h" #import "SandboxInitializationParameters.h" #import #import #import #import #import #import #import using namespace WebCore; const CFStringRef kLSPlugInBundleIdentifierKey = CFSTR("LSPlugInBundleIdentifierKey"); namespace WebKit { class FullscreenWindowTracker { WTF_MAKE_NONCOPYABLE(FullscreenWindowTracker); public: FullscreenWindowTracker() { } template void windowShown(T window); template void windowHidden(T window); private: typedef HashSet WindowSet; WindowSet m_windows; }; static bool rectCoversAnyScreen(NSRect rect) { for (NSScreen *screen in [NSScreen screens]) { if (NSContainsRect(rect, [screen frame])) return YES; } return NO; } #ifndef NP_NO_CARBON static bool windowCoversAnyScreen(WindowRef window) { HIRect bounds; HIWindowGetBounds(window, kWindowStructureRgn, kHICoordSpaceScreenPixel, &bounds); // Convert to Cocoa-style screen coordinates that use a Y offset relative to the zeroth screen's origin. bounds.origin.y = NSHeight([(NSScreen *)[[NSScreen screens] objectAtIndex:0] frame]) - CGRectGetMaxY(bounds); return rectCoversAnyScreen(NSRectFromCGRect(bounds)); } #endif static bool windowCoversAnyScreen(NSWindow* window) { return rectCoversAnyScreen([window frame]); } template void FullscreenWindowTracker::windowShown(T window) { // If this window is already visible then there is nothing to do. WindowSet::iterator it = m_windows.find(window); if (it != m_windows.end()) return; // If the window is not full-screen then we're not interested in it. if (!windowCoversAnyScreen(window)) return; bool windowSetWasEmpty = m_windows.isEmpty(); m_windows.add(window); // If this is the first full screen window to be shown, notify the UI process. if (windowSetWasEmpty) PluginProcess::shared().setFullscreenWindowIsShowing(true); } template void FullscreenWindowTracker::windowHidden(T window) { // If this is not a window that we're tracking then there is nothing to do. WindowSet::iterator it = m_windows.find(window); if (it == m_windows.end()) return; m_windows.remove(it); // If this was the last full screen window that was visible, notify the UI process. if (m_windows.isEmpty()) PluginProcess::shared().setFullscreenWindowIsShowing(false); } static FullscreenWindowTracker& fullscreenWindowTracker() { DEFINE_STATIC_LOCAL(FullscreenWindowTracker, fullscreenWindowTracker, ()); return fullscreenWindowTracker; } #if defined(__i386__) static pthread_once_t shouldCallRealDebuggerOnce = PTHREAD_ONCE_INIT; static bool isUserbreakSet = false; static void initShouldCallRealDebugger() { char* var = getenv("USERBREAK"); if (var) isUserbreakSet = atoi(var); } static bool shouldCallRealDebugger() { pthread_once(&shouldCallRealDebuggerOnce, initShouldCallRealDebugger); return isUserbreakSet; } static bool isWindowActive(WindowRef windowRef, bool& result) { #ifndef NP_NO_CARBON if (NetscapePlugin* plugin = NetscapePlugin::netscapePluginFromWindow(windowRef)) { result = plugin->isWindowActive(); return true; } #endif return false; } static UInt32 getCurrentEventButtonState() { #ifndef NP_NO_CARBON return NetscapePlugin::buttonState(); #else ASSERT_NOT_REACHED(); return 0; #endif } static void carbonWindowShown(WindowRef window) { #ifndef NP_NO_CARBON fullscreenWindowTracker().windowShown(window); #endif } static void carbonWindowHidden(WindowRef window) { #ifndef NP_NO_CARBON fullscreenWindowTracker().windowHidden(window); #endif } static bool openCFURLRef(CFURLRef url, int32_t& status, CFURLRef* launchedURL) { String launchedURLString; if (!PluginProcess::shared().openURL(KURL(url).string(), status, launchedURLString)) return false; if (!launchedURLString.isNull() && launchedURL) *launchedURL = KURL(ParsedURLString, launchedURLString).createCFURL().leakRef(); return true; } #endif static void setModal(bool modalWindowIsShowing) { PluginProcess::shared().setModalWindowIsShowing(modalWindowIsShowing); } static unsigned modalCount = 0; static void beginModal() { #if COMPILER(CLANG) #pragma clang diagnostic push #pragma clang diagnostic ignored "-Wdeprecated-declarations" #endif // Make sure to make ourselves the front process ProcessSerialNumber psn; GetCurrentProcess(&psn); SetFrontProcess(&psn); #if COMPILER(CLANG) #pragma clang diagnostic pop #endif if (!modalCount++) setModal(true); } static void endModal() { if (!--modalCount) setModal(false); } static IMP NSApplication_RunModalForWindow; static NSInteger replacedRunModalForWindow(id self, SEL _cmd, NSWindow* window) { beginModal(); NSInteger result = ((NSInteger (*)(id, SEL, NSWindow *))NSApplication_RunModalForWindow)(self, _cmd, window); endModal(); return result; } #if defined(__i386__) static void initializeShim() { // Initialize the shim for 32-bit only. const PluginProcessShimCallbacks callbacks = { shouldCallRealDebugger, isWindowActive, getCurrentEventButtonState, beginModal, endModal, carbonWindowShown, carbonWindowHidden, setModal, openCFURLRef, }; PluginProcessShimInitializeFunc initFunc = reinterpret_cast(dlsym(RTLD_DEFAULT, "WebKitPluginProcessShimInitialize")); initFunc(callbacks); } #endif static IMP NSConcreteTask_launch; static void replacedNSConcreteTask_launch(NSTask *self, SEL _cmd) { String launchPath = self.launchPath; Vector arguments; arguments.reserveInitialCapacity(self.arguments.count); for (NSString *argument in self.arguments) arguments.uncheckedAppend(argument); if (PluginProcess::shared().launchProcess(launchPath, arguments)) return; NSConcreteTask_launch(self, _cmd); } static NSRunningApplication *(*NSWorkspace_launchApplicationAtURL_options_configuration_error)(NSWorkspace *, SEL, NSURL *, NSWorkspaceLaunchOptions, NSDictionary *, NSError **); static NSRunningApplication *replacedNSWorkspace_launchApplicationAtURL_options_configuration_error(NSWorkspace *self, SEL _cmd, NSURL *url, NSWorkspaceLaunchOptions options, NSDictionary *configuration, NSError **error) { Vector arguments; if (NSArray *argumentsArray = [configuration objectForKey:NSWorkspaceLaunchConfigurationArguments]) { if ([argumentsArray isKindOfClass:[NSArray array]]) { for (NSString *argument in argumentsArray) { if ([argument isKindOfClass:[NSString class]]) arguments.append(argument); } } } if (PluginProcess::shared().launchApplicationAtURL(KURL(url).string(), arguments)) { if (error) *error = nil; return nil; } return NSWorkspace_launchApplicationAtURL_options_configuration_error(self, _cmd, url, options, configuration, error); } static void initializeCocoaOverrides() { // Override -[NSConcreteTask launch:] Method launchMethod = class_getInstanceMethod(objc_getClass("NSConcreteTask"), @selector(launch)); NSConcreteTask_launch = method_setImplementation(launchMethod, reinterpret_cast(replacedNSConcreteTask_launch)); // Override -[NSWorkspace launchApplicationAtURL:options:configuration:error:] Method launchApplicationAtURLOptionsConfigurationErrorMethod = class_getInstanceMethod(objc_getClass("NSWorkspace"), @selector(launchApplicationAtURL:options:configuration:error:)); NSWorkspace_launchApplicationAtURL_options_configuration_error = reinterpret_cast(method_setImplementation(launchApplicationAtURLOptionsConfigurationErrorMethod, reinterpret_cast(replacedNSWorkspace_launchApplicationAtURL_options_configuration_error))); // Override -[NSApplication runModalForWindow:] Method runModalForWindowMethod = class_getInstanceMethod(objc_getClass("NSApplication"), @selector(runModalForWindow:)); NSApplication_RunModalForWindow = method_setImplementation(runModalForWindowMethod, reinterpret_cast(replacedRunModalForWindow)); NSNotificationCenter *defaultCenter = [NSNotificationCenter defaultCenter]; // Track when any Cocoa window is about to be be shown. id orderOnScreenObserver = [defaultCenter addObserverForName:WKWindowWillOrderOnScreenNotification() object:nil queue:nil usingBlock:^(NSNotification *notification) { fullscreenWindowTracker().windowShown([notification object]); }]; // Track when any Cocoa window is about to be hidden. id orderOffScreenObserver = [defaultCenter addObserverForName:WKWindowWillOrderOffScreenNotification() object:nil queue:nil usingBlock:^(NSNotification *notification) { fullscreenWindowTracker().windowHidden([notification object]); }]; // Leak the two observers so that they observe notifications for the lifetime of the process. CFRetain(orderOnScreenObserver); CFRetain(orderOffScreenObserver); } void PluginProcess::setModalWindowIsShowing(bool modalWindowIsShowing) { parentProcessConnection()->send(Messages::PluginProcessProxy::SetModalWindowIsShowing(modalWindowIsShowing), 0); } void PluginProcess::setFullscreenWindowIsShowing(bool fullscreenWindowIsShowing) { parentProcessConnection()->send(Messages::PluginProcessProxy::SetFullscreenWindowIsShowing(fullscreenWindowIsShowing), 0); } bool PluginProcess::launchProcess(const String& launchPath, const Vector& arguments) { bool result; if (!parentProcessConnection()->sendSync(Messages::PluginProcessProxy::LaunchProcess(launchPath, arguments), Messages::PluginProcessProxy::LaunchProcess::Reply(result), 0)) return false; return result; } bool PluginProcess::launchApplicationAtURL(const String& urlString, const Vector& arguments) { bool result = false; if (!parentProcessConnection()->sendSync(Messages::PluginProcessProxy::LaunchApplicationAtURL(urlString, arguments), Messages::PluginProcessProxy::LaunchProcess::Reply(result), 0)) return false; return result; } bool PluginProcess::openURL(const String& urlString, int32_t& status, String& launchedURLString) { bool result; if (!parentProcessConnection()->sendSync(Messages::PluginProcessProxy::OpenURL(urlString), Messages::PluginProcessProxy::OpenURL::Reply(result, status, launchedURLString), 0)) return false; return result; } static void muteAudio(void) { AudioObjectPropertyAddress propertyAddress = { kAudioHardwarePropertyProcessIsAudible, kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMaster }; UInt32 propertyData = 0; OSStatus result = AudioObjectSetPropertyData(kAudioObjectSystemObject, &propertyAddress, 0, 0, sizeof(UInt32), &propertyData); ASSERT_UNUSED(result, result == noErr); } void PluginProcess::platformInitializePluginProcess(const PluginProcessCreationParameters& parameters) { m_compositingRenderServerPort = parameters.acceleratedCompositingPort.port(); if (parameters.processType == PluginProcessTypeSnapshot) muteAudio(); } void PluginProcess::platformInitializeProcess(const ChildProcessInitializationParameters& parameters) { #if defined(__i386__) // Initialize the shim. initializeShim(); #endif // Initialize Cocoa overrides. initializeCocoaOverrides(); // FIXME: It would be better to proxy SetCursor calls over to the UI process instead of // allowing plug-ins to change the mouse cursor at any time. WKEnableSettingCursorWhenInBackground(); #if defined(__i386__) NSDictionary *defaults = [[NSDictionary alloc] initWithObjectsAndKeys:[NSNumber numberWithBool:YES], @"AppleMagnifiedMode", nil]; [[NSUserDefaults standardUserDefaults] registerDefaults:defaults]; [defaults release]; #endif RetainPtr pluginURL = adoptCF(CFURLCreateWithFileSystemPath(0, m_pluginPath.createCFString().get(), kCFURLPOSIXPathStyle, false)); if (!pluginURL) return; RetainPtr pluginBundle = adoptCF(CFBundleCreate(kCFAllocatorDefault, pluginURL.get())); if (!pluginBundle) return; m_pluginBundleIdentifier = CFBundleGetIdentifier(pluginBundle.get()); } void PluginProcess::initializeProcessName(const ChildProcessInitializationParameters& parameters) { NSString *applicationName = [NSString stringWithFormat:WEB_UI_STRING("%@ (%@ Internet plug-in)", "visible name of the plug-in host process. The first argument is the plug-in name and the second argument is the application name."), [[(NSString *)m_pluginPath lastPathComponent] stringByDeletingPathExtension], (NSString *)parameters.uiProcessName]; WKSetVisibleApplicationName((CFStringRef)applicationName); #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1090 if (!m_pluginBundleIdentifier.isEmpty()) WKSetApplicationInformationItem(kLSPlugInBundleIdentifierKey, m_pluginBundleIdentifier.createCFString().get()); #endif } void PluginProcess::initializeSandbox(const ChildProcessInitializationParameters& parameters, SandboxInitializationParameters& sandboxParameters) { if (parameters.extraInitializationData.get("disable-sandbox") == "1") return; String sandboxProfile = pluginSandboxProfile(m_pluginBundleIdentifier); if (sandboxProfile.isEmpty()) return; sandboxParameters.setSandboxProfile(sandboxProfile); #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1080 // Use private temporary and cache directories. char temporaryDirectory[PATH_MAX]; if (!confstr(_CS_DARWIN_USER_TEMP_DIR, temporaryDirectory, sizeof(temporaryDirectory))) { WTFLogAlways("PluginProcess: couldn't retrieve system temporary directory path: %d\n", errno); exit(EX_OSERR); } if (strlcpy(temporaryDirectory, [[[[NSFileManager defaultManager] stringWithFileSystemRepresentation:temporaryDirectory length:strlen(temporaryDirectory)] stringByAppendingPathComponent:@"WebKitPlugin-XXXXXX"] fileSystemRepresentation], sizeof(temporaryDirectory)) >= sizeof(temporaryDirectory) || !mkdtemp(temporaryDirectory)) { WTFLogAlways("PluginProcess: couldn't create private temporary directory '%s'\n", temporaryDirectory); exit(EX_OSERR); } sandboxParameters.setSystemDirectorySuffix([[[[NSFileManager defaultManager] stringWithFileSystemRepresentation:temporaryDirectory length:strlen(temporaryDirectory)] lastPathComponent] fileSystemRepresentation]); #endif sandboxParameters.addPathParameter("PLUGIN_PATH", m_pluginPath); RetainPtr cachePath = adoptCF(WKCopyFoundationCacheDirectory()); sandboxParameters.addPathParameter("NSURL_CACHE_DIR", (NSString *)cachePath.get()); RetainPtr defaults = adoptNS([[NSDictionary alloc] initWithObjectsAndKeys:[NSNumber numberWithBool:YES], @"NSUseRemoteSavePanel", nil]); [[NSUserDefaults standardUserDefaults] registerDefaults:defaults.get()]; ChildProcess::initializeSandbox(parameters, sandboxParameters); } } // namespace WebKit #endif // ENABLE(PLUGIN_PROCESS)