// 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/math.h" #include "units/physical/si/area.h" #include "units/physical/si/frequency.h" #include "units/physical/si/speed.h" #include "units/physical/si/volume.h" #include "units/physical/us/length.h" #include #include namespace { using namespace units; using namespace units::physical::si; template inline constexpr bool compare = DOWNCAST_MODE != 0 ? std::is_same_v : (std::is_same_v || units::equivalent); // class invariants template concept invalid_types = requires { !requires { typename quantity; }; // unit of a different dimension !requires { typename quantity>; }; // quantity used as Rep !requires { typename quantity; }; // reordered arguments }; static_assert(invalid_types); // member types static_assert(is_same_v::rep, int>); static_assert(is_same_v::rep, double>); static_assert(is_same_v::unit, metre>); static_assert(is_same_v::unit, kilometre>); // constructors static_assert(length().count() == 0); constexpr length km{1000}; static_assert(km.count() == 1000); static_assert(length(km).count() == km.count()); static_assert(length(1).count() == 1); static_assert(!std::is_constructible_v, double>); // truncating conversion static_assert(length(1.0).count() == 1.0); static_assert(length(1).count() == 1.0); static_assert(length(3.14).count() == 3.14); static_assert(length(km).count() == 1000); static_assert(!std::is_constructible_v, length>); // truncating conversion static_assert(length(1000.0_q_m).count() == 1000.0); static_assert(length(km).count() == 1000.0); static_assert(length(1_q_km).count() == 1000); static_assert(!std::is_constructible_v, physical::si::time>); // different dimensions static_assert(!std::is_constructible_v, length>); // truncating conversion // assignment operator static_assert([]() { length l1(1), l2(2); return l2 = l1; }().count() == 1); // static member functions static_assert(length::zero().count() == 0); static_assert(length::min().count() == std::numeric_limits::lowest()); static_assert(length::max().count() == std::numeric_limits::max()); static_assert(length::zero().count() == 0.0); static_assert(length::min().count() == std::numeric_limits::lowest()); static_assert(length::max().count() == std::numeric_limits::max()); // unary member operators static_assert((+km).count() == 1000); static_assert((-km).count() == -1000); static_assert((+(-km)).count() == -1000); static_assert((-(-km)).count() == 1000); static_assert([](auto v) { auto vv = v++; return std::pair(v, vv); }(km) == std::pair(length(1001), length(1000))); static_assert([](auto v) { auto vv = ++v; return std::pair(v, vv); }(km) == std::pair(length(1001), length(1001))); static_assert([](auto v) { auto vv = v--; return std::pair(v, vv); }(km) == std::pair(length(999), length(1000))); static_assert([](auto v) { auto vv = --v; return std::pair(v, vv); }(km) == std::pair(length(999), length(999))); // compound assignment static_assert((1_q_m += 1_q_m).count() == 2); static_assert((2_q_m -= 1_q_m).count() == 1); static_assert((1_q_m *= 2).count() == 2); static_assert((2_q_m /= 2).count() == 1); static_assert((7_q_m %= 2).count() == 1); static_assert((7_q_m %= 2_q_m).count() == 1); static_assert((2.5_q_m += 3_q_m).count() == 5.5); static_assert((2.5_q_m += 3.5_q_m).count() == 6); static_assert((2.5_q_m *= 3).count() == 7.5); static_assert((2.5_q_m *= 3.5).count() == 8.75); // operations not allowed for the respective quantities template concept invalid_compound_assignments = requires() { !requires(length l) { l %= 2.; }; !requires(length l) { l %= 2; }; !requires(length l) { l %= 2.; }; !requires(length l) { l %= 2._q_m; }; !requires(length l) { l %= 2_q_m; }; !requires(length l) { l %= 2._q_m; }; !requires(length l) { l += 3.5_q_m; }; !requires(length l) { l *= 3.5_q_m; }; }; static_assert(invalid_compound_assignments); // non-member arithmetic operators static_assert(compare() + length()), length>); static_assert(compare() + length()), length>); static_assert(compare() + length()), length>); static_assert(compare() - length()), length>); static_assert(compare() - length()), length>); static_assert(compare() * 1.0), length>); static_assert(compare()), length>); static_assert( compare() * physical::si::time()), length>); static_assert( compare() * physical::si::time()), length, int>>); static_assert(compare() * physical::si::time()), quantity, units::exponent>, scaled_unit>>); static_assert(compare()), frequency>); static_assert(compare()), frequency, int>>); static_assert(compare()), physical::si::time>); static_assert(compare()), quantity>, scaled_unit>>); static_assert(compare() / 1.0), length>); static_assert(compare() / length()), dimensionless>); static_assert(compare() / length()), dimensionless, double>>); static_assert( compare() / physical::si::time()), speed>); static_assert( compare() / physical::si::time()), speed>>); static_assert(compare() / length()), quantity, units::exponent>, scaled_unit>>); static_assert(compare() % short(1)), length>); static_assert(compare() % length(1)), length>); static_assert((1_q_m + km).count() == 1001); static_assert((1_q_m + 1_q_km).count() == 1001); static_assert((km - 1_q_m).count() == 999); static_assert((1_q_km - 1_q_m).count() == 999); static_assert((2_q_m * 2).count() == 4); static_assert((3 * 3_q_m).count() == 9); static_assert((4_q_m / 2).count() == 2); static_assert((4_q_km / 2_q_m).count() == 2); static_assert((4000_q_m / 2_q_m).count() == 2000); static_assert((7_q_m % 2).count() == 1); static_assert((7_q_m % 2_q_m).count() == 1); static_assert((7_q_km % 2000_q_m).count() == 1000); static_assert((10_q_km2 * 10_q_km2) / 50_q_km2 == 2_q_km2); constexpr auto q1 = 10_q_km / 5_q_m; static_assert(compare, std::int64_t>>); static_assert(q1.count() == 2); constexpr dimensionless q2 = q1; static_assert(q2.count() == 2000); static_assert(quantity_cast(q1).count() == 2000); constexpr auto q3 = 10_q_s * 2_q_kHz; static_assert(compare, std::int64_t>>); static_assert(q3.count() == 20); // comparators static_assert(2_q_m + 1_q_m == 3_q_m); static_assert(!(2_q_m + 2_q_m == 3_q_m)); static_assert(2_q_m + 2_q_m != 3_q_m); static_assert(!(2_q_m + 2_q_m != 4_q_m)); static_assert(2_q_m > 1_q_m); static_assert(!(1_q_m > 1_q_m)); static_assert(1_q_m < 2_q_m); static_assert(!(2_q_m < 2_q_m)); static_assert(2_q_m >= 1_q_m); static_assert(2_q_m >= 2_q_m); static_assert(!(2_q_m >= 3_q_m)); static_assert(1_q_m <= 2_q_m); static_assert(2_q_m <= 2_q_m); static_assert(!(3_q_m <= 2_q_m)); static_assert(3_q_m == 3.0_q_m); static_assert(3_q_m != 3.14_q_m); static_assert(2_q_m > 1.0_q_m); static_assert(1.0_q_m < 2_q_m); static_assert(2.0_q_m >= 1_q_m); static_assert(1_q_m <= 2.0_q_m); static_assert(1000_q_m == 1_q_km); static_assert(1001_q_m != 1_q_km); static_assert(1001_q_m > 1_q_km); static_assert(999_q_m < 1_q_km); static_assert(1000_q_m >= 1_q_km); static_assert(1000_q_m <= 1_q_km); // alias units static_assert(2_q_l + 2_q_ml == 2002_q_ml); static_assert(2_q_l + 2_q_ml == 2002_q_cm3); static_assert(2_q_l + 2_q_cm3 == 2002_q_ml); static_assert(2_q_dm3 + 2_q_cm3 == 2002_q_ml); // is_quantity static_assert(Quantity>); // common_quantity static_assert(compare, length>, length>); static_assert(compare, length>, length>); static_assert( compare, length>, length>); // common_type using namespace units::physical::us::literals; static_assert(std::equality_comparable); static_assert(std::equality_comparable_with); static_assert(0_q_m == 0_q_ft_us); static_assert(std::equality_comparable_with); static_assert(std::equality_comparable_with, int>); static_assert(std::equality_comparable_with, double>); static_assert(std::equality_comparable_with, int>); static_assert(!std::equality_comparable_with, double>); // quantity_cast static_assert(compare>(2_q_km))::unit, metre>); static_assert(quantity_cast>(2_q_km).count() == 2000); static_assert(quantity_cast>(2000_q_m).count() == 2); static_assert(quantity_cast>(1.23_q_m).count() == 1); static_assert(quantity_cast(2_q_km).count() == 2000); static_assert(quantity_cast(2000_q_m).count() == 2); static_assert(quantity_cast(1.23_q_m).count() == 1); // dimensionless static_assert(std::is_convertible_v>); static_assert(std::is_convertible_v>); static_assert(!std::is_convertible_v>); static_assert(std::is_convertible_v>); static_assert(!std::is_convertible_v>>); static_assert(std::is_constructible_v>, double>); static_assert(dimensionless(1.23) + dimensionless(1.23) == dimensionless(2.46)); static_assert(dimensionless(1.23) + dimensionless(1.23) == 2.46); static_assert(dimensionless(1.23) + 1.23 == 2.46); static_assert(1.23 + dimensionless(1.23) == 2.46); static_assert(dimensionless(1) + 1 == 2); static_assert(dimensionless(1) + 1 == 2); template concept invalid_dimensionless_operations = requires { !requires(dimensionless d) { d + 1.23; }; !requires(dimensionless d) { 1.23 + d; }; !requires(dimensionless, Int> d) { 1 + d; }; !requires(dimensionless, Int> d) { d + 1; }; }; static_assert(invalid_dimensionless_operations); static_assert(compare>); static_assert(quantity_cast(50._q_m / 100._q_m).count() == 50); static_assert(50._q_m / 100._q_m == dimensionless(50)); static_assert(dimensionless(dimensionless(50)).count() == 0.5); // time static_assert(1_q_h == 3600_q_s); template concept no_crossdimensional_equality = !requires { 1_q_s == length(1); }; static_assert(no_crossdimensional_equality); // length static_assert(1_q_km == 1000_q_m); static_assert(1_q_km + 1_q_m == 1001_q_m); static_assert(10_q_km / 5_q_km == 2); static_assert(10_q_km / 2 == 5_q_km); // speed static_assert(10_q_m / 5_q_s == 2_q_m_per_s); static_assert(10 / 5_q_s * 1_q_m == 2_q_m_per_s); static_assert(1_q_km / 1_q_s == 1000_q_m_per_s); static_assert(2_q_km_per_h * 2_q_h == 4_q_km); static_assert(2_q_km / 2_q_km_per_h == 1_q_h); static_assert(compare(2_q_m)), decltype(4_q_m2)>); // downcasting #if DOWNCAST_MODE == 0 static_assert(std::is_same_v, units::exp>, scaled_unit, std::int64_t>>); static_assert(std::is_same_v, std::int64_t>>); #else static_assert(std::is_same_v>); static_assert(std::is_same_v>); #endif } // namespace