Home | History | Annotate | Download | only in base
      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.
      4 
      5 #include "media/base/clock.h"
      6 
      7 #include <algorithm>
      8 
      9 #include "base/logging.h"
     10 #include "base/time/tick_clock.h"
     11 #include "media/base/buffers.h"
     12 
     13 namespace media {
     14 
     15 Clock::Clock(base::TickClock* clock)
     16     : clock_(clock),
     17       playing_(false),
     18       underflow_(false),
     19       playback_rate_(1.0f),
     20       max_time_(kNoTimestamp()),
     21       duration_(kNoTimestamp()) {
     22   DCHECK(clock_);
     23 }
     24 
     25 Clock::~Clock() {}
     26 
     27 bool Clock::IsPlaying() const {
     28   return playing_;
     29 }
     30 
     31 base::TimeDelta Clock::Play() {
     32   DCHECK(!playing_);
     33   UpdateReferencePoints();
     34   playing_ = true;
     35   return media_time_;
     36 }
     37 
     38 base::TimeDelta Clock::Pause() {
     39   DCHECK(playing_);
     40   UpdateReferencePoints();
     41   playing_ = false;
     42   return media_time_;
     43 }
     44 
     45 void Clock::SetPlaybackRate(float playback_rate) {
     46   UpdateReferencePoints();
     47   playback_rate_ = playback_rate;
     48 }
     49 
     50 void Clock::SetTime(base::TimeDelta current_time, base::TimeDelta max_time) {
     51   DCHECK(current_time <= max_time);
     52   DCHECK(current_time != kNoTimestamp());
     53 
     54   UpdateReferencePoints(current_time);
     55   max_time_ = ClampToValidTimeRange(max_time);
     56   underflow_ = false;
     57 }
     58 
     59 base::TimeDelta Clock::Elapsed() {
     60   if (duration_ == kNoTimestamp())
     61     return base::TimeDelta();
     62 
     63   // The clock is not advancing, so return the last recorded time.
     64   if (!playing_ || underflow_)
     65     return media_time_;
     66 
     67   base::TimeDelta elapsed = EstimatedElapsedTime();
     68   if (max_time_ != kNoTimestamp() && elapsed > max_time_) {
     69     UpdateReferencePoints(max_time_);
     70     underflow_ = true;
     71     elapsed = max_time_;
     72   }
     73 
     74   return elapsed;
     75 }
     76 
     77 void Clock::SetMaxTime(base::TimeDelta max_time) {
     78   DCHECK(max_time != kNoTimestamp());
     79 
     80   UpdateReferencePoints();
     81   max_time_ = ClampToValidTimeRange(max_time);
     82 
     83   underflow_ = media_time_ > max_time_;
     84   if (underflow_)
     85     media_time_ = max_time_;
     86 }
     87 
     88 void Clock::SetDuration(base::TimeDelta duration) {
     89   DCHECK(duration > base::TimeDelta());
     90   duration_ = duration;
     91 
     92   media_time_ = ClampToValidTimeRange(media_time_);
     93   if (max_time_ != kNoTimestamp())
     94     max_time_ = ClampToValidTimeRange(max_time_);
     95 }
     96 
     97 base::TimeDelta Clock::ElapsedViaProvidedTime(
     98     const base::TimeTicks& time) const {
     99   // TODO(scherkus): floating point badness scaling time by playback rate.
    100   int64 now_us = (time - reference_).InMicroseconds();
    101   now_us = static_cast<int64>(now_us * playback_rate_);
    102   return media_time_ + base::TimeDelta::FromMicroseconds(now_us);
    103 }
    104 
    105 base::TimeDelta Clock::ClampToValidTimeRange(base::TimeDelta time) const {
    106   if (duration_ == kNoTimestamp())
    107     return base::TimeDelta();
    108   return std::max(std::min(time, duration_), base::TimeDelta());
    109 }
    110 
    111 base::TimeDelta Clock::Duration() const {
    112   if (duration_ == kNoTimestamp())
    113     return base::TimeDelta();
    114   return duration_;
    115 }
    116 
    117 void Clock::UpdateReferencePoints() {
    118   UpdateReferencePoints(Elapsed());
    119 }
    120 
    121 void Clock::UpdateReferencePoints(base::TimeDelta current_time) {
    122   media_time_ = ClampToValidTimeRange(current_time);
    123   reference_ = clock_->NowTicks();
    124 }
    125 
    126 base::TimeDelta Clock::EstimatedElapsedTime() {
    127   return ClampToValidTimeRange(ElapsedViaProvidedTime(clock_->NowTicks()));
    128 }
    129 
    130 }  // namespace media
    131