// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // This is the browser side of the resource dispatcher, it receives requests // from the child process (i.e. [Renderer, Plugin, Worker]ProcessHost), and // dispatches them to URLRequests. It then fowards the messages from the // URLRequests back to the correct process for handling. // // See http://dev.chromium.org/developers/design-documents/multi-process-resource-loading #ifndef CHROME_BROWSER_RENDERER_HOST_RESOURCE_DISPATCHER_HOST_H_ #define CHROME_BROWSER_RENDERER_HOST_RESOURCE_DISPATCHER_HOST_H_ #include #include "base/observer_list.h" #include "base/process.h" #include "base/timer.h" #include "chrome/browser/renderer_host/resource_handler.h" #include "chrome/common/child_process_info.h" #include "chrome/common/filter_policy.h" #include "chrome/common/ipc_message.h" #include "net/url_request/url_request.h" #include "webkit/glue/resource_type.h" class CrossSiteResourceHandler; class DownloadFileManager; class DownloadRequestManager; class LoginHandler; class MessageLoop; class PluginService; class SafeBrowsingService; class SaveFileManager; class URLRequestContext; struct ViewHostMsg_Resource_Request; class ResourceDispatcherHost : public URLRequest::Delegate { public: // Implemented by the client of ResourceDispatcherHost to receive messages in // response to a resource load. The messages are intended to be forwarded to // the ResourceDispatcher in the child process via an IPC channel that the // client manages. // // NOTE: This class unfortunately cannot be named 'Delegate' because that // conflicts with the name of ResourceDispatcherHost's base class. // // If the receiver is unable to send a given message (i.e., if Send returns // false), then the ResourceDispatcherHost assumes the receiver has failed, // and the given request will be dropped. (This happens, for example, when a // renderer crashes and the channel dies). typedef IPC::Message::Sender Receiver; // Holds the data we would like to associate with each request class ExtraRequestInfo : public URLRequest::UserData { friend class ResourceDispatcherHost; public: ExtraRequestInfo(ResourceHandler* handler, ChildProcessInfo::ProcessType process_type, int process_id, int route_id, int request_id, bool mixed_content, ResourceType::Type resource_type, uint64 upload_size) : resource_handler(handler), cross_site_handler(NULL), login_handler(NULL), process_type(process_type), process_id(process_id), route_id(route_id), request_id(request_id), pending_data_count(0), is_download(false), pause_count(0), mixed_content(mixed_content), resource_type(resource_type), filter_policy(FilterPolicy::DONT_FILTER), last_load_state(net::LOAD_STATE_IDLE), upload_size(upload_size), last_upload_position(0), waiting_for_upload_progress_ack(false), memory_cost(0), is_paused(false), has_started_reading(false), paused_read_bytes(0) { } // Top-level ResourceHandler servicing this request. scoped_refptr resource_handler; // CrossSiteResourceHandler for this request, if it is a cross-site request. // (NULL otherwise.) This handler is part of the chain of ResourceHandlers // pointed to by resource_handler. CrossSiteResourceHandler* cross_site_handler; LoginHandler* login_handler; ChildProcessInfo::ProcessType process_type; int process_id; int route_id; int request_id; int pending_data_count; // Downloads allowed only as a top level request. bool allow_download; // Whether this is a download. bool is_download; // The number of clients that have called pause on this request. int pause_count; // Whether this request is served over HTTP and the main page was served // over HTTPS. bool mixed_content; ResourceType::Type resource_type; // Whether the content for this request should be filtered (on the renderer // side) to make it more secure: images are stamped, frame content is // replaced with an error message and all other resources are entirely // filtered out. FilterPolicy::Type filter_policy; net::LoadState last_load_state; uint64 upload_size; uint64 last_upload_position; base::TimeTicks last_upload_ticks; bool waiting_for_upload_progress_ack; // The approximate in-memory size (bytes) that we credited this request // as consuming in |outstanding_requests_memory_cost_map_|. int memory_cost; private: // Request is temporarily not handling network data. Should be used only // by the ResourceDispatcherHost, not the event handlers. bool is_paused; // Whether this request has started reading any bytes from the response // yet. Will be true after the first (unpaused) call to Read. bool has_started_reading; // How many bytes have been read while this request has been paused. int paused_read_bytes; }; class Observer { public: virtual void OnRequestStarted(ResourceDispatcherHost* resource_dispatcher, URLRequest* request) = 0; virtual void OnResponseCompleted( ResourceDispatcherHost* resource_dispatcher, URLRequest* request) = 0; virtual void OnReceivedRedirect(ResourceDispatcherHost* resource_dispatcher, URLRequest* request, const GURL& new_url) = 0; }; // Uniquely identifies a URLRequest. struct GlobalRequestID { GlobalRequestID() : process_id(-1), request_id(-1) { } GlobalRequestID(int process_id, int request_id) : process_id(process_id), request_id(request_id) { } int process_id; int request_id; bool operator<(const GlobalRequestID& other) const { if (process_id == other.process_id) return request_id < other.request_id; return process_id < other.process_id; } }; explicit ResourceDispatcherHost(MessageLoop* io_loop); ~ResourceDispatcherHost(); void Initialize(); // Puts the resource dispatcher host in an inactive state (unable to begin // new requests). Cancels all pending requests. void Shutdown(); // Begins a resource request with the given params on behalf of the specified // child process. Responses will be dispatched through the given receiver. The // process ID is used to lookup TabContents from routing_id's in the case of a // request from a renderer. request_context is the cookie/cache context to be // used for this request. // // If sync_result is non-null, then a SyncLoad reply will be generated, else // a normal asynchronous set of response messages will be generated. void BeginRequest(Receiver* receiver, ChildProcessInfo::ProcessType process_type, base::ProcessHandle process_handle, int process_id, int route_id, int request_id, const ViewHostMsg_Resource_Request& request, URLRequestContext* request_context, IPC::Message* sync_result); // Initiates a download from the browser process (as opposed to a resource // request from the renderer or another child process). void BeginDownload(const GURL& url, const GURL& referrer, int process_id, int route_id, URLRequestContext* request_context); // Initiates a save file from the browser process (as opposed to a resource // request from the renderer or another child process). void BeginSaveFile(const GURL& url, const GURL& referrer, int process_id, int route_id, URLRequestContext* request_context); // Cancels the given request if it still exists. We ignore cancels from the // renderer in the event of a download. void CancelRequest(int process_id, int request_id, bool from_renderer); // Decrements the pending_data_count for the request and resumes // the request if it was paused due to too many pending data // messages sent. void OnDataReceivedACK(int process_id, int request_id); // Resets the waiting_for_upload_progress_ack flag. void OnUploadProgressACK(int process_id, int request_id); // Returns true if it's ok to send the data. If there are already too many // data messages pending, it pauses the request and returns false. In this // case the caller should not send the data. bool WillSendData(int process_id, int request_id); // Pauses or resumes network activity for a particular request. void PauseRequest(int process_id, int request_id, bool pause); // Returns the number of pending requests. This is designed for the unittests int pending_requests() const { return static_cast(pending_requests_.size()); } // Intended for unit-tests only. Returns the memory cost of all the // outstanding requests (pending and blocked) for |process_id|. int GetOutstandingRequestsMemoryCost(int process_id) const; // Intended for unit-tests only. Overrides the outstanding requests bound. void set_max_outstanding_requests_cost_per_process(int limit) { max_outstanding_requests_cost_per_process_ = limit; } // The average private bytes increase of the browser for each new pending // request. Experimentally obtained. static const int kAvgBytesPerOutstandingRequest = 4400; DownloadFileManager* download_file_manager() const { return download_file_manager_; } DownloadRequestManager* download_request_manager() const { return download_request_manager_.get(); } SaveFileManager* save_file_manager() const { return save_file_manager_; } SafeBrowsingService* safe_browsing_service() const { return safe_browsing_; } MessageLoop* ui_loop() const { return ui_loop_; } // Called when the onunload handler for a cross-site request has finished. void OnClosePageACK(int process_id, int request_id); // Force cancels any pending requests for the given process. void CancelRequestsForProcess(int process_id); // Force cancels any pending requests for the given route id. This method // acts like CancelRequestsForProcess when route_id is -1. void CancelRequestsForRoute(int process_id, int route_id); // URLRequest::Delegate virtual void OnReceivedRedirect(URLRequest* request, const GURL& new_url); virtual void OnAuthRequired(URLRequest* request, net::AuthChallengeInfo* auth_info); virtual void OnSSLCertificateError(URLRequest* request, int cert_error, net::X509Certificate* cert); virtual void OnResponseStarted(URLRequest* request); virtual void OnReadCompleted(URLRequest* request, int bytes_read); void OnResponseCompleted(URLRequest* request); // Helper function to get our extra data out of a request. The given request // must have been one we created so that it has the proper extra data pointer. static ExtraRequestInfo* ExtraInfoForRequest(URLRequest* request) { ExtraRequestInfo* r = static_cast(request->user_data()); DLOG_IF(WARNING, !r) << "Request doesn't seem to have our data"; return r; } static const ExtraRequestInfo* ExtraInfoForRequest( const URLRequest* request) { const ExtraRequestInfo* r = static_cast(request->user_data()); DLOG_IF(WARNING, !r) << "Request doesn't seem to have our data"; return r; } // Adds an observer. The observer will be called on the IO thread. To // observe resource events on the UI thread, subscribe to the // NOTIFY_RESOURCE_* notifications of the notification service. void AddObserver(Observer* obs); // Removes an observer. void RemoveObserver(Observer* obs); // Retrieves a URLRequest. Must be called from the IO thread. URLRequest* GetURLRequest(GlobalRequestID request_id) const; // A test to determining whether a given request should be forwarded to the // download thread. bool ShouldDownload(const std::string& mime_type, const std::string& content_disposition); // Notifies our observers that a request has been cancelled. void NotifyResponseCompleted(URLRequest* request, int process_id); void RemovePendingRequest(int process_id, int request_id); // Causes all new requests for the route identified by // |process_id| and |route_id| to be blocked (not being // started) until ResumeBlockedRequestsForRoute or // CancelBlockedRequestsForRoute is called. void BlockRequestsForRoute(int process_id, int route_id); // Resumes any blocked request for the specified route id. void ResumeBlockedRequestsForRoute(int process_id, int route_id); // Cancels any blocked request for the specified route id. void CancelBlockedRequestsForRoute(int process_id, int route_id); private: FRIEND_TEST(ResourceDispatcherHostTest, TestBlockedRequestsProcessDies); FRIEND_TEST(ResourceDispatcherHostTest, IncrementOutstandingRequestsMemoryCost); FRIEND_TEST(ResourceDispatcherHostTest, CalculateApproximateMemoryCost); class ShutdownTask; friend class ShutdownTask; struct BlockedRequest { BlockedRequest(URLRequest* url_request, bool mixed_content) : url_request(url_request), mixed_content(mixed_content) { } URLRequest* url_request; bool mixed_content; }; // A shutdown helper that runs on the IO thread. void OnShutdown(); // Returns true if the request is paused. bool PauseRequestIfNeeded(ExtraRequestInfo* info); // Resumes the given request by calling OnResponseStarted or OnReadCompleted. void ResumeRequest(const GlobalRequestID& request_id); // Reads data from the response using our internal buffer as async IO. // Returns true if data is available immediately, false otherwise. If the // return value is false, we will receive a OnReadComplete() callback later. bool Read(URLRequest* request, int* bytes_read); // Internal function to finish an async IO which has completed. Returns // true if there is more data to read (e.g. we haven't read EOF yet and // no errors have occurred). bool CompleteRead(URLRequest *, int* bytes_read); // Internal function to finish handling the ResponseStarted message. Returns // true on success. bool CompleteResponseStarted(URLRequest* request); // Cancels the given request if it still exists. We ignore cancels from the // renderer in the event of a download. If |allow_delete| is true and no IO // is pending, the request is removed and deleted. void CancelRequest(int process_id, int request_id, bool from_renderer, bool allow_delete); // Helper function for regular and download requests. void BeginRequestInternal(URLRequest* request, bool mixed_content); // Updates the "cost" of outstanding requests for |process_id|. // The "cost" approximates how many bytes are consumed by all the in-memory // data structures supporting this request (URLRequest object, // HttpNetworkTransaction, etc...). // The value of |cost| is added to the running total, and the resulting // sum is returned. int IncrementOutstandingRequestsMemoryCost(int cost, int process_id); // Estimate how much heap space |request| will consume to run. static int CalculateApproximateMemoryCost(URLRequest* request); // The list of all requests that we have pending. This list is not really // optimized, and assumes that we have relatively few requests pending at once // since some operations require brute-force searching of the list. // // It may be enhanced in the future to provide some kind of prioritization // mechanism. We should also consider a hashtable or binary tree if it turns // out we have a lot of things here. typedef std::map PendingRequestList; // Deletes the pending request identified by the iterator passed in. // This function will invalidate the iterator passed in. Callers should // not rely on this iterator being valid on return. void RemovePendingRequest(const PendingRequestList::iterator& iter); // Notify our observers that we started receiving a response for a request. void NotifyResponseStarted(URLRequest* request, int process_id); // Notify our observers that a request has been redirected. void NofityReceivedRedirect(URLRequest* request, int process_id, const GURL& new_url); // Tries to handle the url with an external protocol. If the request is // handled, the function returns true. False otherwise. bool HandleExternalProtocol(int request_id, int process_id, int tab_contents_id, const GURL& url, ResourceType::Type resource_type, ResourceHandler* handler); void UpdateLoadStates(); void MaybeUpdateUploadProgress(ExtraRequestInfo *info, URLRequest *request); // Resumes or cancels (if |cancel_requests| is true) any blocked requests. void ProcessBlockedRequestsForRoute(int process_id, int route_id, bool cancel_requests); PendingRequestList pending_requests_; // We cache the UI message loop so we can create new UI-related objects on it. MessageLoop* ui_loop_; // We cache the IO loop to ensure that GetURLRequest is only called from the // IO thread. MessageLoop* io_loop_; // A timer that periodically calls UpdateLoadStates while pending_requests_ // is not empty. base::RepeatingTimer update_load_states_timer_; // We own the download file writing thread and manager scoped_refptr download_file_manager_; // Determines whether a download is allowed. scoped_refptr download_request_manager_; // We own the save file manager. scoped_refptr save_file_manager_; scoped_refptr safe_browsing_; // Request ID for browser initiated requests. request_ids generated by // child processes are counted up from 0, while browser created requests // start at -2 and go down from there. (We need to start at -2 because -1 is // used as a special value all over the resource_dispatcher_host for // uninitialized variables.) This way, we no longer have the unlikely (but // observed in the real world!) event where we have two requests with the same // request_id_. int request_id_; // List of objects observing resource dispatching. ObserverList observer_list_; PluginService* plugin_service_; // For running tasks. ScopedRunnableMethodFactory method_runner_; // True if the resource dispatcher host has been shut down. bool is_shutdown_; typedef std::vector BlockedRequestsList; typedef std::pair ProcessRouteIDs; typedef std::map BlockedRequestMap; BlockedRequestMap blocked_requests_map_; // Maps the process_ids to the approximate number of bytes // being used to service its resource requests. No entry implies 0 cost. typedef std::map OutstandingRequestsMemoryCostMap; OutstandingRequestsMemoryCostMap outstanding_requests_memory_cost_map_; // |max_outstanding_requests_cost_per_process_| is the upper bound on how // many outstanding requests can be issued per child process host. // The constraint is expressed in terms of bytes (where the cost of // individual requests is given by CalculateApproximateMemoryCost). // The total number of outstanding requests is roughly: // (max_outstanding_requests_cost_per_process_ / // kAvgBytesPerOutstandingRequest) int max_outstanding_requests_cost_per_process_; DISALLOW_COPY_AND_ASSIGN(ResourceDispatcherHost); }; #endif // CHROME_BROWSER_RENDERER_HOST_RESOURCE_DISPATCHER_HOST_H_