OLD | NEW |
---|---|
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "content/browser/renderer_host/pepper/pepper_renderer_connection.h" | 5 #include "content/browser/renderer_host/pepper/pepper_renderer_connection.h" |
6 | 6 |
7 #include "base/bind.h" | |
8 #include "base/memory/ref_counted.h" | |
7 #include "content/browser/browser_child_process_host_impl.h" | 9 #include "content/browser/browser_child_process_host_impl.h" |
8 #include "content/browser/ppapi_plugin_process_host.h" | 10 #include "content/browser/ppapi_plugin_process_host.h" |
9 #include "content/browser/renderer_host/pepper/browser_ppapi_host_impl.h" | 11 #include "content/browser/renderer_host/pepper/browser_ppapi_host_impl.h" |
10 #include "content/common/pepper_renderer_instance_data.h" | 12 #include "content/common/pepper_renderer_instance_data.h" |
11 #include "content/common/view_messages.h" | 13 #include "content/common/view_messages.h" |
12 #include "content/browser/renderer_host/pepper/pepper_file_ref_host.h" | 14 #include "content/browser/renderer_host/pepper/pepper_file_ref_host.h" |
13 #include "content/browser/renderer_host/pepper/pepper_file_system_browser_host.h " | 15 #include "content/browser/renderer_host/pepper/pepper_file_system_browser_host.h " |
14 #include "content/public/browser/content_browser_client.h" | 16 #include "content/public/browser/content_browser_client.h" |
15 #include "content/public/common/content_client.h" | 17 #include "content/public/common/content_client.h" |
16 #include "ipc/ipc_message_macros.h" | 18 #include "ipc/ipc_message_macros.h" |
17 #include "ppapi/host/resource_host.h" | 19 #include "ppapi/host/resource_host.h" |
18 #include "ppapi/proxy/ppapi_message_utils.h" | 20 #include "ppapi/proxy/ppapi_message_utils.h" |
19 #include "ppapi/proxy/ppapi_messages.h" | 21 #include "ppapi/proxy/ppapi_messages.h" |
20 #include "ppapi/proxy/ppapi_message_utils.h" | 22 #include "ppapi/proxy/ppapi_message_utils.h" |
21 #include "ppapi/proxy/resource_message_params.h" | 23 #include "ppapi/proxy/resource_message_params.h" |
22 | 24 |
23 namespace content { | 25 namespace content { |
24 | 26 |
27 namespace { | |
28 | |
29 // Responsible for creating the pending resource hosts, holding their IDs until | |
30 // all of them have been created for a single message, and sending the reply to | |
31 // say that the hosts have been created. | |
32 class PendingHostCreator | |
33 : public base::RefCounted<PendingHostCreator> { | |
34 public: | |
35 PendingHostCreator(BrowserPpapiHostImpl* host, | |
36 BrowserMessageFilter* connection, | |
37 int routing_id, | |
38 int sequence_id, | |
39 size_t nested_msgs_size); | |
40 | |
41 // Adds the given resource host as a pending one. The host is remembered as | |
42 // host number |index|, and will ultimately be sent to the plugin to be | |
43 // attached to a real resource. | |
44 void AddPendingResourceHost( | |
45 size_t index, | |
46 scoped_ptr<ppapi::host::ResourceHost> resource_host); | |
47 | |
48 private: | |
49 friend class base::RefCounted<PendingHostCreator>; | |
50 | |
51 // When the last reference to this class is released, all of the resource | |
52 // hosts would have been added. This destructor sends the message to the | |
53 // plugin to tell it to attach real hosts to all of the pending hosts that | |
54 // have been added by this object. | |
55 ~PendingHostCreator(); | |
56 | |
57 BrowserPpapiHostImpl* host_; | |
58 BrowserMessageFilter* connection_; | |
59 int routing_id_; | |
60 int sequence_id_; | |
61 std::vector<int> pending_resource_host_ids_; | |
62 }; | |
63 | |
64 PendingHostCreator::PendingHostCreator(BrowserPpapiHostImpl* host, | |
65 BrowserMessageFilter* connection, | |
66 int routing_id, | |
67 int sequence_id, | |
68 size_t nested_msgs_size) | |
69 : host_(host), | |
70 connection_(connection), | |
71 routing_id_(routing_id), | |
72 sequence_id_(sequence_id), | |
73 pending_resource_host_ids_(nested_msgs_size, 0) {} | |
74 | |
75 void PendingHostCreator::AddPendingResourceHost( | |
76 size_t index, | |
77 scoped_ptr<ppapi::host::ResourceHost> resource_host) { | |
78 pending_resource_host_ids_[index] = | |
79 host_->GetPpapiHost()->AddPendingResourceHost(resource_host.Pass()); | |
80 } | |
81 | |
82 PendingHostCreator::~PendingHostCreator() { | |
83 connection_->Send(new PpapiHostMsg_CreateResourceHostsFromHostReply( | |
84 routing_id_, sequence_id_, pending_resource_host_ids_)); | |
85 } | |
86 | |
87 void HandleNestedCreateMessage( | |
88 BrowserPpapiHostImpl* host, | |
89 const scoped_refptr<PendingHostCreator>& creator, | |
90 const ppapi::proxy::ResourceMessageCallParams& params, | |
91 PP_Instance instance, | |
92 size_t index, | |
93 const IPC::Message& nested_msg) { | |
94 scoped_ptr<ppapi::host::ResourceHost> resource_host; | |
95 if (host->IsValidInstance(instance)) { | |
96 if (nested_msg.type() == PpapiHostMsg_FileRef_CreateExternal::ID) { | |
97 // FileRef_CreateExternal is only permitted from the renderer. Because | |
98 // of this, we handle this message here and not in | |
99 // content_browser_pepper_host_factory.cc. | |
100 base::FilePath external_path; | |
101 if (ppapi::UnpackMessage<PpapiHostMsg_FileRef_CreateExternal>( | |
102 nested_msg, &external_path)) { | |
103 resource_host.reset(new PepperFileRefHost( | |
104 host, instance, params.pp_resource(), external_path)); | |
105 } | |
106 } else if (nested_msg.type() == | |
107 PpapiHostMsg_FileSystem_CreateFromRenderer::ID) { | |
108 // Similarly, FileSystem_CreateFromRenderer is only permitted from the | |
109 // renderer. | |
110 std::string root_url; | |
111 PP_FileSystemType file_system_type; | |
112 if (ppapi::UnpackMessage<PpapiHostMsg_FileSystem_CreateFromRenderer>( | |
113 nested_msg, &root_url, &file_system_type)) { | |
114 PepperFileSystemBrowserHost* browser_host = | |
115 new PepperFileSystemBrowserHost( | |
116 host, instance, params.pp_resource(), file_system_type); | |
117 resource_host.reset(browser_host); | |
118 // Open the file system resource host. This is an asynchronous | |
119 // operation, and we must only add the pending resource host and | |
120 // send the message once it completes. | |
121 browser_host->OpenExisting( | |
122 GURL(root_url), | |
123 base::Bind(&PendingHostCreator::AddPendingResourceHost, | |
124 creator, | |
125 index, | |
126 base::Passed(&resource_host))); | |
127 // Do not fall through; the fall-through case adds the pending | |
128 // resource host to the list. We must do this asynchronously. | |
129 return; | |
130 } | |
131 } | |
132 } | |
133 | |
134 if (!resource_host.get()) { | |
135 resource_host = host->GetPpapiHost()->CreateResourceHost( | |
136 params, instance, nested_msg); | |
137 } | |
138 | |
139 if (resource_host.get()) | |
140 creator->AddPendingResourceHost(index, resource_host.Pass()); | |
141 } | |
142 | |
143 } // namespace | |
144 | |
25 PepperRendererConnection::PepperRendererConnection(int render_process_id) | 145 PepperRendererConnection::PepperRendererConnection(int render_process_id) |
26 : render_process_id_(render_process_id) { | 146 : render_process_id_(render_process_id) { |
27 // Only give the renderer permission for stable APIs. | 147 // Only give the renderer permission for stable APIs. |
28 in_process_host_.reset(new BrowserPpapiHostImpl(this, | 148 in_process_host_.reset(new BrowserPpapiHostImpl(this, |
29 ppapi::PpapiPermissions(), | 149 ppapi::PpapiPermissions(), |
30 "", | 150 "", |
31 base::FilePath(), | 151 base::FilePath(), |
32 base::FilePath(), | 152 base::FilePath(), |
33 true /* in_process */, | 153 true /* in_process */, |
34 false /* external_plugin */)); | 154 false /* external_plugin */)); |
(...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
85 | 205 |
86 return handled; | 206 return handled; |
87 } | 207 } |
88 | 208 |
89 void PepperRendererConnection::OnMsgCreateResourceHostsFromHost( | 209 void PepperRendererConnection::OnMsgCreateResourceHostsFromHost( |
90 int routing_id, | 210 int routing_id, |
91 int child_process_id, | 211 int child_process_id, |
92 const ppapi::proxy::ResourceMessageCallParams& params, | 212 const ppapi::proxy::ResourceMessageCallParams& params, |
93 PP_Instance instance, | 213 PP_Instance instance, |
94 const std::vector<IPC::Message>& nested_msgs) { | 214 const std::vector<IPC::Message>& nested_msgs) { |
95 BrowserPpapiHostImpl* host = GetHostForChildProcess(child_process_id); | 215 BrowserPpapiHostImpl* host = GetHostForChildProcess(child_process_id); |
yzshen1
2013/11/02 00:25:07
You could move line 220 - 221 here and return dire
Matt Giuca
2013/11/02 00:51:45
Done.
| |
216 scoped_refptr<PendingHostCreator> creator = | |
217 new PendingHostCreator( | |
218 host, this, routing_id, params.sequence(), nested_msgs.size()); | |
96 | 219 |
97 std::vector<int> pending_resource_host_ids(nested_msgs.size(), 0); | |
98 if (!host) { | 220 if (!host) { |
99 DLOG(ERROR) << "Invalid plugin process ID."; | 221 DLOG(ERROR) << "Invalid plugin process ID."; |
100 } else { | 222 } else { |
101 for (size_t i = 0; i < nested_msgs.size(); ++i) { | 223 for (size_t i = 0; i < nested_msgs.size(); ++i) { |
102 const IPC::Message& nested_msg = nested_msgs[i]; | 224 HandleNestedCreateMessage( |
103 scoped_ptr<ppapi::host::ResourceHost> resource_host; | 225 host, creator, params, instance, i, nested_msgs[i]); |
104 if (host->IsValidInstance(instance)) { | |
105 if (nested_msg.type() == PpapiHostMsg_FileRef_CreateExternal::ID) { | |
106 // FileRef_CreateExternal is only permitted from the renderer. Because | |
107 // of this, we handle this message here and not in | |
108 // content_browser_pepper_host_factory.cc. | |
109 base::FilePath external_path; | |
110 if (ppapi::UnpackMessage<PpapiHostMsg_FileRef_CreateExternal>( | |
111 nested_msg, &external_path)) { | |
112 resource_host.reset(new PepperFileRefHost( | |
113 host, instance, params.pp_resource(), external_path)); | |
114 } | |
115 } else if (nested_msg.type() == | |
116 PpapiHostMsg_FileSystem_CreateFromRenderer::ID) { | |
117 // Similarly, FileSystem_CreateFromRenderer is only permitted from the | |
118 // renderer. | |
119 std::string root_url; | |
120 PP_FileSystemType file_system_type; | |
121 if (ppapi::UnpackMessage<PpapiHostMsg_FileSystem_CreateFromRenderer>( | |
122 nested_msg, &root_url, &file_system_type)) { | |
123 resource_host.reset( | |
124 new PepperFileSystemBrowserHost(host, | |
125 instance, | |
126 params.pp_resource(), | |
127 GURL(root_url), | |
128 file_system_type)); | |
129 } | |
130 } | |
131 } | |
132 | |
133 if (!resource_host.get()) { | |
134 resource_host = host->GetPpapiHost()->CreateResourceHost( | |
135 params, instance, nested_msg); | |
136 } | |
137 | |
138 if (resource_host.get()) { | |
139 pending_resource_host_ids[i] = | |
140 host->GetPpapiHost()->AddPendingResourceHost(resource_host.Pass()); | |
141 } | |
142 } | 226 } |
143 } | 227 } |
144 | 228 |
145 Send(new PpapiHostMsg_CreateResourceHostsFromHostReply( | 229 // Note: All of the pending host IDs that were added as part of this |
146 routing_id, params.sequence(), pending_resource_host_ids)); | 230 // operation will automatically be sent to the plugin when |creator| is |
231 // released. This may happen immediately, or (if there are asynchronous | |
232 // requests to create resource hosts), once all of them complete. | |
147 } | 233 } |
148 | 234 |
149 void PepperRendererConnection::OnMsgDidCreateInProcessInstance( | 235 void PepperRendererConnection::OnMsgDidCreateInProcessInstance( |
150 PP_Instance instance, | 236 PP_Instance instance, |
151 const PepperRendererInstanceData& instance_data) { | 237 const PepperRendererInstanceData& instance_data) { |
152 PepperRendererInstanceData data = instance_data; | 238 PepperRendererInstanceData data = instance_data; |
153 data.render_process_id = render_process_id_; | 239 data.render_process_id = render_process_id_; |
154 in_process_host_->AddInstance(instance, data); | 240 in_process_host_->AddInstance(instance, data); |
155 } | 241 } |
156 | 242 |
157 void PepperRendererConnection::OnMsgDidDeleteInProcessInstance( | 243 void PepperRendererConnection::OnMsgDidDeleteInProcessInstance( |
158 PP_Instance instance) { | 244 PP_Instance instance) { |
159 in_process_host_->DeleteInstance(instance); | 245 in_process_host_->DeleteInstance(instance); |
160 } | 246 } |
161 | 247 |
162 } // namespace content | 248 } // namespace content |
OLD | NEW |