// Copyright (c) 2012 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. // On Mac, one can't make shortcuts with command-line arguments. Instead, we // produce small app bundles which locate the Chromium framework and load it, // passing the appropriate data. This is the entry point into the framework for // those app bundles. #import #include "apps/app_shim/app_shim_messages.h" #include "base/at_exit.h" #include "base/command_line.h" #include "base/logging.h" #include "base/mac/launch_services_util.h" #include "base/mac/mac_logging.h" #include "base/mac/mac_util.h" #include "base/mac/scoped_nsautorelease_pool.h" #include "base/mac/scoped_nsobject.h" #include "base/message_loop.h" #include "base/path_service.h" #include "base/strings/sys_string_conversions.h" #include "base/threading/thread.h" #include "chrome/common/chrome_paths.h" #include "chrome/common/chrome_paths_internal.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/mac/app_mode_common.h" #include "ipc/ipc_channel_proxy.h" #include "ipc/ipc_listener.h" #include "ipc/ipc_message.h" namespace { const app_mode::ChromeAppModeInfo* g_info; base::Thread* g_io_thread = NULL; } // namespace class AppShimController; @interface AppShimDelegate : NSObject { @private AppShimController* appShimController_; // Weak. Owns us. BOOL terminateNow_; BOOL terminateRequested_; } - (id)initWithController:(AppShimController*)controller; - (void)terminateNow; @end // The AppShimController is responsible for communication with the main Chrome // process, and generally controls the lifetime of the app shim process. class AppShimController : public IPC::Listener { public: AppShimController(); // Connects to Chrome and sends a LaunchApp message. void Init(); // Sends a QuitApp message to Chrome. void QuitApp(); private: // IPC::Listener implemetation. virtual bool OnMessageReceived(const IPC::Message& message) OVERRIDE; virtual void OnChannelError() OVERRIDE; // If Chrome failed to launch the app, |success| will be false and the app // shim process should die. void OnLaunchAppDone(bool success); // Called when the app is activated, either by the user clicking on it in the // dock or by Cmd+Tabbing to it. void OnDidActivateApplication(); // Terminates the app shim process. void Close(); IPC::ChannelProxy* channel_; base::scoped_nsobject nsapp_delegate_; DISALLOW_COPY_AND_ASSIGN(AppShimController); }; AppShimController::AppShimController() : channel_(NULL) {} void AppShimController::Init() { DCHECK(g_io_thread); NSString* chrome_bundle_path = base::SysUTF8ToNSString(g_info->chrome_outer_bundle_path.value()); NSBundle* chrome_bundle = [NSBundle bundleWithPath:chrome_bundle_path]; base::FilePath user_data_dir; if (!chrome::GetUserDataDirectoryForBrowserBundle(chrome_bundle, &user_data_dir)) { Close(); return; } base::FilePath socket_path = user_data_dir.Append(app_mode::kAppShimSocketName); IPC::ChannelHandle handle(socket_path.value()); channel_ = new IPC::ChannelProxy(handle, IPC::Channel::MODE_NAMED_CLIENT, this, g_io_thread->message_loop_proxy()); channel_->Send(new AppShimHostMsg_LaunchApp( g_info->profile_dir, g_info->app_mode_id, CommandLine::ForCurrentProcess()->HasSwitch(app_mode::kNoLaunchApp) ? apps::APP_SHIM_LAUNCH_REGISTER_ONLY : apps::APP_SHIM_LAUNCH_NORMAL)); nsapp_delegate_.reset([[AppShimDelegate alloc] initWithController:this]); DCHECK(![NSApp delegate]); [NSApp setDelegate:nsapp_delegate_]; } void AppShimController::QuitApp() { channel_->Send(new AppShimHostMsg_QuitApp); } bool AppShimController::OnMessageReceived(const IPC::Message& message) { bool handled = true; IPC_BEGIN_MESSAGE_MAP(AppShimController, message) IPC_MESSAGE_HANDLER(AppShimMsg_LaunchApp_Done, OnLaunchAppDone) IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() return handled; } void AppShimController::OnChannelError() { Close(); } void AppShimController::OnLaunchAppDone(bool success) { if (!success) { Close(); return; } [[[NSWorkspace sharedWorkspace] notificationCenter] addObserverForName:NSWorkspaceDidActivateApplicationNotification object:nil queue:nil usingBlock:^(NSNotification* notification) { NSRunningApplication* activated_app = [[notification userInfo] objectForKey:NSWorkspaceApplicationKey]; if ([activated_app isEqual:[NSRunningApplication currentApplication]]) OnDidActivateApplication(); }]; } void AppShimController::Close() { [nsapp_delegate_ terminateNow]; } void AppShimController::OnDidActivateApplication() { channel_->Send(new AppShimHostMsg_FocusApp); } @implementation AppShimDelegate - (id)initWithController:(AppShimController*)controller { if ((self = [super init])) { appShimController_ = controller; } return self; } - (NSApplicationTerminateReply) applicationShouldTerminate:(NSApplication*)sender { if (terminateNow_) return NSTerminateNow; appShimController_->QuitApp(); // Wait for the channel to close before terminating. terminateRequested_ = YES; return NSTerminateLater; } - (void)terminateNow { if (terminateRequested_) { [NSApp replyToApplicationShouldTerminate:NSTerminateNow]; return; } terminateNow_ = YES; [NSApp terminate:nil]; } @end //----------------------------------------------------------------------------- // A ReplyEventHandler is a helper class to send an Apple Event to a process // and call a callback when the reply returns. // // This is used to 'ping' the main Chrome process -- once Chrome has sent back // an Apple Event reply, it's guaranteed that it has opened the IPC channel // that the app shim will connect to. @interface ReplyEventHandler : NSObject { base::Callback onReply_; AEDesc replyEvent_; } // Sends an Apple Event to the process identified by |psn|, and calls |replyFn| // when the reply is received. Internally this creates a ReplyEventHandler, // which will delete itself once the reply event has been received. + (void)pingProcess:(const ProcessSerialNumber&)psn andCall:(base::Callback)replyFn; @end @interface ReplyEventHandler (PrivateMethods) // Initialise the reply event handler. Doesn't register any handlers until // |-pingProcess:| is called. |replyFn| is the function to be called when the // Apple Event reply arrives. - (id)initWithCallback:(base::Callback)replyFn; // Sends an Apple Event ping to the process identified by |psn| and registers // to listen for a reply. - (void)pingProcess:(const ProcessSerialNumber&)psn; // Called when a response is received from the target process for the ping sent // by |-pingProcess:|. - (void)message:(NSAppleEventDescriptor*)event withReply:(NSAppleEventDescriptor*)reply; // Calls |onReply_|, passing it |success| to specify whether the ping was // successful. - (void)closeWithSuccess:(bool)success; @end @implementation ReplyEventHandler + (void)pingProcess:(const ProcessSerialNumber&)psn andCall:(base::Callback)replyFn { // The object will release itself when the reply arrives, or possibly earlier // if an unrecoverable error occurs. ReplyEventHandler* handler = [[ReplyEventHandler alloc] initWithCallback:replyFn]; [handler pingProcess:psn]; } @end @implementation ReplyEventHandler (PrivateMethods) - (id)initWithCallback:(base::Callback)replyFn { if ((self = [super init])) { onReply_ = replyFn; } return self; } - (void)pingProcess:(const ProcessSerialNumber&)psn { // Register the reply listener. NSAppleEventManager* em = [NSAppleEventManager sharedAppleEventManager]; [em setEventHandler:self andSelector:@selector(message:withReply:) forEventClass:'aevt' andEventID:'ansr']; // Craft the Apple Event to send. NSAppleEventDescriptor* target = [NSAppleEventDescriptor descriptorWithDescriptorType:typeProcessSerialNumber bytes:&psn length:sizeof(psn)]; NSAppleEventDescriptor* initial_event = [NSAppleEventDescriptor appleEventWithEventClass:app_mode::kAEChromeAppClass eventID:app_mode::kAEChromeAppPing targetDescriptor:target returnID:kAutoGenerateReturnID transactionID:kAnyTransactionID]; // And away we go. // TODO(jeremya): if we don't care about the contents of the reply, can we // pass NULL for the reply event parameter? OSStatus status = AESendMessage( [initial_event aeDesc], &replyEvent_, kAEQueueReply, kAEDefaultTimeout); if (status != noErr) { OSSTATUS_LOG(ERROR, status) << "AESendMessage"; [self closeWithSuccess:false]; } } - (void)message:(NSAppleEventDescriptor*)event withReply:(NSAppleEventDescriptor*)reply { [self closeWithSuccess:true]; } - (void)closeWithSuccess:(bool)success { onReply_.Run(success); NSAppleEventManager* em = [NSAppleEventManager sharedAppleEventManager]; [em removeEventHandlerForEventClass:'aevt' andEventID:'ansr']; [self release]; } @end //----------------------------------------------------------------------------- namespace { // Called when the main Chrome process responds to the Apple Event ping that // was sent, or when the ping fails (if |success| is false). void OnPingChromeReply(bool success) { if (!success) { [NSApp terminate:nil]; return; } AppShimController* controller = new AppShimController; controller->Init(); } } // namespace extern "C" { // |ChromeAppModeStart()| is the point of entry into the framework from the app // mode loader. __attribute__((visibility("default"))) int ChromeAppModeStart(const app_mode::ChromeAppModeInfo* info); } // extern "C" int ChromeAppModeStart(const app_mode::ChromeAppModeInfo* info) { CommandLine::Init(info->argc, info->argv); base::mac::ScopedNSAutoreleasePool scoped_pool; base::AtExitManager exit_manager; chrome::RegisterPathProvider(); if (info->major_version < app_mode::kCurrentChromeAppModeInfoMajorVersion) { RAW_LOG(ERROR, "App Mode Loader too old."); return 1; } if (info->major_version > app_mode::kCurrentChromeAppModeInfoMajorVersion) { RAW_LOG(ERROR, "Browser Framework too old to load App Shortcut."); return 1; } g_info = info; // Launch the IO thread. base::Thread::Options io_thread_options; io_thread_options.message_loop_type = base::MessageLoop::TYPE_IO; base::Thread *io_thread = new base::Thread("CrAppShimIO"); io_thread->StartWithOptions(io_thread_options); g_io_thread = io_thread; // Find already running instances of Chrome. NSString* chrome_bundle_path = base::SysUTF8ToNSString(g_info->chrome_outer_bundle_path.value()); NSBundle* chrome_bundle = [NSBundle bundleWithPath:chrome_bundle_path]; NSArray* existing_chrome = [NSRunningApplication runningApplicationsWithBundleIdentifier:[chrome_bundle bundleIdentifier]]; // Launch Chrome if it isn't already running. ProcessSerialNumber psn; if ([existing_chrome count] > 0) { OSStatus status = GetProcessForPID( [[existing_chrome objectAtIndex:0] processIdentifier], &psn); if (status) return 1; } else { CommandLine command_line(CommandLine::NO_PROGRAM); command_line.AppendSwitch(switches::kSilentLaunch); bool success = base::mac::OpenApplicationWithPath(info->chrome_outer_bundle_path, command_line, &psn); if (!success) return 1; } // This code abuses the fact that Apple Events sent before the process is // fully initialized don't receive a reply until its run loop starts. Once // the reply is received, Chrome will have opened its IPC port, guaranteed. [ReplyEventHandler pingProcess:psn andCall:base::Bind(&OnPingChromeReply)]; base::MessageLoopForUI main_message_loop; main_message_loop.set_thread_name("MainThread"); base::PlatformThread::SetName("CrAppShimMain"); main_message_loop.Run(); return 0; }