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