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 #ifndef COMPONENTS_NACL_BROWSER_NACL_PROCESS_HOST_H_ 6 #define COMPONENTS_NACL_BROWSER_NACL_PROCESS_HOST_H_ 7 8 #include "build/build_config.h" 9 10 #include "base/files/file.h" 11 #include "base/files/file_path.h" 12 #include "base/files/file_util_proxy.h" 13 #include "base/memory/ref_counted.h" 14 #include "base/memory/shared_memory.h" 15 #include "base/memory/weak_ptr.h" 16 #include "base/message_loop/message_loop.h" 17 #include "base/process/process.h" 18 #include "components/nacl/common/nacl_types.h" 19 #include "content/public/browser/browser_child_process_host_delegate.h" 20 #include "content/public/browser/browser_child_process_host_iterator.h" 21 #include "ipc/ipc_channel_handle.h" 22 #include "native_client/src/public/nacl_file_info.h" 23 #include "net/socket/socket_descriptor.h" 24 #include "ppapi/shared_impl/ppapi_permissions.h" 25 #include "url/gurl.h" 26 27 namespace content { 28 class BrowserChildProcessHost; 29 class BrowserPpapiHost; 30 } 31 32 namespace IPC { 33 class ChannelProxy; 34 } 35 36 namespace nacl { 37 38 class NaClHostMessageFilter; 39 void* AllocateAddressSpaceASLR(base::ProcessHandle process, size_t size); 40 41 // Represents the browser side of the browser <--> NaCl communication 42 // channel. There will be one NaClProcessHost per NaCl process 43 // The browser is responsible for starting the NaCl process 44 // when requested by the renderer. 45 // After that, most of the communication is directly between NaCl plugin 46 // running in the renderer and NaCl processes. 47 class NaClProcessHost : public content::BrowserChildProcessHostDelegate { 48 public: 49 // manifest_url: the URL of the manifest of the Native Client plugin being 50 // executed. 51 // nexe_file: A file that corresponds to the nexe module to be loaded. 52 // nexe_token: A cache validation token for nexe_file. 53 // permissions: PPAPI permissions, to control access to private APIs. 54 // render_view_id: RenderView routing id, to control access to private APIs. 55 // permission_bits: controls which interfaces the NaCl plugin can use. 56 // uses_irt: whether the launched process should use the IRT. 57 // uses_nonsfi_mode: whether the program should be loaded under non-SFI mode. 58 // enable_dyncode_syscalls: whether the launched process should allow dyncode 59 // and mmap with PROT_EXEC. 60 // enable_exception_handling: whether the launched process should allow 61 // hardware exception handling. 62 // enable_crash_throttling: whether a crash of this process contributes 63 // to the crash throttling statistics, and also 64 // whether this process should not start when too 65 // many crashes have been observed. 66 // off_the_record: was the process launched from an incognito renderer? 67 // profile_directory: is the path of current profile directory. 68 NaClProcessHost(const GURL& manifest_url, 69 base::File nexe_file, 70 const NaClFileToken& nexe_token, 71 ppapi::PpapiPermissions permissions, 72 int render_view_id, 73 uint32 permission_bits, 74 bool uses_irt, 75 bool uses_nonsfi_mode, 76 bool enable_dyncode_syscalls, 77 bool enable_exception_handling, 78 bool enable_crash_throttling, 79 bool off_the_record, 80 const base::FilePath& profile_directory); 81 virtual ~NaClProcessHost(); 82 83 virtual void OnProcessCrashed(int exit_status) OVERRIDE; 84 85 // Do any minimal work that must be done at browser startup. 86 static void EarlyStartup(); 87 88 // Specifies throttling time in milliseconds for PpapiHostMsg_Keepalive IPCs. 89 static void SetPpapiKeepAliveThrottleForTesting(unsigned milliseconds); 90 91 // Initialize the new NaCl process. Result is returned by sending ipc 92 // message reply_msg. 93 void Launch(NaClHostMessageFilter* nacl_host_message_filter, 94 IPC::Message* reply_msg, 95 const base::FilePath& manifest_path); 96 97 virtual void OnChannelConnected(int32 peer_pid) OVERRIDE; 98 99 #if defined(OS_WIN) 100 void OnProcessLaunchedByBroker(base::ProcessHandle handle); 101 void OnDebugExceptionHandlerLaunchedByBroker(bool success); 102 #endif 103 104 bool Send(IPC::Message* msg); 105 106 content::BrowserChildProcessHost* process() { return process_.get(); } 107 content::BrowserPpapiHost* browser_ppapi_host() { return ppapi_host_.get(); } 108 109 private: 110 bool LaunchNaClGdb(); 111 112 // Mark the process as using a particular GDB debug stub port and notify 113 // listeners (if the port is not kGdbDebugStubPortUnknown). 114 void SetDebugStubPort(int port); 115 116 #if defined(OS_POSIX) 117 // Create bound TCP socket in the browser process so that the NaCl GDB debug 118 // stub can use it to accept incoming connections even when the Chrome sandbox 119 // is enabled. 120 net::SocketDescriptor GetDebugStubSocketHandle(); 121 #endif 122 123 #if defined(OS_WIN) 124 // Called when the debug stub port has been selected. 125 void OnDebugStubPortSelected(uint16_t debug_stub_port); 126 #endif 127 128 bool LaunchSelLdr(); 129 130 // BrowserChildProcessHostDelegate implementation: 131 virtual bool OnMessageReceived(const IPC::Message& msg) OVERRIDE; 132 virtual void OnProcessLaunched() OVERRIDE; 133 134 void OnResourcesReady(); 135 136 // Enable the PPAPI proxy only for NaCl processes corresponding to a renderer. 137 bool enable_ppapi_proxy() { return render_view_id_ != 0; } 138 139 // Sends the reply message to the renderer who is waiting for the plugin 140 // to load. Returns true on success. 141 bool ReplyToRenderer( 142 const IPC::ChannelHandle& ppapi_channel_handle, 143 const IPC::ChannelHandle& trusted_channel_handle, 144 const IPC::ChannelHandle& manifest_service_channel_handle); 145 146 // Sends the reply with error message to the renderer. 147 void SendErrorToRenderer(const std::string& error_message); 148 149 // Sends the reply message to the renderer. Either result or 150 // error message must be empty. 151 void SendMessageToRenderer(const NaClLaunchResult& result, 152 const std::string& error_message); 153 154 // Sends the message to the NaCl process to load the plugin. Returns true 155 // on success. 156 bool StartNaClExecution(); 157 158 // Does post-process-launching tasks for starting the NaCl process once 159 // we have a connection. 160 // 161 // Returns false on failure. 162 bool StartWithLaunchedProcess(); 163 164 // Message handlers for validation caching. 165 void OnQueryKnownToValidate(const std::string& signature, bool* result); 166 void OnSetKnownToValidate(const std::string& signature); 167 void OnResolveFileToken(uint64 file_token_lo, uint64 file_token_hi, 168 IPC::Message* reply_msg); 169 void OnResolveFileTokenAsync(uint64 file_token_lo, uint64 file_token_hi); 170 void FileResolved(const base::FilePath& file_path, 171 IPC::Message* reply_msg, 172 base::File file); 173 void FileResolvedAsync(uint64_t file_token_lo, 174 uint64_t file_token_hi, 175 const base::FilePath& file_path, 176 base::File file); 177 #if defined(OS_WIN) 178 // Message handler for Windows hardware exception handling. 179 void OnAttachDebugExceptionHandler(const std::string& info, 180 IPC::Message* reply_msg); 181 bool AttachDebugExceptionHandler(const std::string& info, 182 IPC::Message* reply_msg); 183 #endif 184 185 // Called when the PPAPI IPC channels to the browser/renderer have been 186 // created. 187 void OnPpapiChannelsCreated( 188 const IPC::ChannelHandle& browser_channel_handle, 189 const IPC::ChannelHandle& ppapi_renderer_channel_handle, 190 const IPC::ChannelHandle& trusted_renderer_channel_handle, 191 const IPC::ChannelHandle& manifest_service_channel_handle); 192 193 GURL manifest_url_; 194 base::File nexe_file_; 195 NaClFileToken nexe_token_; 196 197 ppapi::PpapiPermissions permissions_; 198 199 #if defined(OS_WIN) 200 // This field becomes true when the broker successfully launched 201 // the NaCl loader. 202 bool process_launched_by_broker_; 203 #endif 204 // The NaClHostMessageFilter that requested this NaCl process. We use 205 // this for sending the reply once the process has started. 206 scoped_refptr<NaClHostMessageFilter> nacl_host_message_filter_; 207 208 // The reply message to send. We must always send this message when the 209 // sub-process either succeeds or fails to unblock the renderer waiting for 210 // the reply. NULL when there is no reply to send. 211 IPC::Message* reply_msg_; 212 #if defined(OS_WIN) 213 bool debug_exception_handler_requested_; 214 scoped_ptr<IPC::Message> attach_debug_exception_handler_reply_msg_; 215 #endif 216 217 // The file path to the manifest is passed to nacl-gdb when it is used to 218 // debug the NaCl loader. 219 base::FilePath manifest_path_; 220 221 scoped_ptr<content::BrowserChildProcessHost> process_; 222 223 bool uses_irt_; 224 bool uses_nonsfi_mode_; 225 226 bool enable_debug_stub_; 227 bool enable_dyncode_syscalls_; 228 bool enable_exception_handling_; 229 bool enable_crash_throttling_; 230 231 bool off_the_record_; 232 233 const base::FilePath profile_directory_; 234 235 // Channel proxy to terminate the NaCl-Browser PPAPI channel. 236 scoped_ptr<IPC::ChannelProxy> ipc_proxy_channel_; 237 // Browser host for plugin process. 238 scoped_ptr<content::BrowserPpapiHost> ppapi_host_; 239 240 int render_view_id_; 241 242 // Throttling time in milliseconds for PpapiHostMsg_Keepalive IPCs. 243 static unsigned keepalive_throttle_interval_milliseconds_; 244 245 // Shared memory provided to the plugin and renderer for 246 // reporting crash information. 247 base::SharedMemory crash_info_shmem_; 248 249 base::File socket_for_renderer_; 250 base::File socket_for_sel_ldr_; 251 252 base::WeakPtrFactory<NaClProcessHost> weak_factory_; 253 254 DISALLOW_COPY_AND_ASSIGN(NaClProcessHost); 255 }; 256 257 } // namespace nacl 258 259 #endif // COMPONENTS_NACL_BROWSER_NACL_PROCESS_HOST_H_ 260