2020-10-07 12:02:08 +02:00
|
|
|
// The MIT License (MIT)
|
|
|
|
//
|
|
|
|
// Copyright (c) 2018 Mateusz Pusz
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
|
|
// copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
// SOFTWARE.
|
|
|
|
|
2022-03-28 10:17:46 -04:00
|
|
|
#include "test_tools.h"
|
2024-04-24 20:53:54 +02:00
|
|
|
#include <mp-units/ext/type_traits.h>
|
|
|
|
#include <mp-units/framework/quantity_point.h> // IWYU pragma: keep
|
|
|
|
#include <mp-units/systems/isq/si_quantities.h>
|
2024-04-25 16:30:52 +02:00
|
|
|
#include <mp-units/systems/si.h>
|
2024-07-16 17:36:00 +02:00
|
|
|
#ifdef MP_UNITS_IMPORT_STD
|
|
|
|
import std;
|
|
|
|
#else
|
2024-04-24 20:53:54 +02:00
|
|
|
#include <chrono>
|
|
|
|
#include <concepts>
|
2024-02-23 21:09:17 +01:00
|
|
|
#include <ratio>
|
2024-07-16 17:36:00 +02:00
|
|
|
#endif
|
2020-10-07 12:02:08 +02:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2023-04-03 19:23:08 +02:00
|
|
|
using namespace mp_units;
|
|
|
|
using namespace mp_units::si::unit_symbols;
|
2020-10-07 12:02:08 +02:00
|
|
|
using namespace std::chrono_literals;
|
2021-02-17 02:57:59 -04:00
|
|
|
using sys_seconds = std::chrono::time_point<std::chrono::system_clock, std::chrono::seconds>;
|
2022-03-17 23:59:48 +01:00
|
|
|
using sys_days =
|
|
|
|
std::chrono::time_point<std::chrono::system_clock,
|
|
|
|
std::chrono::duration<long, std::ratio_multiply<std::ratio<24>, std::chrono::hours::period>>>;
|
2023-04-03 19:23:08 +02:00
|
|
|
template<Unit auto U, typename C, typename Rep = double>
|
2023-04-04 14:28:38 +02:00
|
|
|
using time_point = quantity_point<U, chrono_point_origin<C>, Rep>;
|
2020-10-07 12:02:08 +02:00
|
|
|
|
2021-02-16 19:18:14 +01:00
|
|
|
static_assert(QuantityLike<std::chrono::seconds>);
|
2023-08-23 12:49:14 +02:00
|
|
|
static_assert(!Quantity<std::chrono::seconds>);
|
|
|
|
static_assert(!QuantityPoint<std::chrono::seconds>);
|
|
|
|
static_assert(!QuantityPointLike<std::chrono::seconds>);
|
|
|
|
|
2021-02-17 02:57:59 -04:00
|
|
|
static_assert(QuantityPointLike<sys_seconds>);
|
2023-08-23 12:49:14 +02:00
|
|
|
static_assert(!Quantity<sys_seconds>);
|
|
|
|
static_assert(!QuantityLike<sys_seconds>);
|
|
|
|
static_assert(!QuantityPoint<sys_seconds>);
|
2020-10-07 12:02:08 +02:00
|
|
|
|
2021-01-04 13:05:26 +01:00
|
|
|
// construction - same rep type
|
2022-03-17 23:59:48 +01:00
|
|
|
static_assert(
|
2023-04-03 19:23:08 +02:00
|
|
|
std::constructible_from<quantity<isq::time[si::second], std::chrono::seconds::rep>, std::chrono::seconds>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<std::chrono::seconds, quantity<isq::time[si::second], std::chrono::seconds::rep>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(std::constructible_from<quantity<isq::time[si::hour], std::chrono::hours::rep>, std::chrono::hours>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<std::chrono::hours, quantity<isq::time[si::hour], std::chrono::hours::rep>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(std::constructible_from<quantity<isq::time[si::second], std::chrono::hours::rep>, std::chrono::hours>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<std::chrono::hours, quantity<isq::time[si::second], std::chrono::hours::rep>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(!std::constructible_from<quantity<isq::time[si::hour], std::chrono::seconds::rep>, std::chrono::seconds>);
|
|
|
|
static_assert(!std::convertible_to<std::chrono::seconds, quantity<isq::time[si::hour], std::chrono::seconds::rep>>);
|
2022-03-17 23:59:48 +01:00
|
|
|
static_assert(
|
2023-04-03 19:23:08 +02:00
|
|
|
std::constructible_from<time_point<si::second, std::chrono::system_clock, sys_seconds::rep>, sys_seconds>);
|
|
|
|
static_assert(
|
|
|
|
!std::constructible_from<time_point<si::second, std::chrono::steady_clock, sys_seconds::rep>, sys_seconds>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<sys_seconds, time_point<si::second, std::chrono::system_clock, sys_seconds::rep>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(std::constructible_from<time_point<si::day, std::chrono::system_clock, sys_days::rep>, sys_days>);
|
|
|
|
static_assert(!std::constructible_from<time_point<si::day, std::chrono::steady_clock, sys_days::rep>, sys_days>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<sys_days, time_point<si::day, std::chrono::system_clock, sys_days::rep>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(std::constructible_from<time_point<si::second, std::chrono::system_clock, sys_days::rep>, sys_days>);
|
|
|
|
static_assert(!std::constructible_from<time_point<si::second, std::chrono::steady_clock, sys_days::rep>, sys_days>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<sys_days, time_point<si::second, std::chrono::system_clock, sys_days::rep>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(!std::constructible_from<time_point<si::day, std::chrono::system_clock, sys_seconds::rep>, sys_seconds>);
|
|
|
|
static_assert(!std::convertible_to<sys_seconds, time_point<si::day, std::chrono::system_clock, sys_seconds::rep>>);
|
2020-10-07 12:02:08 +02:00
|
|
|
|
2021-01-04 13:05:26 +01:00
|
|
|
// construction - different rep type (integral to a floating-point)
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(std::constructible_from<quantity<isq::time[si::second]>, std::chrono::seconds>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<std::chrono::seconds, quantity<isq::time[si::second]>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(std::constructible_from<quantity<isq::time[si::second]>, std::chrono::hours>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<std::chrono::hours, quantity<isq::time[si::second]>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(std::constructible_from<quantity<isq::time[si::hour]>, std::chrono::seconds>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<std::chrono::seconds, quantity<isq::time[si::hour]>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(std::constructible_from<time_point<si::second, std::chrono::system_clock>, sys_seconds>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<sys_seconds, time_point<si::second, std::chrono::system_clock>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(std::constructible_from<time_point<si::second, std::chrono::system_clock>, sys_days>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<sys_days, time_point<si::second, std::chrono::system_clock>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(std::constructible_from<time_point<si::day, std::chrono::system_clock>, sys_seconds>);
|
2023-09-29 12:47:02 +02:00
|
|
|
static_assert(std::convertible_to<sys_seconds, time_point<si::day, std::chrono::system_clock>>);
|
2021-01-04 13:05:26 +01:00
|
|
|
|
2023-08-15 11:18:15 +02:00
|
|
|
static_assert(quantity<si::second>{1s} == 1 * s);
|
|
|
|
static_assert(quantity<isq::time[si::second]>{1s} == 1 * s);
|
|
|
|
static_assert(quantity<isq::period_duration[si::second]>{1s} == 1 * s);
|
2021-01-04 13:05:26 +01:00
|
|
|
|
2020-10-07 12:02:08 +02:00
|
|
|
// CTAD
|
2023-04-04 14:28:38 +02:00
|
|
|
static_assert(is_of_type<quantity{1s}, quantity<si::second, std::chrono::seconds::rep>>);
|
|
|
|
static_assert(is_of_type<quantity{1h}, quantity<si::hour, std::chrono::hours::rep>>);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(
|
|
|
|
is_of_type<quantity_point{sys_seconds{1s}}, time_point<si::second, std::chrono::system_clock, sys_seconds::rep>>);
|
|
|
|
static_assert(is_of_type<quantity_point{sys_days{sys_days::duration{1}}},
|
|
|
|
time_point<si::day, std::chrono::system_clock, sys_days::rep>>);
|
2020-10-07 12:02:08 +02:00
|
|
|
|
2024-07-16 12:13:29 +02:00
|
|
|
// conversion to chrono
|
|
|
|
static_assert(
|
|
|
|
std::constructible_from<std::chrono::seconds, quantity<isq::time[si::second], std::chrono::seconds::rep>>);
|
|
|
|
static_assert(std::convertible_to<quantity<isq::time[si::second], std::chrono::seconds::rep>, std::chrono::seconds>);
|
|
|
|
static_assert(std::constructible_from<std::chrono::hours, quantity<isq::time[si::hour], std::chrono::hours::rep>>);
|
|
|
|
static_assert(std::convertible_to<quantity<isq::time[si::hour], std::chrono::hours::rep>, std::chrono::hours>);
|
|
|
|
static_assert(
|
|
|
|
std::constructible_from<sys_seconds, time_point<si::second, std::chrono::system_clock, sys_seconds::rep>>);
|
|
|
|
static_assert(std::convertible_to<time_point<si::second, std::chrono::system_clock, sys_seconds::rep>, sys_seconds>);
|
|
|
|
|
2023-08-15 11:18:15 +02:00
|
|
|
// units mapping
|
|
|
|
static_assert(quantity{1ns} == 1 * ns);
|
|
|
|
static_assert(quantity{1us} == 1 * us);
|
|
|
|
static_assert(quantity{1ms} == 1 * ms);
|
|
|
|
static_assert(quantity{1s} == 1 * s);
|
|
|
|
static_assert(quantity{1min} == 1 * min);
|
|
|
|
static_assert(quantity{1h} == 1 * h);
|
|
|
|
static_assert(quantity{std::chrono::days{1}} == 1 * d);
|
|
|
|
static_assert(quantity{std::chrono::weeks{1}} == 7 * d);
|
|
|
|
static_assert(quantity{std::chrono::months{1}} == 2629746 * s);
|
|
|
|
static_assert(quantity{std::chrono::years{1}} == 31556952 * s);
|
|
|
|
|
2024-09-09 14:37:22 +02:00
|
|
|
// conversion from chrono
|
|
|
|
static_assert(quantity<ns>{1ns} == 1 * ns);
|
|
|
|
static_assert(quantity<ns>{1s} == 1 * s);
|
|
|
|
static_assert(quantity<s>{1ms} == 1 * ms);
|
|
|
|
|
|
|
|
// conversion to chrono
|
|
|
|
static_assert(std::chrono::nanoseconds(quantity{1 * ns}) == 1ns);
|
|
|
|
static_assert(std::chrono::nanoseconds(quantity{1 * s}) == 1s);
|
|
|
|
|
2020-10-07 12:02:08 +02:00
|
|
|
// operators
|
2025-06-19 18:49:21 +02:00
|
|
|
static_assert([q = 1 * s]() mutable { return q += quantity{1s}; }() == 2 * s);
|
|
|
|
static_assert([q = 2 * s]() mutable { return q -= quantity{1s}; }() == 1 * s);
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(quantity{1s} + 1 * s == 2 * s);
|
|
|
|
static_assert(quantity{1s} + 1 * min == 61 * s);
|
2024-10-15 21:04:52 +02:00
|
|
|
static_assert(1 * s + quantity{1s} == 2 * s);
|
|
|
|
static_assert(1 * min + quantity{1s} == 61 * s);
|
2023-09-29 21:40:24 -06:00
|
|
|
static_assert(10 * m / quantity{2s} == 5 * m / s);
|
2023-08-23 12:49:14 +02:00
|
|
|
static_assert(quantity_point{sys_seconds{1s}} + 1 * s == chrono_point_origin<std::chrono::system_clock> + 2 * s);
|
|
|
|
static_assert(quantity_point{sys_seconds{1s}} + 1 * min == chrono_point_origin<std::chrono::system_clock> + 61 * s);
|
2020-10-07 12:02:08 +02:00
|
|
|
|
2024-10-15 21:04:52 +02:00
|
|
|
template<typename... Ts>
|
|
|
|
consteval bool invalid_arithmetic(Ts... ts)
|
|
|
|
{
|
|
|
|
return !requires { (... + ts); } && !requires { (... - ts); };
|
|
|
|
}
|
|
|
|
static_assert(invalid_arithmetic(1s, 1 * s));
|
|
|
|
static_assert(invalid_arithmetic(1 * s, 1s));
|
|
|
|
|
2023-04-21 15:18:58 +01:00
|
|
|
// to_chrono_duration
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(to_chrono_duration(1 * s) == 1s);
|
|
|
|
static_assert(to_chrono_duration(2 * h) == 2h);
|
|
|
|
static_assert(to_chrono_duration(3 * ns) == 3ns);
|
|
|
|
static_assert(to_chrono_duration(quantity{1s}) == 1s);
|
|
|
|
static_assert(to_chrono_duration(quantity{2h}) == 2h);
|
|
|
|
static_assert(to_chrono_duration(quantity{3ns}) == 3ns);
|
|
|
|
static_assert(is_same_v<decltype(to_chrono_duration(1 * s))::period, std::ratio<1>>);
|
|
|
|
static_assert(is_same_v<decltype(to_chrono_duration(2 * h))::period, std::ratio<3600>>);
|
|
|
|
static_assert(is_same_v<decltype(to_chrono_duration(3 * ns))::period, std::nano>);
|
2021-11-23 11:17:47 +01:00
|
|
|
|
2023-04-21 15:18:58 +01:00
|
|
|
// to_chrono_time_point
|
2023-04-03 19:23:08 +02:00
|
|
|
static_assert(to_chrono_time_point(quantity_point{sys_seconds{1s}}) == sys_seconds{1s});
|
|
|
|
static_assert(to_chrono_time_point(quantity_point{sys_days{sys_days::duration{1}}}) == sys_days{sys_days::duration{1}});
|
2021-11-23 14:37:08 +01:00
|
|
|
|
2020-10-07 12:02:08 +02:00
|
|
|
} // namespace
|