OLD | NEW |
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 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 "media/filters/video_renderer_impl.h" | 5 #include "media/filters/video_renderer_impl.h" |
6 | 6 |
7 #include "base/bind.h" | 7 #include "base/bind.h" |
8 #include "base/callback.h" | 8 #include "base/callback.h" |
9 #include "base/callback_helpers.h" | 9 #include "base/callback_helpers.h" |
10 #include "base/debug/trace_event.h" | 10 #include "base/debug/trace_event.h" |
(...skipping 184 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
195 if (state_ != kPlaying || buffering_state_ != BUFFERING_HAVE_ENOUGH) { | 195 if (state_ != kPlaying || buffering_state_ != BUFFERING_HAVE_ENOUGH) { |
196 UpdateStatsAndWait_Locked(kIdleTimeDelta); | 196 UpdateStatsAndWait_Locked(kIdleTimeDelta); |
197 continue; | 197 continue; |
198 } | 198 } |
199 | 199 |
200 // Remain idle until we have the next frame ready for rendering. | 200 // Remain idle until we have the next frame ready for rendering. |
201 if (ready_frames_.empty()) { | 201 if (ready_frames_.empty()) { |
202 if (received_end_of_stream_) { | 202 if (received_end_of_stream_) { |
203 if (!rendered_end_of_stream_) { | 203 if (!rendered_end_of_stream_) { |
204 rendered_end_of_stream_ = true; | 204 rendered_end_of_stream_ = true; |
205 ended_cb_.Run(); | 205 task_runner_->PostTask(FROM_HERE, ended_cb_); |
206 } | 206 } |
207 } else { | 207 } else { |
208 buffering_state_ = BUFFERING_HAVE_NOTHING; | 208 buffering_state_ = BUFFERING_HAVE_NOTHING; |
209 task_runner_->PostTask( | 209 task_runner_->PostTask( |
210 FROM_HERE, base::Bind(buffering_state_cb_, BUFFERING_HAVE_NOTHING)); | 210 FROM_HERE, base::Bind(buffering_state_cb_, BUFFERING_HAVE_NOTHING)); |
211 } | 211 } |
212 | 212 |
213 UpdateStatsAndWait_Locked(kIdleTimeDelta); | 213 UpdateStatsAndWait_Locked(kIdleTimeDelta); |
214 continue; | 214 continue; |
215 } | 215 } |
(...skipping 60 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
276 frames_decoded_++; | 276 frames_decoded_++; |
277 frames_dropped_++; | 277 frames_dropped_++; |
278 | 278 |
279 task_runner_->PostTask( | 279 task_runner_->PostTask( |
280 FROM_HERE, | 280 FROM_HERE, |
281 base::Bind(&VideoRendererImpl::AttemptRead, weak_factory_.GetWeakPtr())); | 281 base::Bind(&VideoRendererImpl::AttemptRead, weak_factory_.GetWeakPtr())); |
282 } | 282 } |
283 | 283 |
284 void VideoRendererImpl::FrameReady(VideoFrameStream::Status status, | 284 void VideoRendererImpl::FrameReady(VideoFrameStream::Status status, |
285 const scoped_refptr<VideoFrame>& frame) { | 285 const scoped_refptr<VideoFrame>& frame) { |
| 286 DCHECK(task_runner_->BelongsToCurrentThread()); |
286 base::AutoLock auto_lock(lock_); | 287 base::AutoLock auto_lock(lock_); |
287 DCHECK_NE(state_, kUninitialized); | 288 DCHECK_NE(state_, kUninitialized); |
288 DCHECK_NE(state_, kFlushed); | 289 DCHECK_NE(state_, kFlushed); |
289 | 290 |
290 CHECK(pending_read_); | 291 CHECK(pending_read_); |
291 pending_read_ = false; | 292 pending_read_ = false; |
292 | 293 |
293 if (status == VideoFrameStream::DECODE_ERROR || | 294 if (status == VideoFrameStream::DECODE_ERROR || |
294 status == VideoFrameStream::DECRYPT_ERROR) { | 295 status == VideoFrameStream::DECRYPT_ERROR) { |
295 DCHECK(!frame.get()); | 296 DCHECK(!frame.get()); |
(...skipping 41 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
337 | 338 |
338 bool VideoRendererImpl::HaveEnoughData_Locked() { | 339 bool VideoRendererImpl::HaveEnoughData_Locked() { |
339 DCHECK_EQ(state_, kPlaying); | 340 DCHECK_EQ(state_, kPlaying); |
340 return received_end_of_stream_ || | 341 return received_end_of_stream_ || |
341 !video_frame_stream_->CanReadWithoutStalling() || | 342 !video_frame_stream_->CanReadWithoutStalling() || |
342 ready_frames_.size() >= static_cast<size_t>(limits::kMaxVideoFrames) || | 343 ready_frames_.size() >= static_cast<size_t>(limits::kMaxVideoFrames) || |
343 (low_delay_ && ready_frames_.size() > 0); | 344 (low_delay_ && ready_frames_.size() > 0); |
344 } | 345 } |
345 | 346 |
346 void VideoRendererImpl::TransitionToHaveEnough_Locked() { | 347 void VideoRendererImpl::TransitionToHaveEnough_Locked() { |
| 348 DCHECK(task_runner_->BelongsToCurrentThread()); |
347 DCHECK_EQ(buffering_state_, BUFFERING_HAVE_NOTHING); | 349 DCHECK_EQ(buffering_state_, BUFFERING_HAVE_NOTHING); |
348 | 350 |
349 if (received_end_of_stream_) | 351 if (received_end_of_stream_) |
350 max_time_cb_.Run(get_duration_cb_.Run()); | 352 max_time_cb_.Run(get_duration_cb_.Run()); |
351 | 353 |
352 if (!ready_frames_.empty()) { | 354 if (!ready_frames_.empty()) { |
353 // Max time isn't reported while we're in a have nothing state as we could | 355 // Max time isn't reported while we're in a have nothing state as we could |
354 // be discarding frames to find |start_timestamp_|. | 356 // be discarding frames to find |start_timestamp_|. |
355 if (!received_end_of_stream_) { | 357 if (!received_end_of_stream_) { |
356 base::TimeDelta max_timestamp = ready_frames_[0]->timestamp(); | 358 base::TimeDelta max_timestamp = ready_frames_[0]->timestamp(); |
357 for (size_t i = 1; i < ready_frames_.size(); ++i) { | 359 for (size_t i = 1; i < ready_frames_.size(); ++i) { |
358 if (ready_frames_[i]->timestamp() > max_timestamp) | 360 if (ready_frames_[i]->timestamp() > max_timestamp) |
359 max_timestamp = ready_frames_[i]->timestamp(); | 361 max_timestamp = ready_frames_[i]->timestamp(); |
360 } | 362 } |
361 max_time_cb_.Run(max_timestamp); | 363 max_time_cb_.Run(max_timestamp); |
362 } | 364 } |
363 | 365 |
364 // Because the clock might remain paused in for an undetermined amount | 366 // Because the clock might remain paused in for an undetermined amount |
365 // of time (e.g., seeking while paused), paint the first frame. | 367 // of time (e.g., seeking while paused), paint the first frame. |
366 PaintNextReadyFrame_Locked(); | 368 PaintNextReadyFrame_Locked(); |
367 } | 369 } |
368 | 370 |
369 buffering_state_ = BUFFERING_HAVE_ENOUGH; | 371 buffering_state_ = BUFFERING_HAVE_ENOUGH; |
370 buffering_state_cb_.Run(BUFFERING_HAVE_ENOUGH); | 372 buffering_state_cb_.Run(BUFFERING_HAVE_ENOUGH); |
371 } | 373 } |
372 | 374 |
373 void VideoRendererImpl::AddReadyFrame_Locked( | 375 void VideoRendererImpl::AddReadyFrame_Locked( |
374 const scoped_refptr<VideoFrame>& frame) { | 376 const scoped_refptr<VideoFrame>& frame) { |
| 377 DCHECK(task_runner_->BelongsToCurrentThread()); |
375 lock_.AssertAcquired(); | 378 lock_.AssertAcquired(); |
376 DCHECK(!frame->end_of_stream()); | 379 DCHECK(!frame->end_of_stream()); |
377 | 380 |
378 // Adjust the incoming frame if its rendering stop time is past the duration | 381 // Adjust the incoming frame if its rendering stop time is past the duration |
379 // of the video itself. This is typically the last frame of the video and | 382 // of the video itself. This is typically the last frame of the video and |
380 // occurs if the container specifies a duration that isn't a multiple of the | 383 // occurs if the container specifies a duration that isn't a multiple of the |
381 // frame rate. Another way for this to happen is for the container to state | 384 // frame rate. Another way for this to happen is for the container to state |
382 // a smaller duration than the largest packet timestamp. | 385 // a smaller duration than the largest packet timestamp. |
383 base::TimeDelta duration = get_duration_cb_.Run(); | 386 base::TimeDelta duration = get_duration_cb_.Run(); |
384 if (frame->timestamp() > duration) { | 387 if (frame->timestamp() > duration) { |
(...skipping 62 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
447 void VideoRendererImpl::UpdateStatsAndWait_Locked( | 450 void VideoRendererImpl::UpdateStatsAndWait_Locked( |
448 base::TimeDelta wait_duration) { | 451 base::TimeDelta wait_duration) { |
449 lock_.AssertAcquired(); | 452 lock_.AssertAcquired(); |
450 DCHECK_GE(frames_decoded_, 0); | 453 DCHECK_GE(frames_decoded_, 0); |
451 DCHECK_LE(frames_dropped_, frames_decoded_); | 454 DCHECK_LE(frames_dropped_, frames_decoded_); |
452 | 455 |
453 if (frames_decoded_) { | 456 if (frames_decoded_) { |
454 PipelineStatistics statistics; | 457 PipelineStatistics statistics; |
455 statistics.video_frames_decoded = frames_decoded_; | 458 statistics.video_frames_decoded = frames_decoded_; |
456 statistics.video_frames_dropped = frames_dropped_; | 459 statistics.video_frames_dropped = frames_dropped_; |
457 statistics_cb_.Run(statistics); | 460 task_runner_->PostTask(FROM_HERE, base::Bind(statistics_cb_, statistics)); |
458 | 461 |
459 frames_decoded_ = 0; | 462 frames_decoded_ = 0; |
460 frames_dropped_ = 0; | 463 frames_dropped_ = 0; |
461 } | 464 } |
462 | 465 |
463 frame_available_.TimedWait(wait_duration); | 466 frame_available_.TimedWait(wait_duration); |
464 } | 467 } |
465 | 468 |
466 } // namespace media | 469 } // namespace media |
OLD | NEW |