Skip to content

Commit f85d4f6

Browse files
cramertjCQ Bot Account
authored and
CQ Bot Account
committed
pw_chrono: Add VirtualClock
This CL introduces an extension to VirtualSystemClock which is clock-generic and changes the existing VirtualSystemClock into a specialization of this new, more generic interface. Change-Id: Ibb03dd9257eeed957bcfe2b93b1a0038870083bd Reviewed-on: https://pigweed-review.googlesource.com/c/pigweed/pigweed/+/233031 Reviewed-by: Aaron Green <aarongreen@google.com> Lint: Lint 🤖 <android-build-ayeaye@system.gserviceaccount.com> Commit-Queue: Taylor Cramer <cramertj@google.com>
1 parent dc92fa9 commit f85d4f6

11 files changed

+109
-14
lines changed

docs/BUILD.gn

+2
Original file line numberDiff line numberDiff line change
@@ -224,8 +224,10 @@ _doxygen_input_files = [ # keep-sorted: start
224224
"$dir_pw_channel/public/pw_channel/stream_channel.h",
225225
"$dir_pw_chre/public/pw_chre/chre.h",
226226
"$dir_pw_chre/public/pw_chre/host_link.h",
227+
"$dir_pw_chrono/public/pw_chrono/simulated_system_clock.h",
227228
"$dir_pw_chrono/public/pw_chrono/system_clock.h",
228229
"$dir_pw_chrono/public/pw_chrono/system_timer.h",
230+
"$dir_pw_chrono/public/pw_chrono/virtual_clock.h",
229231
"$dir_pw_clock_tree/public/pw_clock_tree/clock_tree.h",
230232
"$dir_pw_clock_tree_mcuxpresso/public/pw_clock_tree_mcuxpresso/clock_tree.h",
231233
"$dir_pw_containers/public/pw_containers/filtered_view.h",

pw_async2/BUILD.bazel

+1
Original file line numberDiff line numberDiff line change
@@ -271,6 +271,7 @@ cc_library(
271271
includes = ["public"],
272272
deps = [
273273
":dispatcher",
274+
"//pw_chrono:virtual_clock",
274275
"//pw_containers:intrusive_list",
275276
"//pw_sync:interrupt_spin_lock",
276277
"//pw_toolchain:no_destructor",

pw_async2/public/pw_async2/simulated_time_provider.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,7 @@ class SimulatedTimeProvider final : public TimeProvider<Clock> {
6161
/// one of `AdvanceTime`, `SetTime`, or `RunExpiredTimers` has been called.
6262
void RunExpiredTimers() { RunExpired(now()); }
6363

64-
typename Clock::time_point now() const final {
64+
typename Clock::time_point now() final {
6565
std::lock_guard lock(lock_);
6666
return now_;
6767
}

pw_async2/public/pw_async2/time_provider.h

+7-8
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
#include <mutex>
2020

2121
#include "pw_async2/dispatcher.h"
22+
#include "pw_chrono/virtual_clock.h"
2223
#include "pw_containers/intrusive_list.h"
2324
#include "pw_sync/interrupt_spin_lock.h"
2425
#include "pw_sync/lock_annotations.h"
@@ -44,9 +45,8 @@ class TimeFuture;
4445

4546
/// A factory for time and timers.
4647
///
47-
/// This provides similar functionality as the `VirtualSystemClock` interface
48-
/// except it can also create timers, and its template parameter means that it
49-
/// can be used with non-SystemClock clocks.
48+
/// This extends the `VirtualClock` interface with the ability to create async
49+
/// timers.
5050
///
5151
/// `TimeProvider` is designed to be dependency-injection friendly so that
5252
/// code that uses time and timers is not bound to real wall-clock time.
@@ -57,15 +57,14 @@ class TimeFuture;
5757
/// Note that `Timer` objects must not outlive the `TimeProvider` from which
5858
/// they were created.
5959
template <typename Clock>
60-
class TimeProvider {
60+
class TimeProvider : public chrono::VirtualClock<Clock> {
6161
public:
62-
/// Gets the current `time_point` for the provided clock.
63-
virtual typename Clock::time_point now() const = 0;
64-
65-
virtual ~TimeProvider() {
62+
~TimeProvider() override {
6663
internal::AssertTimeFutureObjectsAllGone(futures_.empty());
6764
}
6865

66+
typename Clock::time_point now() override = 0;
67+
6968
/// Queues the `callback` to be invoked after `delay`.
7069
///
7170
/// This method is thread-safe and can be invoked from `callback` but may not

pw_async2/system_time_provider.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ class SystemTimeProvider final : public TimeProvider<SystemClock> {
2929
[this](SystemClock::time_point expired) { RunExpired(expired); }) {}
3030

3131
private:
32-
SystemClock::time_point now() const final { return SystemClock::now(); }
32+
SystemClock::time_point now() final { return SystemClock::now(); }
3333

3434
void DoInvokeAt(SystemClock::time_point time_point) final {
3535
timer_.InvokeAt(time_point);

pw_chrono/BUILD.bazel

+9
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,14 @@ cc_library(
3333
includes = ["public"],
3434
)
3535

36+
cc_library(
37+
name = "virtual_clock",
38+
hdrs = [
39+
"public/pw_chrono/virtual_clock.h",
40+
],
41+
includes = ["public"],
42+
)
43+
3644
pw_facade(
3745
name = "system_clock",
3846
srcs = [
@@ -46,6 +54,7 @@ pw_facade(
4654
includes = ["public"],
4755
deps = [
4856
":epoch",
57+
":virtual_clock",
4958
"//pw_preprocessor",
5059
],
5160
)

pw_chrono/BUILD.gn

+6
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,11 @@ pw_source_set("epoch") {
3030
public_configs = [ ":public_include_path" ]
3131
}
3232

33+
pw_source_set("virtual_clock") {
34+
public = [ "public/pw_chrono/virtual_clock.h" ]
35+
public_configs = [ ":public_include_path" ]
36+
}
37+
3338
pw_facade("system_clock") {
3439
backend = pw_chrono_SYSTEM_CLOCK_BACKEND
3540
public_configs = [ ":public_include_path" ]
@@ -39,6 +44,7 @@ pw_facade("system_clock") {
3944
]
4045
public_deps = [
4146
":epoch",
47+
":virtual_clock",
4248
"$dir_pw_preprocessor",
4349
]
4450
sources = [ "system_clock.cc" ]

pw_chrono/CMakeLists.txt

+8
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,13 @@ pw_add_library(pw_chrono.epoch INTERFACE
2424
public
2525
)
2626

27+
pw_add_library(pw_chrono.virtual_clock INTERFACE
28+
HEADERS
29+
public/pw_chrono/virtual_clock.h
30+
PUBLIC_INCLUDES
31+
public
32+
)
33+
2734
pw_add_facade(pw_chrono.system_clock STATIC
2835
BACKEND
2936
pw_chrono.system_clock_BACKEND
@@ -34,6 +41,7 @@ pw_add_facade(pw_chrono.system_clock STATIC
3441
public
3542
PUBLIC_DEPS
3643
pw_chrono.epoch
44+
pw_chrono.virtual_clock
3745
pw_preprocessor
3846
SOURCES
3947
system_clock.cc

pw_chrono/docs.rst

+21
Original file line numberDiff line numberDiff line change
@@ -446,6 +446,27 @@ Example in C++
446446
}
447447
}
448448
449+
VirtualClock
450+
============
451+
Pigweed also includes a virtual base class for timers,
452+
:cpp:class:`pw::chrono::VirtualClock`. This class allows for writing
453+
timing-sensitive code that can be tested using simulated clocks such as
454+
:cpp:class:`pw::chrono::SimulatedSystemClock`.
455+
456+
Using simulated clocks in tests allow tests to avoid sleeping or timeouts,
457+
resulting in faster and more reliable tests.
458+
459+
See also :cpp:class:`pw::async2::TimeProvider` for creating testable
460+
time-sensitive code using asynchronous timers.
461+
462+
.. doxygenclass:: pw::chrono::VirtualClock
463+
:members:
464+
465+
.. doxygenclass:: pw::chrono::SimulatedSystemClock
466+
:members:
467+
468+
469+
449470
Protobuf
450471
========
451472
Sometimes it's desirable to communicate high resolution time points and

pw_chrono/public/pw_chrono/system_clock.h

+16-4
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,8 @@
3333
#include <chrono>
3434
#include <ratio>
3535

36+
#include "pw_chrono/virtual_clock.h"
37+
3638
namespace pw::chrono {
3739
namespace backend {
3840

@@ -133,14 +135,21 @@ struct SystemClock {
133135
}
134136
};
135137

138+
// NOTE: VirtualClock here is specialized on SystemClock in order to provide
139+
// the `RealClock` function.
140+
//
141+
// SystemClock is defined in this file, so there's no risk of an ODR violation
142+
// as other libraries are unable to spell VirtualClock<SystemClock> unless
143+
// they have first included this file.
144+
136145
/// An abstract interface representing a SystemClock.
137146
///
138147
/// This interface allows decoupling code that uses time from the code that
139148
/// creates a point in time. You can use this to your advantage by injecting
140149
/// Clocks into interfaces rather than having implementations call
141150
/// `SystemClock::now()` directly. However, this comes at a cost of a vtable per
142151
/// implementation and more importantly passing and maintaining references to
143-
/// the VirtualSystemCLock for all of the users.
152+
/// the VirtualClock for all of the users.
144153
///
145154
/// The `VirtualSystemClock::RealClock()` function returns a reference to the
146155
/// real global SystemClock.
@@ -162,17 +171,20 @@ struct SystemClock {
162171
/// @endcode
163172
///
164173
/// This interface is thread and IRQ safe.
165-
class VirtualSystemClock {
174+
template <>
175+
class VirtualClock<SystemClock> {
166176
public:
167177
/// Returns a reference to the real system clock to aid instantiation.
168-
static VirtualSystemClock& RealClock();
178+
static VirtualClock<SystemClock>& RealClock();
169179

170-
virtual ~VirtualSystemClock() = default;
180+
virtual ~VirtualClock() = default;
171181

172182
/// Returns the current time.
173183
virtual SystemClock::time_point now() = 0;
174184
};
175185

186+
using VirtualSystemClock = VirtualClock<SystemClock>;
187+
176188
} // namespace pw::chrono
177189

178190
// The backend can opt to include an inlined implementation of the following:
+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
// Copyright 2024 The Pigweed Authors
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
4+
// use this file except in compliance with the License. You may obtain a copy of
5+
// the License at
6+
//
7+
// https://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11+
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12+
// License for the specific language governing permissions and limitations under
13+
// the License.
14+
#pragma once
15+
16+
namespace pw::chrono {
17+
18+
/// An abstract interface representing a Clock.
19+
///
20+
/// This interface allows decoupling code that uses time from the code that
21+
/// creates a point in time. You can use this to your advantage by injecting
22+
/// Clocks into interfaces rather than having implementations call
23+
/// `SystemClock::now()` directly. However, this comes at a cost of a vtable per
24+
/// implementation and more importantly passing and maintaining references to
25+
/// the VirtualClock for all of the users.
26+
///
27+
/// This interface is thread and IRQ safe.
28+
template <typename Clock>
29+
class VirtualClock {
30+
public:
31+
virtual ~VirtualClock() = default;
32+
33+
/// Returns the current time.
34+
virtual typename Clock::time_point now() = 0;
35+
};
36+
37+
} // namespace pw::chrono

0 commit comments

Comments
 (0)