1
    2
    3
    4
    5
    6
    7
    8
    9
   10
   11
   12
   13
   14
   15
   16
   17
   18
   19
   20
   21
   22
   23
   24
   25
   26
   27
   28
   29
   30
   31
   32
   33
   34
   35
   36
   37
   38
   39
   40
   41
   42
   43
   44
   45
   46
   47
   48
   49
   50
   51
   52
   53
   54
   55
   56
   57
   58
   59
   60
   61
   62
   63
   64
   65

media / base / simple_watch_timer.cc [blame]

// Copyright 2019 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "media/base/simple_watch_timer.h"

#include "base/location.h"
#include "media/base/timestamp_constants.h"

namespace media {

namespace {

constexpr base::TimeDelta kQueryInterval = base::Milliseconds(750);

}  // namespace

SimpleWatchTimer::SimpleWatchTimer(TickCB tick_cb,
                                   GetCurrentTimeCB get_current_time_cb)
    : tick_cb_(std::move(tick_cb)),
      get_current_time_cb_(std::move(get_current_time_cb)) {
  DCHECK(!tick_cb_.is_null());
  DCHECK(!get_current_time_cb_.is_null());
}

SimpleWatchTimer::~SimpleWatchTimer() {}

void SimpleWatchTimer::Start() {
  if (timer_.IsRunning())
    return;

  last_current_time_ = get_current_time_cb_.Run();
  timer_.Start(FROM_HERE, kQueryInterval, this, &SimpleWatchTimer::Tick);
}

void SimpleWatchTimer::Stop() {
  if (!timer_.IsRunning())
    return;

  timer_.Stop();
  Tick();
}

void SimpleWatchTimer::Tick() {
  base::TimeDelta current_time = get_current_time_cb_.Run();
  base::TimeDelta duration;
  if (last_current_time_ != kNoTimestamp &&
      last_current_time_ != kInfiniteDuration) {
    duration = current_time - last_current_time_;
  }
  last_current_time_ = current_time;

  // Accumulate watch time if the duration is reasonable.
  if (duration.is_positive() && duration < kQueryInterval * 2) {
    unreported_ms_ += duration.InMilliseconds();
  }

  // Tick if the accumulated time is about a second.
  if (unreported_ms_ >= 500) {
    unreported_ms_ -= 1000;
    tick_cb_.Run();
  }
}

}  // namespace media