File: persistent_repeating_timer.cc

package info (click to toggle)
chromium 138.0.7204.183-1~deb12u1
  • links: PTS, VCS
  • area: main
  • in suites: bookworm-proposed-updates
  • size: 6,080,960 kB
  • sloc: cpp: 34,937,079; ansic: 7,176,967; javascript: 4,110,704; python: 1,419,954; asm: 946,768; xml: 739,971; pascal: 187,324; sh: 89,623; perl: 88,663; objc: 79,944; sql: 50,304; cs: 41,786; fortran: 24,137; makefile: 21,811; php: 13,980; tcl: 13,166; yacc: 8,925; ruby: 7,485; awk: 3,720; lisp: 3,096; lex: 1,327; ada: 727; jsp: 228; sed: 36
file content (60 lines) | stat: -rw-r--r-- 1,767 bytes parent folder | download | duplicates (5)
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
// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "content/browser/btm/persistent_repeating_timer.h"

#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/memory/raw_ptr.h"
#include "base/time/time.h"

namespace content {

PersistentRepeatingTimer::Storage::~Storage() = default;

PersistentRepeatingTimer::PersistentRepeatingTimer(
    std::unique_ptr<Storage> time_storage,
    base::TimeDelta delay,
    base::RepeatingClosure task)
    : storage_(std::move(time_storage)), delay_(delay), user_task_(task) {}

PersistentRepeatingTimer::~PersistentRepeatingTimer() = default;

void PersistentRepeatingTimer::Start() {
  if (timer_.IsRunning()) {
    return;  // Already started.
  }

  storage_->GetLastFired(
      base::BindOnce(&PersistentRepeatingTimer::StartWithLastFired,
                     weak_factory_.GetWeakPtr()));
}

void PersistentRepeatingTimer::StartWithLastFired(
    std::optional<base::Time> last_fired) {
  if (timer_.IsRunning()) {
    return;  // Already started.
  }

  const base::TimeDelta time_since_update =
      base::Time::Now() - last_fired.value_or(base::Time());
  if (time_since_update >= delay_) {
    OnTimerFired();
  } else {
    timer_.Start(FROM_HERE, delay_ - time_since_update,
                 base::BindRepeating(&PersistentRepeatingTimer::OnTimerFired,
                                     base::Unretained(this)));
  }
  DCHECK(timer_.IsRunning());
}

void PersistentRepeatingTimer::OnTimerFired() {
  DCHECK(!timer_.IsRunning());
  const base::Time now = base::Time::Now();
  storage_->SetLastFired(now);
  user_task_.Run();
  StartWithLastFired(now);
}

}  // namespace content