| OLD | NEW |
| 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "webkit/glue/media/buffered_data_source.h" | 5 #include "webkit/glue/media/buffered_data_source.h" |
| 6 | 6 |
| 7 #include "media/base/filter_host.h" | 7 #include "media/base/filter_host.h" |
| 8 #include "media/base/media_log.h" | 8 #include "media/base/media_log.h" |
| 9 #include "net/base/net_errors.h" | 9 #include "net/base/net_errors.h" |
| 10 #include "webkit/glue/media/web_data_source_factory.h" | 10 #include "webkit/glue/media/web_data_source_factory.h" |
| (...skipping 43 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 54 read_position_(0), | 54 read_position_(0), |
| 55 read_size_(0), | 55 read_size_(0), |
| 56 read_buffer_(NULL), | 56 read_buffer_(NULL), |
| 57 intermediate_read_buffer_(new uint8[kInitialReadBufferSize]), | 57 intermediate_read_buffer_(new uint8[kInitialReadBufferSize]), |
| 58 intermediate_read_buffer_size_(kInitialReadBufferSize), | 58 intermediate_read_buffer_size_(kInitialReadBufferSize), |
| 59 render_loop_(render_loop), | 59 render_loop_(render_loop), |
| 60 stop_signal_received_(false), | 60 stop_signal_received_(false), |
| 61 stopped_on_render_loop_(false), | 61 stopped_on_render_loop_(false), |
| 62 media_is_paused_(true), | 62 media_is_paused_(true), |
| 63 media_has_played_(false), | 63 media_has_played_(false), |
| 64 preload_(media::METADATA), | 64 preload_(media::AUTO), |
| 65 using_range_request_(true), | 65 using_range_request_(true), |
| 66 cache_miss_retries_left_(kNumCacheMissRetries), | 66 cache_miss_retries_left_(kNumCacheMissRetries), |
| 67 bitrate_(0), |
| 68 playback_rate_(0.0), |
| 67 media_log_(media_log) { | 69 media_log_(media_log) { |
| 68 } | 70 } |
| 69 | 71 |
| 70 BufferedDataSource::~BufferedDataSource() {} | 72 BufferedDataSource::~BufferedDataSource() {} |
| 71 | 73 |
| 72 // A factory method to create BufferedResourceLoader using the read parameters. | 74 // A factory method to create BufferedResourceLoader using the read parameters. |
| 73 // This method can be overrided to inject mock BufferedResourceLoader object | 75 // This method can be overrided to inject mock BufferedResourceLoader object |
| 74 // for testing purpose. | 76 // for testing purpose. |
| 75 BufferedResourceLoader* BufferedDataSource::CreateResourceLoader( | 77 BufferedResourceLoader* BufferedDataSource::CreateResourceLoader( |
| 76 int64 first_byte_position, int64 last_byte_position) { | 78 int64 first_byte_position, int64 last_byte_position) { |
| 77 DCHECK(MessageLoop::current() == render_loop_); | 79 DCHECK(MessageLoop::current() == render_loop_); |
| 78 | 80 |
| 79 return new BufferedResourceLoader(url_, | 81 return new BufferedResourceLoader(url_, |
| 80 first_byte_position, | 82 first_byte_position, |
| 81 last_byte_position, | 83 last_byte_position, |
| 84 ChooseDeferStrategy(), |
| 85 bitrate_, |
| 86 playback_rate_, |
| 82 media_log_); | 87 media_log_); |
| 83 } | 88 } |
| 84 | 89 |
| 85 void BufferedDataSource::set_host(media::FilterHost* host) { | 90 void BufferedDataSource::set_host(media::FilterHost* host) { |
| 86 DataSource::set_host(host); | 91 DataSource::set_host(host); |
| 87 | 92 |
| 88 if (loader_.get()) { | 93 if (loader_.get()) { |
| 89 base::AutoLock auto_lock(lock_); | 94 base::AutoLock auto_lock(lock_); |
| 90 UpdateHostState_Locked(); | 95 UpdateHostState_Locked(); |
| 91 } | 96 } |
| (...skipping 55 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 147 NewRunnableMethod(this, &BufferedDataSource::SetPlaybackRateTask, | 152 NewRunnableMethod(this, &BufferedDataSource::SetPlaybackRateTask, |
| 148 playback_rate)); | 153 playback_rate)); |
| 149 } | 154 } |
| 150 | 155 |
| 151 void BufferedDataSource::SetPreload(media::Preload preload) { | 156 void BufferedDataSource::SetPreload(media::Preload preload) { |
| 152 render_loop_->PostTask(FROM_HERE, | 157 render_loop_->PostTask(FROM_HERE, |
| 153 NewRunnableMethod(this, &BufferedDataSource::SetPreloadTask, | 158 NewRunnableMethod(this, &BufferedDataSource::SetPreloadTask, |
| 154 preload)); | 159 preload)); |
| 155 } | 160 } |
| 156 | 161 |
| 162 void BufferedDataSource::SetBitrate(int bitrate) { |
| 163 render_loop_->PostTask(FROM_HERE, |
| 164 NewRunnableMethod(this, &BufferedDataSource::SetBitrateTask, bitrate)); |
| 165 } |
| 166 |
| 157 ///////////////////////////////////////////////////////////////////////////// | 167 ///////////////////////////////////////////////////////////////////////////// |
| 158 // media::DataSource implementation. | 168 // media::DataSource implementation. |
| 159 void BufferedDataSource::Read(int64 position, size_t size, uint8* data, | 169 void BufferedDataSource::Read(int64 position, size_t size, uint8* data, |
| 160 media::DataSource::ReadCallback* read_callback) { | 170 media::DataSource::ReadCallback* read_callback) { |
| 161 VLOG(1) << "Read: " << position << " offset, " << size << " bytes"; | 171 VLOG(1) << "Read: " << position << " offset, " << size << " bytes"; |
| 162 DCHECK(read_callback); | 172 DCHECK(read_callback); |
| 163 | 173 |
| 164 { | 174 { |
| 165 base::AutoLock auto_lock(lock_); | 175 base::AutoLock auto_lock(lock_); |
| 166 DCHECK(!read_callback_.get()); | 176 DCHECK(!read_callback_.get()); |
| (...skipping 130 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 297 return; | 307 return; |
| 298 | 308 |
| 299 { | 309 { |
| 300 // If there's no outstanding read then return early. | 310 // If there's no outstanding read then return early. |
| 301 base::AutoLock auto_lock(lock_); | 311 base::AutoLock auto_lock(lock_); |
| 302 if (!read_callback_.get()) | 312 if (!read_callback_.get()) |
| 303 return; | 313 return; |
| 304 } | 314 } |
| 305 | 315 |
| 306 loader_ = CreateResourceLoader(read_position_, kPositionNotSpecified); | 316 loader_ = CreateResourceLoader(read_position_, kPositionNotSpecified); |
| 307 BufferedResourceLoader::DeferStrategy strategy = ChooseDeferStrategy(); | |
| 308 loader_->UpdateDeferStrategy(strategy); | |
| 309 loader_->Start( | 317 loader_->Start( |
| 310 NewCallback(this, &BufferedDataSource::PartialReadStartCallback), | 318 NewCallback(this, &BufferedDataSource::PartialReadStartCallback), |
| 311 NewCallback(this, &BufferedDataSource::NetworkEventCallback), | 319 NewCallback(this, &BufferedDataSource::NetworkEventCallback), |
| 312 frame_); | 320 frame_); |
| 313 } | 321 } |
| 314 | 322 |
| 315 void BufferedDataSource::SetPlaybackRateTask(float playback_rate) { | 323 void BufferedDataSource::SetPlaybackRateTask(float playback_rate) { |
| 316 DCHECK(MessageLoop::current() == render_loop_); | 324 DCHECK(MessageLoop::current() == render_loop_); |
| 317 DCHECK(loader_.get()); | 325 DCHECK(loader_.get()); |
| 318 | 326 |
| 327 playback_rate_ = playback_rate; |
| 328 loader_->SetPlaybackRate(playback_rate); |
| 329 |
| 319 bool previously_paused = media_is_paused_; | 330 bool previously_paused = media_is_paused_; |
| 320 media_is_paused_ = (playback_rate == 0.0); | 331 media_is_paused_ = (playback_rate == 0.0); |
| 321 | 332 |
| 322 if (!media_has_played_ && previously_paused && !media_is_paused_) | 333 if (!media_has_played_ && previously_paused && !media_is_paused_) |
| 323 media_has_played_ = true; | 334 media_has_played_ = true; |
| 324 | 335 |
| 325 BufferedResourceLoader::DeferStrategy strategy = ChooseDeferStrategy(); | 336 BufferedResourceLoader::DeferStrategy strategy = ChooseDeferStrategy(); |
| 326 loader_->UpdateDeferStrategy(strategy); | 337 loader_->UpdateDeferStrategy(strategy); |
| 327 } | 338 } |
| 328 | 339 |
| 329 void BufferedDataSource::SetPreloadTask(media::Preload preload) { | 340 void BufferedDataSource::SetPreloadTask(media::Preload preload) { |
| 330 DCHECK(MessageLoop::current() == render_loop_); | 341 DCHECK(MessageLoop::current() == render_loop_); |
| 331 preload_ = preload; | 342 preload_ = preload; |
| 332 } | 343 } |
| 333 | 344 |
| 345 void BufferedDataSource::SetBitrateTask(int bitrate) { |
| 346 DCHECK(MessageLoop::current() == render_loop_); |
| 347 DCHECK(loader_.get()); |
| 348 |
| 349 bitrate_ = bitrate; |
| 350 loader_->SetBitrate(bitrate); |
| 351 } |
| 352 |
| 334 BufferedResourceLoader::DeferStrategy | 353 BufferedResourceLoader::DeferStrategy |
| 335 BufferedDataSource::ChooseDeferStrategy() { | 354 BufferedDataSource::ChooseDeferStrategy() { |
| 336 DCHECK(MessageLoop::current() == render_loop_); | 355 DCHECK(MessageLoop::current() == render_loop_); |
| 337 // If the user indicates preload=metadata, then just load exactly | 356 // If the page indicated preload=metadata, then load exactly what is needed |
| 338 // what is needed for starting the pipeline and prerolling frames. | 357 // needed for starting playback. |
| 339 if (preload_ == media::METADATA && !media_has_played_) | 358 if (!media_has_played_ && preload_ == media::METADATA) |
| 340 return BufferedResourceLoader::kReadThenDefer; | 359 return BufferedResourceLoader::kReadThenDefer; |
| 341 | 360 |
| 342 // In general, we want to try to buffer the entire video when the video | 361 // If the playback has started (at which point the preload value is ignored) |
| 343 // is paused. But we don't want to do this if the video hasn't played yet | 362 // and we're paused, then try to load as much as possible. |
| 344 // and preload!=auto. | 363 if (media_has_played_ && media_is_paused_) |
| 345 if (media_is_paused_ && | |
| 346 (preload_ == media::AUTO || media_has_played_)) { | |
| 347 return BufferedResourceLoader::kNeverDefer; | 364 return BufferedResourceLoader::kNeverDefer; |
| 348 } | |
| 349 | 365 |
| 350 // When the video is playing, regardless of preload state, we buffer up | 366 // If media is currently playing or the page indicated preload=auto, |
| 351 // to a hard limit and enable/disable deferring when the buffer is | 367 // use threshold strategy to enable/disable deferring when the buffer |
| 352 // depleted/full. | 368 // is full/depleted. |
| 353 return BufferedResourceLoader::kThresholdDefer; | 369 return BufferedResourceLoader::kThresholdDefer; |
| 354 } | 370 } |
| 355 | 371 |
| 356 // This method is the place where actual read happens, |loader_| must be valid | 372 // This method is the place where actual read happens, |loader_| must be valid |
| 357 // prior to make this method call. | 373 // prior to make this method call. |
| 358 void BufferedDataSource::ReadInternal() { | 374 void BufferedDataSource::ReadInternal() { |
| 359 DCHECK(MessageLoop::current() == render_loop_); | 375 DCHECK(MessageLoop::current() == render_loop_); |
| 360 DCHECK(loader_); | 376 DCHECK(loader_); |
| 361 | 377 |
| 362 // First we prepare the intermediate read buffer for BufferedResourceLoader | 378 // First we prepare the intermediate read buffer for BufferedResourceLoader |
| (...skipping 41 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 404 | 420 |
| 405 ///////////////////////////////////////////////////////////////////////////// | 421 ///////////////////////////////////////////////////////////////////////////// |
| 406 // BufferedResourceLoader callback methods. | 422 // BufferedResourceLoader callback methods. |
| 407 void BufferedDataSource::HttpInitialStartCallback(int error) { | 423 void BufferedDataSource::HttpInitialStartCallback(int error) { |
| 408 DCHECK(MessageLoop::current() == render_loop_); | 424 DCHECK(MessageLoop::current() == render_loop_); |
| 409 DCHECK(loader_.get()); | 425 DCHECK(loader_.get()); |
| 410 | 426 |
| 411 int64 instance_size = loader_->instance_size(); | 427 int64 instance_size = loader_->instance_size(); |
| 412 bool success = error == net::OK; | 428 bool success = error == net::OK; |
| 413 | 429 |
| 414 | |
| 415 bool initialize_cb_is_null = false; | 430 bool initialize_cb_is_null = false; |
| 416 { | 431 { |
| 417 base::AutoLock auto_lock(lock_); | 432 base::AutoLock auto_lock(lock_); |
| 418 initialize_cb_is_null = initialize_cb_.is_null(); | 433 initialize_cb_is_null = initialize_cb_.is_null(); |
| 419 } | 434 } |
| 420 if (initialize_cb_is_null) { | 435 if (initialize_cb_is_null) { |
| 421 loader_->Stop(); | 436 loader_->Stop(); |
| 422 return; | 437 return; |
| 423 } | 438 } |
| 424 | 439 |
| (...skipping 229 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 654 | 669 |
| 655 if (streaming_) { | 670 if (streaming_) { |
| 656 filter_host->SetStreaming(true); | 671 filter_host->SetStreaming(true); |
| 657 } else { | 672 } else { |
| 658 filter_host->SetTotalBytes(total_bytes_); | 673 filter_host->SetTotalBytes(total_bytes_); |
| 659 filter_host->SetBufferedBytes(buffered_bytes_); | 674 filter_host->SetBufferedBytes(buffered_bytes_); |
| 660 } | 675 } |
| 661 } | 676 } |
| 662 | 677 |
| 663 } // namespace webkit_glue | 678 } // namespace webkit_glue |
| OLD | NEW |