// Copyright (c) 2012 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. #ifndef WEBKIT_MEDIA_BUFFERED_RESOURCE_LOADER_H_ #define WEBKIT_MEDIA_BUFFERED_RESOURCE_LOADER_H_ #include #include "base/callback.h" #include "base/memory/scoped_ptr.h" #include "base/timer.h" #include "googleurl/src/gurl.h" #include "media/base/seekable_buffer.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h" #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebURLLoader.h" #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebURLLoaderClient.h" #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebURLRequest.h" #include "webkit/media/active_loader.h" namespace media { class MediaLog; class SeekableBuffer; } namespace webkit_media { const int64 kPositionNotSpecified = -1; const char kHttpScheme[] = "http"; const char kHttpsScheme[] = "https"; const char kDataScheme[] = "data"; // This class works inside demuxer thread and render thread. It contains a // WebURLLoader and does the actual resource loading. This object does // buffering internally, it defers the resource loading if buffer is full // and un-defers the resource loading if it is under buffered. class BufferedResourceLoader : public WebKit::WebURLLoaderClient { public: // kNeverDefer - Aggresively buffer; never defer loading while paused. // kReadThenDefer - Request only enough data to fulfill read requests. // kThresholdDefer - Try to keep amount of buffered data at a threshold. enum DeferStrategy { kNeverDefer, kReadThenDefer, kThresholdDefer, }; // Status codes for start/read operations on BufferedResourceLoader. enum Status { // Everything went as planned. kOk, // The operation failed, which may have been due to: // - Page navigation // - Server replied 4xx/5xx // - The response was invalid // - Connection was terminated // // At this point you should delete the loader. kFailed, // The loader will never be able to satisfy the read request. Please stop, // delete, create a new loader, and try again. kCacheMiss, }; // |url| - URL for the resource to be loaded. // |first_byte_position| - First byte to start loading from, // |kPositionNotSpecified| for not specified. // |last_byte_position| - Last byte to be loaded, // |kPositionNotSpecified| for not specified. // |strategy| is the initial loading strategy to use. // |bitrate| is the bitrate of the media, 0 if unknown. // |playback_rate| is the current playback rate of the media. BufferedResourceLoader(const GURL& url, int64 first_byte_position, int64 last_byte_position, DeferStrategy strategy, int bitrate, float playback_rate, media::MediaLog* media_log); virtual ~BufferedResourceLoader(); // Start the resource loading with the specified URL and range. // // |event_cb| is called to notify the client of network activity in the // following situations: // - Data was received // - Reading was suspended/resumed // - Loading completed // - Loading failed typedef base::Callback StartCB; void Start(const StartCB& start_cb, const base::Closure& event_cb, WebKit::WebFrame* frame); // Stops everything associated with this loader, including active URL loads // and pending callbacks. // // It is safe to delete a BufferedResourceLoader after calling Stop(). void Stop(); // Copies |read_size| bytes from |position| into |buffer|, executing |read_cb| // when the operation has completed. // // The callback will contain the number of bytes read iff the status is kOk, // zero otherwise. // // If necessary will temporarily increase forward capacity of buffer to // accomodate an unusually large read. typedef base::Callback ReadCB; void Read(int64 position, int read_size, uint8* buffer, const ReadCB& read_cb); // Returns the position of the last byte buffered. Returns // |kPositionNotSpecified| if such value is not available. int64 GetBufferedPosition(); // Gets the content length in bytes of the instance after this loader has been // started. If this value is |kPositionNotSpecified|, then content length is // unknown. int64 content_length(); // Gets the original size of the file requested. If this value is // |kPositionNotSpecified|, then the size is unknown. int64 instance_size(); // Returns true if the server supports byte range requests. bool range_supported(); // Returns true if the resource loader is currently downloading data. bool is_downloading_data(); // Returns resulting URL. const GURL& url(); // WebKit::WebURLLoaderClient implementation. virtual void willSendRequest( WebKit::WebURLLoader* loader, WebKit::WebURLRequest& newRequest, const WebKit::WebURLResponse& redirectResponse); virtual void didSendData( WebKit::WebURLLoader* loader, unsigned long long bytesSent, unsigned long long totalBytesToBeSent); virtual void didReceiveResponse( WebKit::WebURLLoader* loader, const WebKit::WebURLResponse& response); virtual void didDownloadData( WebKit::WebURLLoader* loader, int data_length); virtual void didReceiveData( WebKit::WebURLLoader* loader, const char* data, int data_length, int encoded_data_length); virtual void didReceiveCachedMetadata( WebKit::WebURLLoader* loader, const char* data, int dataLength); virtual void didFinishLoading( WebKit::WebURLLoader* loader, double finishTime); virtual void didFail( WebKit::WebURLLoader* loader, const WebKit::WebURLError&); // Returns true if the media resource has a single origin, false otherwise. // Only valid to call after Start() has completed. bool HasSingleOrigin() const; // Sets the defer strategy to the given value. void UpdateDeferStrategy(DeferStrategy strategy); // Sets the playback rate to the given value and updates buffer window // accordingly. void SetPlaybackRate(float playback_rate); // Sets the bitrate to the given value and updates buffer window // accordingly. void SetBitrate(int bitrate); // Parse a Content-Range header into its component pieces and return true if // each of the expected elements was found & parsed correctly. // |*instance_size| may be set to kPositionNotSpecified if the range ends in // "/*". // NOTE: only public for testing! This is an implementation detail of // VerifyPartialResponse (a private method). static bool ParseContentRange( const std::string& content_range_str, int64* first_byte_position, int64* last_byte_position, int64* instance_size); private: friend class BufferedDataSourceTest; friend class BufferedResourceLoaderTest; friend class MockBufferedDataSource; // Updates the |buffer_|'s forward and backward capacities. void UpdateBufferWindow(); // Returns true if we should defer resource loading based on the current // buffering scheme. bool ShouldEnableDefer() const; // Returns true if we should enable resource loading based on the current // buffering scheme. bool ShouldDisableDefer() const; // Updates deferring behavior based on current buffering scheme. void UpdateDeferBehavior(); // Set defer state to |deferred| and cease/continue downloading data // accordingly. void SetDeferred(bool deferred); // Returns true if the current read request can be fulfilled by what is in // the buffer. bool CanFulfillRead() const; // Returns true if the current read request will be fulfilled in the future. bool WillFulfillRead() const; // Method that does the actual read and calls the |read_cb_|, assuming the // request range is in |buffer_|. void ReadInternal(); // If we have made a range request, verify the response from the server. bool VerifyPartialResponse(const WebKit::WebURLResponse& response); // Returns the value for a range request header using parameters // |first_byte_position| and |last_byte_position|. Negative numbers other // than |kPositionNotSpecified| are not allowed for |first_byte_position| and // |last_byte_position|. |first_byte_position| should always be less than or // equal to |last_byte_position| if they are both not |kPositionNotSpecified|. // Empty string is returned on invalid parameters. std::string GenerateHeaders(int64 first_byte_position, int64 last_byte_position); // Done with read. Invokes the read callback and reset parameters for the // read request. void DoneRead(Status status, int bytes_read); // Done with start. Invokes the start callback and reset it. void DoneStart(Status status); // Calls |event_cb_| in terms of a network event. void NotifyNetworkEvent(); bool HasPendingRead() { return !read_cb_.is_null(); } // Helper function that returns true if a range request was specified. bool IsRangeRequest() const; // Log everything interesting to |media_log_|. void Log(); // A sliding window of buffer. media::SeekableBuffer buffer_; // Keeps track of an active WebURLLoader and associated state. scoped_ptr active_loader_; // Tracks if |active_loader_| failed. If so, then all calls to Read() will // fail. bool loader_failed_; // Current buffering algorithm in place for resource loading. DeferStrategy defer_strategy_; // True if Range header is supported. bool range_supported_; // Forward capacity to reset to after an extension. size_t saved_forward_capacity_; GURL url_; int64 first_byte_position_; int64 last_byte_position_; bool single_origin_; // Closure that listens to network events. base::Closure event_cb_; // Members used during request start. StartCB start_cb_; int64 offset_; int64 content_length_; int64 instance_size_; // Members used during a read operation. They should be reset after each // read has completed or failed. ReadCB read_cb_; int64 read_position_; int read_size_; uint8* read_buffer_; // Offsets of the requested first byte and last byte in |buffer_|. They are // written by Read(). int first_offset_; int last_offset_; // Injected WebURLLoader instance for testing purposes. scoped_ptr test_loader_; // Bitrate of the media. Set to 0 if unknown. int bitrate_; // Playback rate of the media. float playback_rate_; scoped_refptr media_log_; DISALLOW_COPY_AND_ASSIGN(BufferedResourceLoader); }; } // namespace webkit_media #endif // WEBKIT_MEDIA_BUFFERED_RESOURCE_LOADER_H_