• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 #include "content/browser/renderer_host/render_widget_helper.h"
6 
7 #include "base/bind.h"
8 #include "base/bind_helpers.h"
9 #include "base/lazy_instance.h"
10 #include "base/posix/eintr_wrapper.h"
11 #include "base/threading/thread.h"
12 #include "base/threading/thread_restrictions.h"
13 #include "content/browser/gpu/gpu_process_host_ui_shim.h"
14 #include "content/browser/gpu/gpu_surface_tracker.h"
15 #include "content/browser/loader/resource_dispatcher_host_impl.h"
16 #include "content/browser/renderer_host/render_process_host_impl.h"
17 #include "content/browser/renderer_host/render_view_host_impl.h"
18 #include "content/browser/dom_storage/session_storage_namespace_impl.h"
19 #include "content/common/view_messages.h"
20 
21 namespace content {
22 namespace {
23 
24 typedef std::map<int, RenderWidgetHelper*> WidgetHelperMap;
25 base::LazyInstance<WidgetHelperMap> g_widget_helpers =
26     LAZY_INSTANCE_INITIALIZER;
27 
AddWidgetHelper(int render_process_id,const scoped_refptr<RenderWidgetHelper> & widget_helper)28 void AddWidgetHelper(int render_process_id,
29                      const scoped_refptr<RenderWidgetHelper>& widget_helper) {
30   DCHECK_CURRENTLY_ON(BrowserThread::IO);
31   // We don't care if RenderWidgetHelpers overwrite an existing process_id. Just
32   // want this to be up to date.
33   g_widget_helpers.Get()[render_process_id] = widget_helper.get();
34 }
35 
36 }  // namespace
37 
RenderWidgetHelper()38 RenderWidgetHelper::RenderWidgetHelper()
39     : render_process_id_(-1),
40       resource_dispatcher_host_(NULL) {
41 }
42 
~RenderWidgetHelper()43 RenderWidgetHelper::~RenderWidgetHelper() {
44   DCHECK_CURRENTLY_ON(BrowserThread::IO);
45 
46   // Delete this RWH from the map if it is found.
47   WidgetHelperMap& widget_map = g_widget_helpers.Get();
48   WidgetHelperMap::iterator it = widget_map.find(render_process_id_);
49   if (it != widget_map.end() && it->second == this)
50     widget_map.erase(it);
51 
52 #if defined(OS_POSIX) && !defined(OS_ANDROID)
53   ClearAllocatedDIBs();
54 #endif
55 }
56 
Init(int render_process_id,ResourceDispatcherHostImpl * resource_dispatcher_host)57 void RenderWidgetHelper::Init(
58     int render_process_id,
59     ResourceDispatcherHostImpl* resource_dispatcher_host) {
60   render_process_id_ = render_process_id;
61   resource_dispatcher_host_ = resource_dispatcher_host;
62 
63   BrowserThread::PostTask(
64       BrowserThread::IO, FROM_HERE,
65       base::Bind(&AddWidgetHelper,
66                  render_process_id_, make_scoped_refptr(this)));
67 }
68 
GetNextRoutingID()69 int RenderWidgetHelper::GetNextRoutingID() {
70   return next_routing_id_.GetNext() + 1;
71 }
72 
73 // static
FromProcessHostID(int render_process_host_id)74 RenderWidgetHelper* RenderWidgetHelper::FromProcessHostID(
75     int render_process_host_id) {
76   DCHECK_CURRENTLY_ON(BrowserThread::IO);
77   WidgetHelperMap::const_iterator ci = g_widget_helpers.Get().find(
78       render_process_host_id);
79   return (ci == g_widget_helpers.Get().end())? NULL : ci->second;
80 }
81 
ResumeDeferredNavigation(const GlobalRequestID & request_id)82 void RenderWidgetHelper::ResumeDeferredNavigation(
83     const GlobalRequestID& request_id) {
84   BrowserThread::PostTask(
85       BrowserThread::IO, FROM_HERE,
86       base::Bind(&RenderWidgetHelper::OnResumeDeferredNavigation,
87                  this,
88                  request_id));
89 }
90 
ResumeResponseDeferredAtStart(const GlobalRequestID & request_id)91 void RenderWidgetHelper::ResumeResponseDeferredAtStart(
92     const GlobalRequestID& request_id) {
93   BrowserThread::PostTask(
94       BrowserThread::IO,
95       FROM_HERE,
96       base::Bind(&RenderWidgetHelper::OnResumeResponseDeferredAtStart,
97                  this,
98                  request_id));
99 }
100 
ResumeRequestsForView(int route_id)101 void RenderWidgetHelper::ResumeRequestsForView(int route_id) {
102   // We only need to resume blocked requests if we used a valid route_id.
103   // See CreateNewWindow.
104   if (route_id != MSG_ROUTING_NONE) {
105     BrowserThread::PostTask(
106         BrowserThread::IO, FROM_HERE,
107         base::Bind(&RenderWidgetHelper::OnResumeRequestsForView,
108             this, route_id));
109   }
110 }
111 
OnResumeDeferredNavigation(const GlobalRequestID & request_id)112 void RenderWidgetHelper::OnResumeDeferredNavigation(
113     const GlobalRequestID& request_id) {
114   resource_dispatcher_host_->ResumeDeferredNavigation(request_id);
115 }
116 
OnResumeResponseDeferredAtStart(const GlobalRequestID & request_id)117 void RenderWidgetHelper::OnResumeResponseDeferredAtStart(
118     const GlobalRequestID& request_id) {
119   resource_dispatcher_host_->ResumeResponseDeferredAtStart(request_id);
120 }
121 
CreateNewWindow(const ViewHostMsg_CreateWindow_Params & params,bool no_javascript_access,base::ProcessHandle render_process,int * route_id,int * main_frame_route_id,int * surface_id,SessionStorageNamespace * session_storage_namespace)122 void RenderWidgetHelper::CreateNewWindow(
123     const ViewHostMsg_CreateWindow_Params& params,
124     bool no_javascript_access,
125     base::ProcessHandle render_process,
126     int* route_id,
127     int* main_frame_route_id,
128     int* surface_id,
129     SessionStorageNamespace* session_storage_namespace) {
130   if (params.opener_suppressed || no_javascript_access) {
131     // If the opener is supppressed or script access is disallowed, we should
132     // open the window in a new BrowsingInstance, and thus a new process. That
133     // means the current renderer process will not be able to route messages to
134     // it. Because of this, we will immediately show and navigate the window
135     // in OnCreateWindowOnUI, using the params provided here.
136     *route_id = MSG_ROUTING_NONE;
137     *main_frame_route_id = MSG_ROUTING_NONE;
138     *surface_id = 0;
139   } else {
140     *route_id = GetNextRoutingID();
141     *main_frame_route_id = GetNextRoutingID();
142     *surface_id = GpuSurfaceTracker::Get()->AddSurfaceForRenderer(
143         render_process_id_, *route_id);
144     // Block resource requests until the view is created, since the HWND might
145     // be needed if a response ends up creating a plugin.
146     resource_dispatcher_host_->BlockRequestsForRoute(
147         render_process_id_, *route_id);
148     resource_dispatcher_host_->BlockRequestsForRoute(
149         render_process_id_, *main_frame_route_id);
150   }
151 
152   BrowserThread::PostTask(
153       BrowserThread::UI, FROM_HERE,
154       base::Bind(&RenderWidgetHelper::OnCreateWindowOnUI,
155                  this, params, *route_id, *main_frame_route_id,
156                  make_scoped_refptr(session_storage_namespace)));
157 }
158 
OnCreateWindowOnUI(const ViewHostMsg_CreateWindow_Params & params,int route_id,int main_frame_route_id,SessionStorageNamespace * session_storage_namespace)159 void RenderWidgetHelper::OnCreateWindowOnUI(
160     const ViewHostMsg_CreateWindow_Params& params,
161     int route_id,
162     int main_frame_route_id,
163     SessionStorageNamespace* session_storage_namespace) {
164   RenderViewHostImpl* host =
165       RenderViewHostImpl::FromID(render_process_id_, params.opener_id);
166   if (host)
167     host->CreateNewWindow(route_id, main_frame_route_id, params,
168         session_storage_namespace);
169 }
170 
OnResumeRequestsForView(int route_id)171 void RenderWidgetHelper::OnResumeRequestsForView(int route_id) {
172   resource_dispatcher_host_->ResumeBlockedRequestsForRoute(
173       render_process_id_, route_id);
174 }
175 
CreateNewWidget(int opener_id,blink::WebPopupType popup_type,int * route_id,int * surface_id)176 void RenderWidgetHelper::CreateNewWidget(int opener_id,
177                                          blink::WebPopupType popup_type,
178                                          int* route_id,
179                                          int* surface_id) {
180   *route_id = GetNextRoutingID();
181   *surface_id = GpuSurfaceTracker::Get()->AddSurfaceForRenderer(
182       render_process_id_, *route_id);
183   BrowserThread::PostTask(
184       BrowserThread::UI, FROM_HERE,
185       base::Bind(
186           &RenderWidgetHelper::OnCreateWidgetOnUI, this, opener_id, *route_id,
187           popup_type));
188 }
189 
CreateNewFullscreenWidget(int opener_id,int * route_id,int * surface_id)190 void RenderWidgetHelper::CreateNewFullscreenWidget(int opener_id,
191                                                    int* route_id,
192                                                    int* surface_id) {
193   *route_id = GetNextRoutingID();
194   *surface_id = GpuSurfaceTracker::Get()->AddSurfaceForRenderer(
195       render_process_id_, *route_id);
196   BrowserThread::PostTask(
197       BrowserThread::UI, FROM_HERE,
198       base::Bind(
199           &RenderWidgetHelper::OnCreateFullscreenWidgetOnUI, this,
200           opener_id, *route_id));
201 }
202 
OnCreateWidgetOnUI(int opener_id,int route_id,blink::WebPopupType popup_type)203 void RenderWidgetHelper::OnCreateWidgetOnUI(
204     int opener_id, int route_id, blink::WebPopupType popup_type) {
205   RenderViewHostImpl* host = RenderViewHostImpl::FromID(
206       render_process_id_, opener_id);
207   if (host)
208     host->CreateNewWidget(route_id, popup_type);
209 }
210 
OnCreateFullscreenWidgetOnUI(int opener_id,int route_id)211 void RenderWidgetHelper::OnCreateFullscreenWidgetOnUI(int opener_id,
212                                                       int route_id) {
213   RenderViewHostImpl* host = RenderViewHostImpl::FromID(
214       render_process_id_, opener_id);
215   if (host)
216     host->CreateNewFullscreenWidget(route_id);
217 }
218 
219 #if defined(OS_POSIX) && !defined(OS_ANDROID)
AllocTransportDIB(uint32 size,bool cache_in_browser,TransportDIB::Handle * result)220 void RenderWidgetHelper::AllocTransportDIB(uint32 size,
221                                            bool cache_in_browser,
222                                            TransportDIB::Handle* result) {
223   scoped_ptr<base::SharedMemory> shared_memory(new base::SharedMemory());
224   if (!shared_memory->CreateAnonymous(size)) {
225     result->fd = -1;
226     result->auto_close = false;
227     return;
228   }
229 
230   shared_memory->GiveToProcess(0 /* pid, not needed */, result);
231 
232   if (cache_in_browser) {
233     // Keep a copy of the file descriptor around
234     base::AutoLock locked(allocated_dibs_lock_);
235     allocated_dibs_[shared_memory->id()] = dup(result->fd);
236   }
237 }
238 
FreeTransportDIB(TransportDIB::Id dib_id)239 void RenderWidgetHelper::FreeTransportDIB(TransportDIB::Id dib_id) {
240   base::AutoLock locked(allocated_dibs_lock_);
241 
242   const std::map<TransportDIB::Id, int>::iterator
243     i = allocated_dibs_.find(dib_id);
244 
245   if (i != allocated_dibs_.end()) {
246     if (IGNORE_EINTR(close(i->second)) < 0)
247       PLOG(ERROR) << "close";
248     allocated_dibs_.erase(i);
249   } else {
250     DLOG(WARNING) << "Renderer asked us to free unknown transport DIB";
251   }
252 }
253 
ClearAllocatedDIBs()254 void RenderWidgetHelper::ClearAllocatedDIBs() {
255   for (std::map<TransportDIB::Id, int>::iterator
256        i = allocated_dibs_.begin(); i != allocated_dibs_.end(); ++i) {
257     if (IGNORE_EINTR(close(i->second)) < 0)
258       PLOG(ERROR) << "close: " << i->first;
259   }
260 
261   allocated_dibs_.clear();
262 }
263 #endif
264 
265 }  // namespace content
266