Upstream version 9.38.198.0
[platform/framework/web/crosswalk.git] / src / content / browser / loader / resource_dispatcher_host_impl.h
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 // This is the browser side of the resource dispatcher, it receives requests
6 // from the child process (i.e. [Renderer, Plugin, Worker]ProcessHost), and
7 // dispatches them to URLRequests. It then forwards the messages from the
8 // URLRequests back to the correct process for handling.
9 //
10 // See http://dev.chromium.org/developers/design-documents/multi-process-resource-loading
11
12 #ifndef CONTENT_BROWSER_LOADER_RESOURCE_DISPATCHER_HOST_IMPL_H_
13 #define CONTENT_BROWSER_LOADER_RESOURCE_DISPATCHER_HOST_IMPL_H_
14
15 #include <map>
16 #include <set>
17 #include <string>
18 #include <vector>
19
20 #include "base/basictypes.h"
21 #include "base/gtest_prod_util.h"
22 #include "base/memory/linked_ptr.h"
23 #include "base/memory/scoped_ptr.h"
24 #include "base/observer_list.h"
25 #include "base/time/time.h"
26 #include "base/timer/timer.h"
27 #include "content/browser/download/download_resource_handler.h"
28 #include "content/browser/loader/global_routing_id.h"
29 #include "content/browser/loader/resource_loader.h"
30 #include "content/browser/loader/resource_loader_delegate.h"
31 #include "content/browser/loader/resource_scheduler.h"
32 #include "content/common/content_export.h"
33 #include "content/common/resource_request_body.h"
34 #include "content/public/browser/child_process_data.h"
35 #include "content/public/browser/download_item.h"
36 #include "content/public/browser/download_url_parameters.h"
37 #include "content/public/browser/global_request_id.h"
38 #include "content/public/browser/notification_types.h"
39 #include "content/public/browser/resource_dispatcher_host.h"
40 #include "content/public/common/resource_type.h"
41 #include "ipc/ipc_message.h"
42 #include "net/cookies/canonical_cookie.h"
43 #include "net/url_request/url_request.h"
44
45 class ResourceHandler;
46 struct ResourceHostMsg_Request;
47
48 namespace net {
49 class URLRequestJobFactory;
50 }
51
52 namespace webkit_blob {
53 class ShareableFileReference;
54 }
55
56 namespace content {
57 class ResourceContext;
58 class ResourceDispatcherHostDelegate;
59 class ResourceMessageDelegate;
60 class ResourceMessageFilter;
61 class ResourceRequestInfoImpl;
62 class SaveFileManager;
63 class WebContentsImpl;
64 struct DownloadSaveInfo;
65 struct NavigationRequestInfo;
66 struct Referrer;
67
68 class CONTENT_EXPORT ResourceDispatcherHostImpl
69     : public ResourceDispatcherHost,
70       public ResourceLoaderDelegate {
71  public:
72   ResourceDispatcherHostImpl();
73   virtual ~ResourceDispatcherHostImpl();
74
75   // Returns the current ResourceDispatcherHostImpl. May return NULL if it
76   // hasn't been created yet.
77   static ResourceDispatcherHostImpl* Get();
78
79   // ResourceDispatcherHost implementation:
80   virtual void SetDelegate(ResourceDispatcherHostDelegate* delegate) OVERRIDE;
81   virtual void SetAllowCrossOriginAuthPrompt(bool value) OVERRIDE;
82   virtual DownloadInterruptReason BeginDownload(
83       scoped_ptr<net::URLRequest> request,
84       const Referrer& referrer,
85       bool is_content_initiated,
86       ResourceContext* context,
87       int child_id,
88       int route_id,
89       bool prefer_cache,
90       scoped_ptr<DownloadSaveInfo> save_info,
91       uint32 download_id,
92       const DownloadStartedCallback& started_callback) OVERRIDE;
93   virtual void ClearLoginDelegateForRequest(net::URLRequest* request) OVERRIDE;
94   virtual void BlockRequestsForRoute(int child_id, int route_id) OVERRIDE;
95   virtual void ResumeBlockedRequestsForRoute(
96       int child_id, int route_id) OVERRIDE;
97
98   // Puts the resource dispatcher host in an inactive state (unable to begin
99   // new requests).  Cancels all pending requests.
100   void Shutdown();
101
102   // Notify the ResourceDispatcherHostImpl of a new resource context.
103   void AddResourceContext(ResourceContext* context);
104
105   // Notify the ResourceDispatcherHostImpl of a resource context destruction.
106   void RemoveResourceContext(ResourceContext* context);
107
108   // Resumes a request that deferred at response start.
109   void ResumeResponseDeferredAtStart(const GlobalRequestID& id);
110
111   // Force cancels any pending requests for the given |context|. This is
112   // necessary to ensure that before |context| goes away, all requests
113   // for it are dead.
114   void CancelRequestsForContext(ResourceContext* context);
115
116   // Returns true if the message was a resource message that was processed.
117   bool OnMessageReceived(const IPC::Message& message,
118                          ResourceMessageFilter* filter);
119
120   // Initiates a save file from the browser process (as opposed to a resource
121   // request from the renderer or another child process).
122   void BeginSaveFile(const GURL& url,
123                      const Referrer& referrer,
124                      int child_id,
125                      int route_id,
126                      ResourceContext* context);
127
128   // Cancels the given request if it still exists.
129   void CancelRequest(int child_id, int request_id);
130
131   // Marks the request as "parked". This happens if a request is
132   // redirected cross-site and needs to be resumed by a new render view.
133   void MarkAsTransferredNavigation(const GlobalRequestID& id);
134
135   // Cancels a request previously marked as being transferred, for use when a
136   // navigation was cancelled.
137   void CancelTransferringNavigation(const GlobalRequestID& id);
138
139   // Resumes the request without transferring it to a new render view.
140   void ResumeDeferredNavigation(const GlobalRequestID& id);
141
142   // Returns the number of pending requests. This is designed for the unittests
143   int pending_requests() const {
144     return static_cast<int>(pending_loaders_.size());
145   }
146
147   // Intended for unit-tests only. Overrides the outstanding requests bound.
148   void set_max_outstanding_requests_cost_per_process(int limit) {
149     max_outstanding_requests_cost_per_process_ = limit;
150   }
151   void set_max_num_in_flight_requests_per_process(int limit) {
152     max_num_in_flight_requests_per_process_ = limit;
153   }
154   void set_max_num_in_flight_requests(int limit) {
155     max_num_in_flight_requests_ = limit;
156   }
157
158   // The average private bytes increase of the browser for each new pending
159   // request. Experimentally obtained.
160   static const int kAvgBytesPerOutstandingRequest = 4400;
161
162   SaveFileManager* save_file_manager() const {
163     return save_file_manager_.get();
164   }
165
166   // Called when a RenderViewHost is created.
167   void OnRenderViewHostCreated(int child_id, int route_id);
168
169   // Called when a RenderViewHost is deleted.
170   void OnRenderViewHostDeleted(int child_id, int route_id);
171
172   // Force cancels any pending requests for the given process.
173   void CancelRequestsForProcess(int child_id);
174
175   void OnUserGesture(WebContentsImpl* contents);
176
177   // Retrieves a net::URLRequest.  Must be called from the IO thread.
178   net::URLRequest* GetURLRequest(const GlobalRequestID& request_id);
179
180   void RemovePendingRequest(int child_id, int request_id);
181
182   // Cancels any blocked request for the specified route id.
183   void CancelBlockedRequestsForRoute(int child_id, int route_id);
184
185   // Maintains a collection of temp files created in support of
186   // the download_to_file capability. Used to grant access to the
187   // child process and to defer deletion of the file until it's
188   // no longer needed.
189   void RegisterDownloadedTempFile(
190       int child_id, int request_id,
191       const base::FilePath& file_path);
192   void UnregisterDownloadedTempFile(int child_id, int request_id);
193
194   // Needed for the sync IPC message dispatcher macros.
195   bool Send(IPC::Message* message);
196
197   // Indicates whether third-party sub-content can pop-up HTTP basic auth
198   // dialog boxes.
199   bool allow_cross_origin_auth_prompt();
200
201   ResourceDispatcherHostDelegate* delegate() {
202     return delegate_;
203   }
204
205   // Must be called after the ResourceRequestInfo has been created
206   // and associated with the request.
207   // |id| should be |content::DownloadItem::kInvalidId| to request automatic
208   // assignment.
209   scoped_ptr<ResourceHandler> CreateResourceHandlerForDownload(
210       net::URLRequest* request,
211       bool is_content_initiated,
212       bool must_download,
213       uint32 id,
214       scoped_ptr<DownloadSaveInfo> save_info,
215       const DownloadUrlParameters::OnStartedCallback& started_cb);
216
217   // Must be called after the ResourceRequestInfo has been created
218   // and associated with the request.  If |payload| is set to a non-empty value,
219   // the value will be sent to the old resource handler instead of cancelling
220   // it, except on HTTP errors.
221   scoped_ptr<ResourceHandler> MaybeInterceptAsStream(
222       net::URLRequest* request,
223       ResourceResponse* response,
224       std::string* payload);
225
226   void ClearSSLClientAuthHandlerForRequest(net::URLRequest* request);
227
228   ResourceScheduler* scheduler() { return scheduler_.get(); }
229
230   // Called by a ResourceHandler when it's ready to start reading data and
231   // sending it to the renderer. Returns true if there are enough file
232   // descriptors available for the shared memory buffer. If false is returned,
233   // the request should cancel.
234   bool HasSufficientResourcesForRequest(const net::URLRequest* request_);
235
236   // Called by a ResourceHandler after it has finished its request and is done
237   // using its shared memory buffer. Frees up that file descriptor to be used
238   // elsewhere.
239   void FinishedWithResourcesForRequest(const net::URLRequest* request_);
240
241   // Called by NavigationRequest to start a navigation request in the node
242   // identified by |frame_node_id|.
243   void NavigationRequest(const NavigationRequestInfo& info,
244                          scoped_refptr<ResourceRequestBody> request_body,
245                          int64 frame_node_id);
246
247  private:
248   friend class ResourceDispatcherHostTest;
249
250   FRIEND_TEST_ALL_PREFIXES(ResourceDispatcherHostTest,
251                            TestBlockedRequestsProcessDies);
252   FRIEND_TEST_ALL_PREFIXES(ResourceDispatcherHostTest,
253                            CalculateApproximateMemoryCost);
254   FRIEND_TEST_ALL_PREFIXES(ResourceDispatcherHostTest,
255                            DetachableResourceTimesOut);
256   FRIEND_TEST_ALL_PREFIXES(ResourceDispatcherHostTest,
257                            TestProcessCancelDetachableTimesOut);
258
259   class ShutdownTask;
260
261   struct OustandingRequestsStats {
262     int memory_cost;
263     int num_requests;
264   };
265
266   friend class ShutdownTask;
267   friend class ResourceMessageDelegate;
268
269   // ResourceLoaderDelegate implementation:
270   virtual ResourceDispatcherHostLoginDelegate* CreateLoginDelegate(
271       ResourceLoader* loader,
272       net::AuthChallengeInfo* auth_info) OVERRIDE;
273   virtual bool HandleExternalProtocol(ResourceLoader* loader,
274                                       const GURL& url) OVERRIDE;
275   virtual void DidStartRequest(ResourceLoader* loader) OVERRIDE;
276   virtual void DidReceiveRedirect(ResourceLoader* loader,
277                                   const GURL& new_url) OVERRIDE;
278   virtual void DidReceiveResponse(ResourceLoader* loader) OVERRIDE;
279   virtual void DidFinishLoading(ResourceLoader* loader) OVERRIDE;
280
281   // An init helper that runs on the IO thread.
282   void OnInit();
283
284   // A shutdown helper that runs on the IO thread.
285   void OnShutdown();
286
287   // Helper function for regular and download requests.
288   void BeginRequestInternal(scoped_ptr<net::URLRequest> request,
289                             scoped_ptr<ResourceHandler> handler);
290
291   void StartLoading(ResourceRequestInfoImpl* info,
292                     const linked_ptr<ResourceLoader>& loader);
293
294   // We keep track of how much memory each request needs and how many requests
295   // are issued by each renderer. These are known as OustandingRequestStats.
296   // Memory limits apply to all requests sent to us by the renderers. There is a
297   // limit for each renderer. File descriptor limits apply to requests that are
298   // receiving their body. These are known as in-flight requests. There is a
299   // global limit that applies for the browser process. Each render is allowed
300   // to use up to a fraction of that.
301
302   // Returns the OustandingRequestsStats for |info|'s renderer, or an empty
303   // struct if that renderer has no outstanding requests.
304   OustandingRequestsStats GetOutstandingRequestsStats(
305       const ResourceRequestInfoImpl& info);
306
307   // Updates |outstanding_requests_stats_map_| with the specified |stats| for
308   // the renderer that made the request in |info|.
309   void UpdateOutstandingRequestsStats(const ResourceRequestInfoImpl& info,
310                                       const OustandingRequestsStats& stats);
311
312   // Called every time an outstanding request is created or deleted. |count|
313   // indicates whether the request is new or deleted. |count| must be 1 or -1.
314   OustandingRequestsStats IncrementOutstandingRequestsMemory(
315       int count,
316       const ResourceRequestInfoImpl& info);
317
318   // Called every time an in flight request is issued or finished. |count|
319   // indicates whether the request is issuing or finishing. |count| must be 1
320   // or -1.
321   OustandingRequestsStats IncrementOutstandingRequestsCount(
322       int count,
323       const ResourceRequestInfoImpl& info);
324
325   // Estimate how much heap space |request| will consume to run.
326   static int CalculateApproximateMemoryCost(net::URLRequest* request);
327
328   // Force cancels any pending requests for the given route id.  This method
329   // acts like CancelRequestsForProcess when route_id is -1.
330   void CancelRequestsForRoute(int child_id, int route_id);
331
332   // The list of all requests that we have pending. This list is not really
333   // optimized, and assumes that we have relatively few requests pending at once
334   // since some operations require brute-force searching of the list.
335   //
336   // It may be enhanced in the future to provide some kind of prioritization
337   // mechanism. We should also consider a hashtable or binary tree if it turns
338   // out we have a lot of things here.
339   typedef std::map<GlobalRequestID, linked_ptr<ResourceLoader> > LoaderMap;
340
341   // Deletes the pending request identified by the iterator passed in.
342   // This function will invalidate the iterator passed in. Callers should
343   // not rely on this iterator being valid on return.
344   void RemovePendingLoader(const LoaderMap::iterator& iter);
345
346   // Checks all pending requests and updates the load states and upload
347   // progress if necessary.
348   void UpdateLoadStates();
349
350   // Resumes or cancels (if |cancel_requests| is true) any blocked requests.
351   void ProcessBlockedRequestsForRoute(int child_id,
352                                       int route_id,
353                                       bool cancel_requests);
354
355   void OnRequestResource(int routing_id,
356                          int request_id,
357                          const ResourceHostMsg_Request& request_data);
358   void OnSyncLoad(int request_id,
359                   const ResourceHostMsg_Request& request_data,
360                   IPC::Message* sync_result);
361
362   // Update the ResourceRequestInfo and internal maps when a request is
363   // transferred from one process to another.
364   void UpdateRequestForTransfer(int child_id,
365                                 int route_id,
366                                 int request_id,
367                                 const ResourceHostMsg_Request& request_data,
368                                 const linked_ptr<ResourceLoader>& loader);
369
370   void BeginRequest(int request_id,
371                     const ResourceHostMsg_Request& request_data,
372                     IPC::Message* sync_result,  // only valid for sync
373                     int route_id);  // only valid for async
374
375   // Creates a ResourceHandler to be used by BeginRequest() for normal resource
376   // loading.
377   scoped_ptr<ResourceHandler> CreateResourceHandler(
378       net::URLRequest* request,
379       const ResourceHostMsg_Request& request_data,
380       IPC::Message* sync_result,
381       int route_id,
382       int process_type,
383       int child_id,
384       ResourceContext* resource_context);
385
386   void OnDataDownloadedACK(int request_id);
387   void OnUploadProgressACK(int request_id);
388   void OnCancelRequest(int request_id);
389   void OnReleaseDownloadedFile(int request_id);
390
391   // Creates ResourceRequestInfoImpl for a download or page save.
392   // |download| should be true if the request is a file download.
393   ResourceRequestInfoImpl* CreateRequestInfo(
394       int child_id,
395       int route_id,
396       bool download,
397       ResourceContext* context);
398
399   // Relationship of resource being authenticated with the top level page.
400   enum HttpAuthRelationType {
401     HTTP_AUTH_RELATION_TOP,            // Top-level page itself
402     HTTP_AUTH_RELATION_SAME_DOMAIN,    // Sub-content from same domain
403     HTTP_AUTH_RELATION_BLOCKED_CROSS,  // Blocked Sub-content from cross domain
404     HTTP_AUTH_RELATION_ALLOWED_CROSS,  // Allowed Sub-content per command line
405     HTTP_AUTH_RELATION_LAST
406   };
407
408   HttpAuthRelationType HttpAuthRelationTypeOf(const GURL& request_url,
409                                               const GURL& first_party);
410
411   // Returns whether the URLRequest identified by |transferred_request_id| is
412   // currently in the process of being transferred to a different renderer.
413   // This happens if a request is redirected cross-site and needs to be resumed
414   // by a new render view.
415   bool IsTransferredNavigation(
416       const GlobalRequestID& transferred_request_id) const;
417
418   ResourceLoader* GetLoader(const GlobalRequestID& id) const;
419   ResourceLoader* GetLoader(int child_id, int request_id) const;
420
421   // Registers |delegate| to receive resource IPC messages targeted to the
422   // specified |id|.
423   void RegisterResourceMessageDelegate(const GlobalRequestID& id,
424                                        ResourceMessageDelegate* delegate);
425   void UnregisterResourceMessageDelegate(const GlobalRequestID& id,
426                                          ResourceMessageDelegate* delegate);
427
428   int BuildLoadFlagsForRequest(const ResourceHostMsg_Request& request_data,
429                                int child_id,
430                                bool is_sync_load);
431
432   LoaderMap pending_loaders_;
433
434   // Collection of temp files downloaded for child processes via
435   // the download_to_file mechanism. We avoid deleting them until
436   // the client no longer needs them.
437   typedef std::map<int, scoped_refptr<webkit_blob::ShareableFileReference> >
438       DeletableFilesMap;  // key is request id
439   typedef std::map<int, DeletableFilesMap>
440       RegisteredTempFiles;  // key is child process id
441   RegisteredTempFiles registered_temp_files_;
442
443   // A timer that periodically calls UpdateLoadStates while pending_requests_
444   // is not empty.
445   scoped_ptr<base::RepeatingTimer<ResourceDispatcherHostImpl> >
446       update_load_states_timer_;
447
448   // We own the save file manager.
449   scoped_refptr<SaveFileManager> save_file_manager_;
450
451   // Request ID for browser initiated requests. request_ids generated by
452   // child processes are counted up from 0, while browser created requests
453   // start at -2 and go down from there. (We need to start at -2 because -1 is
454   // used as a special value all over the resource_dispatcher_host for
455   // uninitialized variables.) This way, we no longer have the unlikely (but
456   // observed in the real world!) event where we have two requests with the same
457   // request_id_.
458   int request_id_;
459
460   // True if the resource dispatcher host has been shut down.
461   bool is_shutdown_;
462
463   typedef std::vector<linked_ptr<ResourceLoader> > BlockedLoadersList;
464   typedef std::map<GlobalRoutingID, BlockedLoadersList*> BlockedLoadersMap;
465   BlockedLoadersMap blocked_loaders_map_;
466
467   // Maps the child_ids to the approximate number of bytes
468   // being used to service its resource requests. No entry implies 0 cost.
469   typedef std::map<int, OustandingRequestsStats> OutstandingRequestsStatsMap;
470   OutstandingRequestsStatsMap outstanding_requests_stats_map_;
471
472   // |num_in_flight_requests_| is the total number of requests currently issued
473   // summed across all renderers.
474   int num_in_flight_requests_;
475
476   // |max_num_in_flight_requests_| is the upper bound on how many requests
477   // can be in flight at once. It's based on the maximum number of file
478   // descriptors open per process. We need a global limit for the browser
479   // process.
480   int max_num_in_flight_requests_;
481
482   // |max_num_in_flight_requests_| is the upper bound on how many requests
483   // can be issued at once. It's based on the maximum number of file
484   // descriptors open per process. We need a per-renderer limit so that no
485   // single renderer can hog the browser's limit.
486   int max_num_in_flight_requests_per_process_;
487
488   // |max_outstanding_requests_cost_per_process_| is the upper bound on how
489   // many outstanding requests can be issued per child process host.
490   // The constraint is expressed in terms of bytes (where the cost of
491   // individual requests is given by CalculateApproximateMemoryCost).
492   // The total number of outstanding requests is roughly:
493   //   (max_outstanding_requests_cost_per_process_ /
494   //       kAvgBytesPerOutstandingRequest)
495   int max_outstanding_requests_cost_per_process_;
496
497   // Time of the last user gesture. Stored so that we can add a load
498   // flag to requests occurring soon after a gesture to indicate they
499   // may be because of explicit user action.
500   base::TimeTicks last_user_gesture_time_;
501
502   // Used during IPC message dispatching so that the handlers can get a pointer
503   // to the source of the message.
504   ResourceMessageFilter* filter_;
505
506   ResourceDispatcherHostDelegate* delegate_;
507
508   bool allow_cross_origin_auth_prompt_;
509
510   // http://crbug.com/90971 - Assists in tracking down use-after-frees on
511   // shutdown.
512   std::set<const ResourceContext*> active_resource_contexts_;
513
514   typedef std::map<GlobalRequestID,
515                    ObserverList<ResourceMessageDelegate>*> DelegateMap;
516   DelegateMap delegate_map_;
517
518   scoped_ptr<ResourceScheduler> scheduler_;
519
520   DISALLOW_COPY_AND_ASSIGN(ResourceDispatcherHostImpl);
521 };
522
523 }  // namespace content
524
525 #endif  // CONTENT_BROWSER_LOADER_RESOURCE_DISPATCHER_HOST_IMPL_H_