2022-09-28 12:49:28 +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.
|
|
|
|
|
|
|
|
#include <units/dimension.h>
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2022-10-18 17:45:32 +02:00
|
|
|
using namespace units;
|
|
|
|
|
2022-09-28 12:49:28 +02:00
|
|
|
template<auto V, typename T>
|
|
|
|
inline constexpr bool is_of_type = std::is_same_v<std::remove_cvref_t<decltype(V)>, T>;
|
|
|
|
|
|
|
|
using one_dim_ = struct one_dim;
|
|
|
|
|
|
|
|
// clang-format off
|
|
|
|
inline constexpr struct length_dim_ : base_dimension<"L"> {} length_dim;
|
|
|
|
inline constexpr struct time_dim_ : base_dimension<"T"> {} time_dim;
|
2022-10-18 17:45:32 +02:00
|
|
|
inline constexpr struct mass_dim_ : base_dimension<"M"> {} mass_dim;
|
2022-09-28 12:49:28 +02:00
|
|
|
|
|
|
|
inline constexpr struct frequency_dim_ : decltype(1 / time_dim) {} frequency_dim;
|
2022-10-09 21:32:38 +01:00
|
|
|
inline constexpr struct action_dim_ : decltype(1 / time_dim) {} action_dim;
|
2022-09-28 12:49:28 +02:00
|
|
|
inline constexpr struct area_dim_ : decltype(length_dim * length_dim) {} area_dim;
|
|
|
|
inline constexpr struct volume_dim_ : decltype(area_dim * length_dim) {} volume_dim;
|
|
|
|
inline constexpr struct speed_dim_ : decltype(length_dim / time_dim) {} speed_dim;
|
2022-10-09 21:32:38 +01:00
|
|
|
inline constexpr struct velocity_dim_ : speed_dim_ {} velocity_dim;
|
2022-09-28 12:49:28 +02:00
|
|
|
inline constexpr struct acceleration_dim_ : decltype(speed_dim / time_dim) {} acceleration_dim;
|
2022-10-18 17:45:32 +02:00
|
|
|
inline constexpr struct force_dim_ : decltype(mass_dim * acceleration_dim) {} force_dim;
|
|
|
|
inline constexpr struct moment_of_force_dim_ : decltype(length_dim * force_dim) {} moment_of_force_dim;
|
|
|
|
inline constexpr struct torque_dim_ : decltype(moment_of_force_dim) {} torque_dim;
|
|
|
|
inline constexpr struct pressure_dim_ : decltype(force_dim / area_dim) {} pressure_dim;
|
|
|
|
inline constexpr struct stress_dim_ : decltype(pressure_dim) {} stress_dim;
|
|
|
|
inline constexpr struct strain_dim_ : decltype(stress_dim / stress_dim) {} strain_dim;
|
|
|
|
inline constexpr struct power_dim_ : decltype(force_dim * speed_dim) {} power_dim;
|
|
|
|
inline constexpr struct efficiency_dim_ : decltype(power_dim / power_dim) {} efficiency_dim;
|
|
|
|
inline constexpr struct energy_dim_ : decltype(force_dim * length_dim) {} energy_dim;
|
2022-09-28 12:49:28 +02:00
|
|
|
// clang-format on
|
|
|
|
|
|
|
|
// concepts verification
|
|
|
|
static_assert(BaseDimension<length_dim_>);
|
|
|
|
static_assert(!BaseDimension<frequency_dim_>);
|
|
|
|
static_assert(!DerivedDimension<length_dim_>);
|
|
|
|
static_assert(DerivedDimension<frequency_dim_>);
|
|
|
|
static_assert(Dimension<length_dim_>);
|
|
|
|
static_assert(Dimension<frequency_dim_>);
|
|
|
|
|
|
|
|
static_assert(DerivedDimension<one_dim_>);
|
|
|
|
static_assert(DerivedDimension<decltype(length_dim / length_dim)>); // one_dim
|
|
|
|
static_assert(BaseDimension<decltype(speed_dim * time_dim)>); // length_dim
|
|
|
|
|
|
|
|
// derived dimension expression template syntax verification
|
|
|
|
static_assert(is_of_type<1 / time_dim, derived_dimension<one_dim_, per<time_dim_>>>);
|
|
|
|
static_assert(is_of_type<1 / (1 / time_dim), time_dim_>);
|
|
|
|
|
|
|
|
static_assert(is_of_type<one_dim * time_dim, time_dim_>);
|
|
|
|
static_assert(is_of_type<time_dim * one_dim, time_dim_>);
|
|
|
|
static_assert(is_of_type<one_dim * (1 / time_dim), derived_dimension<one_dim_, per<time_dim_>>>);
|
|
|
|
static_assert(is_of_type<1 / time_dim * one_dim, derived_dimension<one_dim_, per<time_dim_>>>);
|
|
|
|
|
|
|
|
static_assert(is_of_type<length_dim * time_dim, derived_dimension<length_dim_, time_dim_>>);
|
2022-10-20 14:06:22 +02:00
|
|
|
static_assert(is_of_type<length_dim * length_dim, derived_dimension<power<length_dim_, 2>>>);
|
2022-09-28 12:49:28 +02:00
|
|
|
|
2022-10-20 14:06:22 +02:00
|
|
|
static_assert(is_of_type<length_dim * length_dim * time_dim, derived_dimension<power<length_dim_, 2>, time_dim_>>);
|
|
|
|
static_assert(is_of_type<length_dim * time_dim * length_dim, derived_dimension<power<length_dim_, 2>, time_dim_>>);
|
2022-09-28 12:49:28 +02:00
|
|
|
|
2022-10-20 14:06:22 +02:00
|
|
|
static_assert(is_of_type<length_dim*(time_dim* length_dim), derived_dimension<power<length_dim_, 2>, time_dim_>>);
|
|
|
|
static_assert(is_of_type<time_dim*(length_dim* length_dim), derived_dimension<power<length_dim_, 2>, time_dim_>>);
|
2022-09-28 12:49:28 +02:00
|
|
|
|
|
|
|
static_assert(is_of_type<1 / time_dim * length_dim, derived_dimension<length_dim_, per<time_dim_>>>);
|
|
|
|
static_assert(is_of_type<1 / time_dim * time_dim, one_dim_>);
|
|
|
|
|
|
|
|
static_assert(is_of_type<time_dim / one_dim, time_dim_>);
|
|
|
|
static_assert(is_of_type<1 / time_dim / one_dim, derived_dimension<one_dim_, per<time_dim_>>>);
|
|
|
|
|
|
|
|
static_assert(is_of_type<length_dim / time_dim * time_dim, length_dim_>);
|
2022-10-20 14:06:22 +02:00
|
|
|
static_assert(is_of_type<1 / time_dim * (1 / time_dim), derived_dimension<one_dim_, per<power<time_dim_, 2>>>>);
|
|
|
|
static_assert(is_of_type<1 / (time_dim * time_dim), derived_dimension<one_dim_, per<power<time_dim_, 2>>>>);
|
|
|
|
static_assert(is_of_type<1 / (1 / (time_dim * time_dim)), derived_dimension<power<time_dim_, 2>>>);
|
2022-09-28 12:49:28 +02:00
|
|
|
|
|
|
|
static_assert(
|
2022-10-20 14:06:22 +02:00
|
|
|
is_of_type<length_dim / time_dim * (1 / time_dim), derived_dimension<length_dim_, per<power<time_dim_, 2>>>>);
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(is_of_type<length_dim / time_dim*(length_dim / time_dim),
|
2022-10-20 14:06:22 +02:00
|
|
|
derived_dimension<power<length_dim_, 2>, per<power<time_dim_, 2>>>>);
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(is_of_type<length_dim / time_dim*(time_dim / length_dim), one_dim_>);
|
|
|
|
|
|
|
|
static_assert(is_of_type<speed_dim / acceleration_dim, time_dim_>);
|
|
|
|
static_assert(is_of_type<acceleration_dim / speed_dim, derived_dimension<one_dim_, per<time_dim_>>>);
|
2022-10-20 14:06:22 +02:00
|
|
|
static_assert(is_of_type<speed_dim * speed_dim / length_dim, derived_dimension<length_dim_, per<power<time_dim_, 2>>>>);
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(
|
2022-10-20 14:06:22 +02:00
|
|
|
is_of_type<1 / (speed_dim * speed_dim) * length_dim, derived_dimension<power<time_dim_, 2>, per<length_dim_>>>);
|
2022-09-28 12:49:28 +02:00
|
|
|
|
2022-10-18 17:45:32 +02:00
|
|
|
// comparisons of the same dimensions
|
|
|
|
static_assert(length_dim == length_dim);
|
|
|
|
static_assert(speed_dim == speed_dim);
|
|
|
|
|
|
|
|
// comparisons of equivalent dimensions (named vs unnamed/derived)
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(length_dim / length_dim == one_dim);
|
|
|
|
|
2022-10-06 23:43:01 +01:00
|
|
|
static_assert(1 / time_dim != frequency_dim);
|
2022-10-09 21:32:38 +01:00
|
|
|
static_assert(convertible(1 / time_dim, frequency_dim));
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(1 / frequency_dim == time_dim);
|
|
|
|
static_assert(frequency_dim * time_dim == one_dim);
|
2022-10-09 21:32:38 +01:00
|
|
|
static_assert(std::is_same_v<std::common_type_t<decltype(1 / time_dim), decltype(frequency_dim)>, frequency_dim_>);
|
|
|
|
static_assert(std::is_same_v<std::common_type_t<decltype(frequency_dim), decltype(1 / time_dim)>, frequency_dim_>);
|
2022-09-28 12:49:28 +02:00
|
|
|
|
2022-10-06 23:43:01 +01:00
|
|
|
static_assert(length_dim * length_dim != area_dim);
|
2022-10-09 21:32:38 +01:00
|
|
|
static_assert(convertible(length_dim * length_dim, area_dim));
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(length_dim * length_dim != volume_dim);
|
|
|
|
static_assert(area_dim / length_dim == length_dim);
|
2022-10-09 21:32:38 +01:00
|
|
|
static_assert(std::is_same_v<std::common_type_t<decltype(length_dim * length_dim), decltype(area_dim)>, area_dim_>);
|
|
|
|
static_assert(std::is_same_v<std::common_type_t<decltype(area_dim), decltype(length_dim * length_dim)>, area_dim_>);
|
2022-09-28 12:49:28 +02:00
|
|
|
|
2022-10-06 23:43:01 +01:00
|
|
|
static_assert(length_dim * length_dim * length_dim != volume_dim);
|
|
|
|
static_assert(area_dim * length_dim != volume_dim);
|
|
|
|
static_assert(volume_dim / length_dim != area_dim);
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(volume_dim / length_dim / length_dim == length_dim);
|
2022-10-06 23:43:01 +01:00
|
|
|
static_assert(area_dim * area_dim / length_dim != volume_dim);
|
|
|
|
static_assert(area_dim * (area_dim / length_dim) != volume_dim);
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(volume_dim / (length_dim * length_dim) == length_dim);
|
|
|
|
|
2022-10-06 23:43:01 +01:00
|
|
|
static_assert(length_dim / time_dim != speed_dim);
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(length_dim * time_dim != speed_dim);
|
|
|
|
static_assert(length_dim / time_dim / time_dim != speed_dim);
|
|
|
|
static_assert(length_dim / speed_dim == time_dim);
|
|
|
|
static_assert(speed_dim * time_dim == length_dim);
|
2022-10-09 21:32:38 +01:00
|
|
|
static_assert(std::is_same_v<std::common_type_t<decltype(length_dim / time_dim), decltype(speed_dim)>, speed_dim_>);
|
|
|
|
static_assert(std::is_same_v<std::common_type_t<decltype(speed_dim), decltype(length_dim / time_dim)>, speed_dim_>);
|
|
|
|
static_assert(std::is_same_v<std::common_type_t<decltype(length_dim / time_dim), decltype(length_dim / time_dim)>,
|
|
|
|
decltype(length_dim / time_dim)>);
|
2022-09-28 12:49:28 +02:00
|
|
|
|
2022-10-06 23:43:01 +01:00
|
|
|
static_assert(length_dim / time_dim / time_dim != acceleration_dim);
|
|
|
|
static_assert(length_dim / (time_dim * time_dim) != acceleration_dim);
|
|
|
|
static_assert(speed_dim / time_dim != acceleration_dim);
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(speed_dim / acceleration_dim == time_dim);
|
2022-10-06 23:43:01 +01:00
|
|
|
static_assert(acceleration_dim * time_dim != speed_dim);
|
2022-09-28 12:49:28 +02:00
|
|
|
static_assert(acceleration_dim * (time_dim * time_dim) == length_dim);
|
2022-10-06 23:43:01 +01:00
|
|
|
static_assert(acceleration_dim / speed_dim != frequency_dim);
|
2022-10-09 21:32:38 +01:00
|
|
|
|
2022-10-18 17:45:32 +02:00
|
|
|
// comparison of convertible named dimensions
|
|
|
|
static_assert(velocity_dim != speed_dim);
|
|
|
|
static_assert(convertible(speed_dim, velocity_dim));
|
|
|
|
static_assert(std::is_same_v<std::common_type_t<decltype(velocity_dim), decltype(speed_dim)>, velocity_dim_>);
|
|
|
|
static_assert(std::is_same_v<std::common_type_t<decltype(speed_dim), decltype(velocity_dim)>, velocity_dim_>);
|
|
|
|
|
2022-10-19 11:21:15 +02:00
|
|
|
// comparison of convertible unnamed dimensions
|
|
|
|
static_assert(
|
2022-10-20 14:06:22 +02:00
|
|
|
is_of_type<mass_dim * acceleration_dim, derived_dimension<length_dim_, mass_dim_, per<power<time_dim_, 2>>>>);
|
2022-10-19 11:21:15 +02:00
|
|
|
static_assert(
|
2022-10-20 14:06:22 +02:00
|
|
|
is_of_type<acceleration_dim * mass_dim, derived_dimension<length_dim_, mass_dim_, per<power<time_dim_, 2>>>>);
|
2022-10-19 11:21:15 +02:00
|
|
|
static_assert(mass_dim * acceleration_dim == acceleration_dim * mass_dim);
|
|
|
|
static_assert(convertible(mass_dim * acceleration_dim, acceleration_dim* mass_dim));
|
|
|
|
|
2022-10-18 17:45:32 +02:00
|
|
|
// comparisons of equivalent but not convertible dimensions
|
|
|
|
static_assert(energy_dim != torque_dim);
|
|
|
|
static_assert(!convertible(energy_dim, torque_dim));
|
|
|
|
|
|
|
|
static_assert(force_dim * length_dim != energy_dim);
|
|
|
|
static_assert(force_dim * length_dim != torque_dim);
|
|
|
|
static_assert(convertible(force_dim * length_dim, energy_dim));
|
|
|
|
static_assert(convertible(force_dim * length_dim, torque_dim));
|
2022-10-09 21:32:38 +01:00
|
|
|
template<auto T1, auto T2>
|
|
|
|
concept no_common_type = requires {
|
|
|
|
requires !requires { typename std::common_type_t<decltype(T1), decltype(T2)>; };
|
|
|
|
requires !requires { typename std::common_type_t<decltype(T2), decltype(T1)>; };
|
|
|
|
};
|
2022-10-18 17:45:32 +02:00
|
|
|
static_assert(no_common_type<energy_dim, torque_dim>);
|
|
|
|
|
|
|
|
static_assert(frequency_dim != action_dim);
|
|
|
|
static_assert(!convertible(frequency_dim, action_dim));
|
2022-10-09 21:32:38 +01:00
|
|
|
static_assert(no_common_type<frequency_dim, action_dim>);
|
|
|
|
|
2022-10-18 17:45:32 +02:00
|
|
|
// Dimensionless
|
|
|
|
static_assert(convertible(power_dim / power_dim, efficiency_dim));
|
|
|
|
static_assert(power_dim / power_dim != efficiency_dim);
|
|
|
|
static_assert(one_dim != efficiency_dim);
|
|
|
|
|
|
|
|
static_assert(!convertible(efficiency_dim, strain_dim));
|
|
|
|
static_assert(efficiency_dim != strain_dim);
|
|
|
|
|
|
|
|
static_assert(stress_dim / stress_dim != strain_dim);
|
|
|
|
static_assert(stress_dim / stress_dim != efficiency_dim);
|
|
|
|
static_assert(convertible(stress_dim / stress_dim, strain_dim));
|
|
|
|
static_assert(convertible(stress_dim / stress_dim, efficiency_dim));
|
|
|
|
|
|
|
|
// comparison of not equivalent dimensions
|
|
|
|
static_assert(length_dim != time_dim);
|
|
|
|
static_assert(!convertible(length_dim, time_dim));
|
|
|
|
|
|
|
|
static_assert(acceleration_dim != speed_dim);
|
|
|
|
static_assert(!convertible(acceleration_dim, speed_dim));
|
2022-10-09 21:32:38 +01:00
|
|
|
|
2022-10-23 10:47:12 +02:00
|
|
|
// power
|
|
|
|
static_assert(is_of_type<pow<2>(length_dim), derived_dimension<power<length_dim_, 2>>>);
|
|
|
|
static_assert(is_of_type<pow<1, 2>(length_dim), derived_dimension<power<length_dim_, 1, 2>>>);
|
|
|
|
static_assert(is_of_type<pow<1, 2>(length_dim* length_dim), length_dim_>);
|
|
|
|
static_assert(is_of_type<pow<1, 3>(length_dim* length_dim* length_dim), length_dim_>);
|
|
|
|
static_assert(is_of_type<pow<1, 3>(length_dim* length_dim), derived_dimension<power<length_dim_, 2, 3>>>);
|
|
|
|
static_assert(is_of_type<pow<1, 2>(length_dim / time_dim),
|
|
|
|
derived_dimension<power<length_dim_, 1, 2>, per<power<time_dim_, 1, 2>>>>);
|
|
|
|
static_assert(is_of_type<pow<1, 2>(length_dim / (time_dim * time_dim)),
|
|
|
|
derived_dimension<power<length_dim_, 1, 2>, per<time_dim_>>>);
|
|
|
|
|
|
|
|
static_assert(is_same_v<decltype(pow<2>(length_dim)), decltype(length_dim * length_dim)>);
|
|
|
|
static_assert(
|
|
|
|
is_same_v<decltype(pow<2>(length_dim / time_dim)), decltype(length_dim * length_dim / time_dim / time_dim)>);
|
|
|
|
|
2022-10-09 21:32:38 +01:00
|
|
|
} // namespace
|