libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
types.hxx
Go to the documentation of this file.
1#pragma once
2
3// Linux
4#include <time.h>
5
6// C++
7#include <chrono>
8
9// cosmos
10#include <cosmos/types.hxx>
11
18namespace cosmos {
19
21
25enum class ClockTicks : clock_t {
26};
27
29enum class ClockType : clockid_t {
31 REALTIME = CLOCK_REALTIME,
33 REALTIME_COARSE = CLOCK_REALTIME_COARSE,
35 ATOMIC_REALTIME = CLOCK_TAI,
37
40 MONOTONIC = CLOCK_MONOTONIC,
42 MONOTONIC_RAW = CLOCK_MONOTONIC_RAW,
44 MONOTONIC_COARSE = CLOCK_MONOTONIC_COARSE,
46 BOOTTIME = CLOCK_BOOTTIME,
48 PROCESS_CPUTIME = CLOCK_PROCESS_CPUTIME_ID,
50 THREAD_CPUTIME = CLOCK_THREAD_CPUTIME_ID,
51 INVALID = clockid_t{-1}
52};
53
55template <ClockType CLOCK>
56class TimeSpec :
57 public timespec {
58public:
59 explicit TimeSpec(time_t seconds, long nano_seconds = 0) {
60 this->tv_sec = seconds;
61 this->tv_nsec = nano_seconds;
62 }
63
64 explicit TimeSpec(const std::chrono::milliseconds ms) {
65 set(ms);
66 }
67
68 explicit TimeSpec(const std::chrono::nanoseconds ns) {
69 set(ns);
70 }
71
72 TimeSpec() { reset(); }
73
75 explicit TimeSpec(const no_init_t) {
76
77 }
78
79 bool isZero() const { return this->tv_sec == 0 && this->tv_nsec == 0; }
80 void reset() { this->tv_sec = 0; this->tv_nsec = 0; }
81
82 time_t getSeconds() const { return this->tv_sec; }
83 long getNanoseconds() const { return this->tv_nsec; }
84
85 void setSeconds(const time_t seconds) { this->tv_sec = seconds; }
86 void setNanoseconds(const long nano_seconds) { this->tv_nsec = nano_seconds; }
87
88 void addSeconds(const time_t seconds) {
89 this->tv_sec += seconds;
90 }
91
92 void addNanoseconds(const long nano_seconds) {
93 this->tv_nsec += nano_seconds;
94 }
95
96 TimeSpec& setAsMilliseconds(const size_t milliseconds) {
97 auto left_ms = milliseconds % 1000;
98 this->tv_sec = (milliseconds - left_ms) / 1000;
99 this->tv_nsec = left_ms * 1000 * 1000;
100 return *this;
101 }
102
103 TimeSpec& set(const std::chrono::milliseconds ms) {
104 this->tv_sec = ms.count() / 1000;
105 this->tv_nsec = (ms.count() % 1000) * 1000 * 1000;
106 return *this;
107 }
108
109 TimeSpec& set(const std::chrono::nanoseconds ns) {
110 this->tv_sec = ns.count() / NANOSECOND_BASE;
111 this->tv_nsec = (ns.count() % NANOSECOND_BASE);
112 return *this;
113 }
114
116 size_t toMilliseconds() const {
117 size_t ret = this->tv_sec * 1000;
118 ret += (this->tv_nsec / 1000 / 1000);
119 return ret;
120 }
121
122 explicit operator std::chrono::milliseconds() const {
123 return std::chrono::milliseconds{toMilliseconds()};
124 }
125
126 bool operator<(const TimeSpec &other) const {
127 if (this->tv_sec < other.tv_sec)
128 return true;
129 else if (this->tv_sec != other.tv_sec)
130 return false;
131
132 // so seconds are equal
133 if (this->tv_nsec < other.tv_nsec)
134 return true;
135
136 return false;
137 }
138
139 bool operator>=(const TimeSpec &other) const {
140 return !operator<(other);
141 }
142
143 bool operator==(const TimeSpec &other) const {
144 return this->tv_sec == other.tv_sec &&
145 this->tv_nsec == other.tv_nsec;
146 }
147
148 bool operator!=(const TimeSpec &other) const { return !(*this == other); }
149
150 bool operator<=(const TimeSpec &other) const {
151 return *this < other || *this == other;
152 }
153
154 TimeSpec operator-(const TimeSpec &other) const {
155 TimeSpec ret;
156
157 ret.tv_sec = this->tv_sec - other.tv_sec;
158 ret.tv_nsec = this->tv_nsec - other.tv_nsec;
159
160 if (ret.tv_nsec < 0) {
161 --ret.tv_sec;
162 ret.tv_nsec += NANOSECOND_BASE;
163 }
164
165 return ret;
166 }
167
168 TimeSpec operator+(const TimeSpec &other) const {
169 TimeSpec ret;
170
171 ret.tv_sec = this->tv_sec + other.tv_sec;
172 ret.tv_nsec = this->tv_nsec + other.tv_nsec;
173
174 if (ret.tv_nsec >= NANOSECOND_BASE) {
175 ++ret.tv_sec;
176 ret.tv_nsec -= NANOSECOND_BASE;
177 }
178
179 return ret;
180 }
181
182protected: // functions
183
184 static constexpr long NANOSECOND_BASE{1000 * 1000 * 1000};
185};
186
187using AtomicRealTime = TimeSpec<ClockType::ATOMIC_REALTIME>;
188using BootTime = TimeSpec<ClockType::BOOTTIME>;
189using CoarseMonotonicTime = TimeSpec<ClockType::MONOTONIC_COARSE>;
190using MonotonicTime = TimeSpec<ClockType::MONOTONIC>;
191using RawMonotonicTime = TimeSpec<ClockType::MONOTONIC_RAW>;
192using ProcessCpuTime = TimeSpec<ClockType::PROCESS_CPUTIME>;
193using CoarseRealTime = TimeSpec<ClockType::REALTIME_COARSE>;
194using RealTime = TimeSpec<ClockType::REALTIME>;
195using ThreadCpuTime = TimeSpec<ClockType::THREAD_CPUTIME>;
198
199} // end ns
A C++ wrapper around the POSIX struct timespec coupled to a specific CLOCK type.
Definition types.hxx:57
TimeSpec(const no_init_t)
Deliberately don't initialize the members for performance reasons.
Definition types.hxx:75
size_t toMilliseconds() const
Converts the time representation into a single milliseconds value.
Definition types.hxx:116
Type used to invoke constructors that explicitly don't zero-initialize low level data structures.
Definition types.hxx:37
ClockType
Available clock types for time operations.
Definition types.hxx:29
@ MONOTONIC
System-wide clock representing monotonic time since some unspecified point in the past.
@ REALTIME_COARSE
A faster but less precise version of REALTIME, not settable.
@ REALTIME
System-wide wall clock time, settable.
@ MONOTONIC_COARSE
A faster but less precise version of MONOTONIC, does not count suspend time.
@ MONOTONIC_RAW
Like MONOTONIC but not affected by NTP adjustments.
@ PROCESS_CPUTIME
Counts the CPU time consumed by the calling process.
@ BOOTTIME
Like MONOTONIC but also counts suspend time.
@ ATOMIC_REALTIME
System-wide wall clock time based on international atomic time (TAI) - it is ignoring leap seconds.
@ THREAD_CPUTIME
Counts the CPU time consumed by the calling thread.
ClockTicks
Type used to express time in clock ticks unit in some APIs.
Definition types.hxx:25