1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 // On Mac, shortcuts can't have command-line arguments. Instead, produce small 6 // app bundles which locate the Chromium framework and load it, passing the 7 // appropriate data. This is the code for such an app bundle. It should be kept 8 // minimal and do as little work as possible (with as much work done on 9 // framework side as possible). 10 11 #include <dlfcn.h> 12 13 #import <Cocoa/Cocoa.h> 14 15 #include "base/command_line.h" 16 #include "base/files/file_path.h" 17 #include "base/files/file_util.h" 18 #include "base/logging.h" 19 #include "base/mac/foundation_util.h" 20 #include "base/mac/launch_services_util.h" 21 #include "base/mac/scoped_nsautorelease_pool.h" 22 #include "base/process/launch.h" 23 #include "base/strings/string_number_conversions.h" 24 #include "base/strings/sys_string_conversions.h" 25 #include "chrome/common/chrome_constants.h" 26 #include "chrome/common/chrome_switches.h" 27 #import "chrome/common/mac/app_mode_chrome_locator.h" 28 #include "chrome/common/mac/app_mode_common.h" 29 30 namespace { 31 32 typedef int (*StartFun)(const app_mode::ChromeAppModeInfo*); 33 34 int LoadFrameworkAndStart(app_mode::ChromeAppModeInfo* info) { 35 using base::SysNSStringToUTF8; 36 using base::SysNSStringToUTF16; 37 using base::mac::CFToNSCast; 38 using base::mac::CFCastStrict; 39 using base::mac::NSToCFCast; 40 41 base::mac::ScopedNSAutoreleasePool scoped_pool; 42 43 // Get the current main bundle, i.e., that of the app loader that's running. 44 NSBundle* app_bundle = [NSBundle mainBundle]; 45 CHECK(app_bundle) << "couldn't get loader bundle"; 46 47 // ** 1: Get path to outer Chrome bundle. 48 // Get the bundle ID of the browser that created this app bundle. 49 NSString* cr_bundle_id = base::mac::ObjCCast<NSString>( 50 [app_bundle objectForInfoDictionaryKey:app_mode::kBrowserBundleIDKey]); 51 CHECK(cr_bundle_id) << "couldn't get browser bundle ID"; 52 53 // First check if Chrome exists at the last known location. 54 base::FilePath cr_bundle_path; 55 NSString* cr_bundle_path_ns = 56 [CFToNSCast(CFCastStrict<CFStringRef>(CFPreferencesCopyAppValue( 57 NSToCFCast(app_mode::kLastRunAppBundlePathPrefsKey), 58 NSToCFCast(cr_bundle_id)))) autorelease]; 59 cr_bundle_path = base::mac::NSStringToFilePath(cr_bundle_path_ns); 60 bool found_bundle = 61 !cr_bundle_path.empty() && base::DirectoryExists(cr_bundle_path); 62 63 if (!found_bundle) { 64 // If no such bundle path exists, try to search by bundle ID. 65 if (!app_mode::FindBundleById(cr_bundle_id, &cr_bundle_path)) { 66 // TODO(jeremy): Display UI to allow user to manually locate the Chrome 67 // bundle. 68 LOG(FATAL) << "Failed to locate bundle by identifier"; 69 } 70 } 71 72 // ** 2: Read the running Chrome version. 73 // The user_data_dir for shims actually contains the app_data_path. 74 // I.e. <user_data_dir>/<profile_dir>/Web Applications/_crx_extensionid/ 75 base::FilePath app_data_dir = base::mac::NSStringToFilePath([app_bundle 76 objectForInfoDictionaryKey:app_mode::kCrAppModeUserDataDirKey]); 77 base::FilePath user_data_dir = app_data_dir.DirName().DirName().DirName(); 78 CHECK(!user_data_dir.empty()); 79 80 // If the version file does not exist, |cr_version_str| will be empty and 81 // app_mode::GetChromeBundleInfo will default to the latest version. 82 base::FilePath cr_version_str; 83 base::ReadSymbolicLink( 84 user_data_dir.Append(app_mode::kRunningChromeVersionSymlinkName), 85 &cr_version_str); 86 87 // If the version file does exist, it may have been left by a crashed Chrome 88 // process. Ensure the process is still running. 89 if (!cr_version_str.empty()) { 90 NSArray* existing_chrome = [NSRunningApplication 91 runningApplicationsWithBundleIdentifier:cr_bundle_id]; 92 if ([existing_chrome count] == 0) 93 cr_version_str.clear(); 94 } 95 96 // ** 3: Read information from the Chrome bundle. 97 base::FilePath executable_path; 98 base::FilePath version_path; 99 base::FilePath framework_shlib_path; 100 if (!app_mode::GetChromeBundleInfo(cr_bundle_path, 101 cr_version_str.value(), 102 &executable_path, 103 &version_path, 104 &framework_shlib_path)) { 105 LOG(FATAL) << "Couldn't ready Chrome bundle info"; 106 } 107 base::FilePath app_mode_bundle_path = 108 base::mac::NSStringToFilePath([app_bundle bundlePath]); 109 110 // ** 4: Fill in ChromeAppModeInfo. 111 info->chrome_outer_bundle_path = cr_bundle_path; 112 info->chrome_versioned_path = version_path; 113 info->app_mode_bundle_path = app_mode_bundle_path; 114 115 // Read information about the this app shortcut from the Info.plist. 116 // Don't check for null-ness on optional items. 117 NSDictionary* info_plist = [app_bundle infoDictionary]; 118 CHECK(info_plist) << "couldn't get loader Info.plist"; 119 120 info->app_mode_id = SysNSStringToUTF8( 121 [info_plist objectForKey:app_mode::kCrAppModeShortcutIDKey]); 122 CHECK(info->app_mode_id.size()) << "couldn't get app shortcut ID"; 123 124 info->app_mode_name = SysNSStringToUTF16( 125 [info_plist objectForKey:app_mode::kCrAppModeShortcutNameKey]); 126 127 info->app_mode_url = SysNSStringToUTF8( 128 [info_plist objectForKey:app_mode::kCrAppModeShortcutURLKey]); 129 130 info->user_data_dir = base::mac::NSStringToFilePath( 131 [info_plist objectForKey:app_mode::kCrAppModeUserDataDirKey]); 132 133 info->profile_dir = base::mac::NSStringToFilePath( 134 [info_plist objectForKey:app_mode::kCrAppModeProfileDirKey]); 135 136 // ** 5: Open the framework. 137 StartFun ChromeAppModeStart = NULL; 138 void* cr_dylib = dlopen(framework_shlib_path.value().c_str(), RTLD_LAZY); 139 if (cr_dylib) { 140 // Find the entry point. 141 ChromeAppModeStart = (StartFun)dlsym(cr_dylib, "ChromeAppModeStart"); 142 if (!ChromeAppModeStart) 143 LOG(ERROR) << "Couldn't get entry point: " << dlerror(); 144 } else { 145 LOG(ERROR) << "Couldn't load framework: " << dlerror(); 146 } 147 148 if (ChromeAppModeStart) 149 return ChromeAppModeStart(info); 150 151 LOG(ERROR) << "Loading Chrome failed, launching Chrome with command line"; 152 CommandLine command_line(executable_path); 153 // The user_data_dir from the plist is actually the app data dir. 154 command_line.AppendSwitchPath( 155 switches::kUserDataDir, 156 info->user_data_dir.DirName().DirName().DirName()); 157 if (CommandLine::ForCurrentProcess()->HasSwitch( 158 app_mode::kLaunchedByChromeProcessId) || 159 info->app_mode_id == app_mode::kAppListModeId) { 160 // Pass --app-shim-error to have Chrome rebuild this shim. 161 // If Chrome has rebuilt this shim once already, then rebuilding doesn't fix 162 // the problem, so don't try again. 163 if (!CommandLine::ForCurrentProcess()->HasSwitch( 164 app_mode::kLaunchedAfterRebuild)) { 165 command_line.AppendSwitchPath(app_mode::kAppShimError, 166 app_mode_bundle_path); 167 } 168 } else { 169 // If the shim was launched directly (instead of by Chrome), first ask 170 // Chrome to launch the app. Chrome will launch the shim again, the same 171 // error will occur and be handled above. This approach allows the app to be 172 // started without blocking on fixing the shim and guarantees that the 173 // profile is loaded when Chrome receives --app-shim-error. 174 command_line.AppendSwitchPath(switches::kProfileDirectory, 175 info->profile_dir); 176 command_line.AppendSwitchASCII(switches::kAppId, info->app_mode_id); 177 } 178 // Launch the executable directly since base::mac::OpenApplicationWithPath 179 // uses LSOpenApplication which doesn't pass command line arguments if the 180 // application is already running. 181 if (!base::LaunchProcess(command_line, base::LaunchOptions(), NULL)) { 182 LOG(ERROR) << "Could not launch Chrome: " 183 << command_line.GetCommandLineString(); 184 return 1; 185 } 186 187 return 0; 188 } 189 190 } // namespace 191 192 __attribute__((visibility("default"))) 193 int main(int argc, char** argv) { 194 CommandLine::Init(argc, argv); 195 app_mode::ChromeAppModeInfo info; 196 197 // Hard coded info parameters. 198 info.major_version = app_mode::kCurrentChromeAppModeInfoMajorVersion; 199 info.minor_version = app_mode::kCurrentChromeAppModeInfoMinorVersion; 200 info.argc = argc; 201 info.argv = argv; 202 203 // Exit instead of returning to avoid the the removal of |main()| from stack 204 // backtraces under tail call optimization. 205 exit(LoadFrameworkAndStart(&info)); 206 } 207