TL;DR a safe alternative to std::chrono::duration_cast
Converting between std::chrono::duration types can invoke
- overflow
- underflow
- undefined behaviour
For instance, the following may be an unpleaseant surprise:
using Sec = std::chrono::duration<int>;
using mSec = std::chrono::duration<int, std::milli>;
const Sec from{ INT_MAX / 1000 + 1 }; // change +1 to -1, and it works
const mSec to = from;
assert(to.count() / 1000 == from.count() && "oops, wrong answer!");
The above assert will trigger. There is luckily no ub, since std::chrono::duration_cast internally uses a very big integer type for the intermediate calculations. But if you have a larger type, you can expose undefined behaviour inside chrono. See all the examples at problematic examples
This library provides the function template
namespace safe_duration_cast {
template<typename To, typename From>
constexpr To
safe_duration_cast(From from, int& ec);
}
which works like std::chrono::duration_cast, but with error checking. (limitations apply, see below).
The result will either be correct, or the error code ec will be set to a nonzero value.
In case you like error reporting through exceptions, you can use the throwing variant
namespace safe_duration_cast {
template<typename To, typename From>
constexpr To
safe_duration_cast(From from);
}
This form either reports the correct result, or throws an exception. It should be possible to use the library with exceptions disabled (this has yet not been tested), so this function signature is only enabled if the compiler has exceptions enabled (-fno-exceptions on gcc and clang).
An integral type is one which std::is_integral says is integral. For converting between integral durations you wont get under/overflow or the wrong result without ec being set. You won't get exposed to the internal overflow which may happen in std::chrono::duration_cast, or the result not being representable in the output type.
An floating point type is one which std::is_floating_point says is: float, double, long double.
Converting durations with floating point representation won't give you under/overflow. This can happen for instance if you have a huge number and convert it to a smaller unit, say seconds to microseconds. Instead, the error code ec will be set to a nonzero value.
input | output | error code ec |
---|---|---|
NaN | NaN | 0 |
+Inf | +Inf | 0 |
-Inf | -Inf | 0 |
possible to convert | correct result | 0 |
not possible to convert | - | nonzero |
One can consider what to do with subnormals. Perhaps it had been wise to also signal errors in case subnormal results appear.
This is not yet supported.
If you have a duration with a representation which is not recognized as std::is_arithmetic, you will get a compile time error.
In the future, it might be worth considering conversion of supporting types that support numeric_limits, like boost multiprecision.
There is a limited benchmark comparing std::chrono::duration_cast with safe_duration_cast. See the files in benchmark which converts uint64 timestamps from period 1 to 5/3. The speed difference is smaller than the random fluctuations in measurement, on an optimized build (64 bit gcc 8.3).
There are unit tests and fuzz testing. Actually, fuzz testing was used to smoke out all the corner cases. So far it has only been tested on Ubuntu 18.04 64bit, using gcc and clang.
There are also exhaustive tests testing each possible 32 bit value for the supported types to make sure the result is either signaled as an error, or consistent with std::chrono::duration_cast.
Arvid Nordberg suggested the use of cfenv to search for floating point exceptions, thanks!
The idea is to have a permissive license. Therefore, it is dual licensed so everyone gets what they want. If this is not good enough, let me know.
- Boost 1.0
- GNU GPL v2 (or later, at your option)