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 CONTENT_BROWSER_PLUGIN_PROCESS_HOST_H_ 6 #define CONTENT_BROWSER_PLUGIN_PROCESS_HOST_H_ 7 8 #include "build/build_config.h" 9 10 #include <list> 11 #include <map> 12 #include <set> 13 #include <string> 14 #include <vector> 15 16 #include "base/basictypes.h" 17 #include "base/compiler_specific.h" 18 #include "base/memory/ref_counted.h" 19 #include "base/process/process_handle.h" 20 #include "content/common/content_export.h" 21 #include "content/public/browser/browser_child_process_host_delegate.h" 22 #include "content/public/browser/browser_child_process_host_iterator.h" 23 #include "content/public/common/process_type.h" 24 #include "content/public/common/resource_type.h" 25 #include "content/public/common/webplugininfo.h" 26 #include "ipc/ipc_channel_proxy.h" 27 #include "ui/gfx/native_widget_types.h" 28 29 struct ResourceHostMsg_Request; 30 31 namespace gfx { 32 class Rect; 33 } 34 35 namespace IPC { 36 struct ChannelHandle; 37 } 38 39 namespace net { 40 class URLRequestContext; 41 } 42 43 namespace content { 44 class BrowserChildProcessHostImpl; 45 class ResourceContext; 46 47 // Represents the browser side of the browser <--> plugin communication 48 // channel. Different plugins run in their own process, but multiple instances 49 // of the same plugin run in the same process. There will be one 50 // PluginProcessHost per plugin process, matched with a corresponding 51 // PluginProcess running in the plugin process. The browser is responsible for 52 // starting the plugin process when a plugin is created that doesn't already 53 // have a process. After that, most of the communication is directly between 54 // the renderer and plugin processes. 55 class CONTENT_EXPORT PluginProcessHost : public BrowserChildProcessHostDelegate, 56 public IPC::Sender { 57 public: 58 class Client { 59 public: 60 // Returns an opaque unique identifier for the process requesting 61 // the channel. 62 virtual int ID() = 0; 63 // Returns the resource context for the renderer requesting the channel. 64 virtual ResourceContext* GetResourceContext() = 0; 65 virtual bool OffTheRecord() = 0; 66 virtual void SetPluginInfo(const WebPluginInfo& info) = 0; 67 virtual void OnFoundPluginProcessHost(PluginProcessHost* host) = 0; 68 virtual void OnSentPluginChannelRequest() = 0; 69 // The client should delete itself when one of these methods is called. 70 virtual void OnChannelOpened(const IPC::ChannelHandle& handle) = 0; 71 virtual void OnError() = 0; 72 73 protected: 74 virtual ~Client() {} 75 }; 76 77 PluginProcessHost(); 78 virtual ~PluginProcessHost(); 79 80 // IPC::Sender implementation: 81 virtual bool Send(IPC::Message* message) OVERRIDE; 82 83 // Initialize the new plugin process, returning true on success. This must 84 // be called before the object can be used. 85 bool Init(const WebPluginInfo& info); 86 87 // Force the plugin process to shutdown (cleanly). 88 void ForceShutdown(); 89 90 virtual bool OnMessageReceived(const IPC::Message& msg) OVERRIDE; 91 virtual void OnChannelConnected(int32 peer_pid) OVERRIDE; 92 virtual void OnChannelError() OVERRIDE; 93 94 // Tells the plugin process to create a new channel for communication with a 95 // renderer. When the plugin process responds with the channel name, 96 // OnChannelOpened in the client is called. 97 void OpenChannelToPlugin(Client* client); 98 99 // This function is called to cancel pending requests to open new channels. 100 void CancelPendingRequest(Client* client); 101 102 // This function is called to cancel sent requests to open new channels. 103 void CancelSentRequest(Client* client); 104 105 // This function is called on the IO thread once we receive a reply from the 106 // modal HTML dialog (in the form of a JSON string). This function forwards 107 // that reply back to the plugin that requested the dialog. 108 void OnModalDialogResponse(const std::string& json_retval, 109 IPC::Message* sync_result); 110 111 #if defined(OS_MACOSX) 112 // This function is called on the IO thread when the browser becomes the 113 // active application. 114 void OnAppActivation(); 115 #endif 116 117 const WebPluginInfo& info() const { return info_; } 118 119 #if defined(OS_WIN) 120 // Tracks plugin parent windows created on the browser UI thread. 121 void AddWindow(HWND window); 122 #endif 123 124 // Given a pid of a plugin process, returns the plugin information in |info| 125 // if we know about that process. Otherwise returns false. 126 // This method can be called on any thread. 127 static bool GetWebPluginInfoFromPluginPid(base::ProcessId pid, 128 WebPluginInfo* info); 129 130 private: 131 // Sends a message to the plugin process to request creation of a new channel 132 // for the given mime type. 133 void RequestPluginChannel(Client* client); 134 135 // Message handlers. 136 void OnChannelCreated(const IPC::ChannelHandle& channel_handle); 137 void OnChannelDestroyed(int renderer_id); 138 139 #if defined(OS_WIN) 140 void OnPluginWindowDestroyed(HWND window, HWND parent); 141 #endif 142 143 #if defined(OS_MACOSX) 144 void OnPluginShowWindow(uint32 window_id, gfx::Rect window_rect, 145 bool modal); 146 void OnPluginHideWindow(uint32 window_id, gfx::Rect window_rect); 147 void OnPluginSetCursorVisibility(bool visible); 148 #endif 149 150 virtual bool CanShutdown() OVERRIDE; 151 virtual void OnProcessCrashed(int exit_code) OVERRIDE; 152 153 void CancelRequests(); 154 155 // Callback for ResourceMessageFilter. 156 void GetContexts(const ResourceHostMsg_Request& request, 157 ResourceContext** resource_context, 158 net::URLRequestContext** request_context); 159 160 // These are channel requests that we are waiting to send to the 161 // plugin process once the channel is opened. 162 std::vector<Client*> pending_requests_; 163 164 // These are the channel requests that we have already sent to 165 // the plugin process, but haven't heard back about yet. 166 std::list<Client*> sent_requests_; 167 168 // Information about the plugin. 169 WebPluginInfo info_; 170 171 // The pid of the plugin process. 172 int pid_; 173 174 #if defined(OS_WIN) 175 // Tracks plugin parent windows created on the UI thread. 176 std::set<HWND> plugin_parent_windows_set_; 177 #endif 178 #if defined(OS_MACOSX) 179 // Tracks plugin windows currently visible. 180 std::set<uint32> plugin_visible_windows_set_; 181 // Tracks full screen windows currently visible. 182 std::set<uint32> plugin_fullscreen_windows_set_; 183 // Tracks modal windows currently visible. 184 std::set<uint32> plugin_modal_windows_set_; 185 // Tracks the current visibility of the cursor. 186 bool plugin_cursor_visible_; 187 #endif 188 189 // Map from render_process_id to its ResourceContext. Instead of storing the 190 // raw pointer, we store the struct below. This is needed because a renderer 191 // process can actually have multiple IPC channels to the same plugin process, 192 // depending on timing conditions with plugin instance creation and shutdown. 193 struct ResourceContextEntry { 194 ResourceContext* resource_context; 195 int ref_count; 196 }; 197 typedef std::map<int, ResourceContextEntry> ResourceContextMap; 198 ResourceContextMap resource_context_map_; 199 200 scoped_ptr<BrowserChildProcessHostImpl> process_; 201 202 DISALLOW_COPY_AND_ASSIGN(PluginProcessHost); 203 }; 204 205 class PluginProcessHostIterator 206 : public BrowserChildProcessHostTypeIterator<PluginProcessHost> { 207 public: 208 PluginProcessHostIterator() 209 : BrowserChildProcessHostTypeIterator<PluginProcessHost>( 210 PROCESS_TYPE_PLUGIN) {} 211 }; 212 213 } // namespace content 214 215 #endif // CONTENT_BROWSER_PLUGIN_PROCESS_HOST_H_ 216