/* * Copyright (C) 2008-2019 TrinityCore * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation; either version 2 of the License, or (at your * option) any later version. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License along * with this program. If not, see . */ #ifndef TRINITY_ADVSTD_H #define TRINITY_ADVSTD_H #include #include #include #include // this namespace holds implementations of upcoming stdlib features that our c++ version doesn't have yet namespace advstd { // C++17 std::apply (constrained to only function pointers, not all callable) template using apply_tuple_type = std::tuple>...>; template R apply_impl(R(*func)(Ts...), apply_tuple_type&& args, std::index_sequence) { return func(std::get(std::forward>(args))...); } template R apply(R(*func)(Ts...), apply_tuple_type&& args) { return apply_impl(func, std::forward>(args), std::index_sequence_for{}); } #define forward_1v(stdname, type) template constexpr type stdname ## _v = std::stdname::value #define forward_2v(stdname, type) template constexpr type stdname ## _v = std::stdname::value // C++17 std::is_same_v forward_2v(is_same, bool); // C++17 std::is_integral_v forward_1v(is_integral, bool); // C++17 std::is_assignable_v forward_2v(is_assignable, bool); // C++17 std::is_signed_v forward_1v(is_signed, bool); // C++17 std::is_unsigned_v forward_1v(is_unsigned, bool); // C++17 std::is_base_of_v forward_2v(is_base_of, bool); // C++17 std::is_floating_point_v forward_1v(is_floating_point, bool); // C++17 std::is_pointer_v forward_1v(is_pointer, bool); // C++17 std::is_reference_v forward_1v(is_reference, bool); // C++17 std::tuple_size_v forward_1v(tuple_size, size_t); // C++17 std::is_enum_v forward_1v(is_enum, bool); // C++17 std::is_arithmetic_v forward_1v(is_arithmetic, bool); // C++17 std::is_move_assignable_v forward_1v(is_move_assignable, bool); #undef forward_1v #undef forward_2v // C++17 std::size template constexpr auto size(const C& c) { return c.size(); } template constexpr std::size_t size(const T(&)[N]) noexcept { return N; } // C++17 std::data template constexpr auto data(C& c) { return c.data(); } template constexpr auto data(C const& c) { return c.data(); } template constexpr T* data(T(&a)[N]) noexcept { return a; } template constexpr T const* data(const T(&a)[N]) noexcept { return a; } template constexpr T const* data(std::initializer_list l) noexcept { return l.begin(); } // C++17 std::gcd template constexpr std::enable_if_t && advstd::is_unsigned_v, std::common_type_t> gcd(T1 _m, T2 _n) { using T = std::common_type_t; T n=_n, m=_m; while (n) { T o = m; m = n; n = o%n; } return m; } // C++17 std::lcm template constexpr std::enable_if_t && advstd::is_unsigned_v, std::common_type_t> lcm(T1 m, T2 n) { return (m/gcd(m, n))*n; } // C++20 std::remove_cvref_t template using remove_cvref_t = std::remove_cv_t>; } #endif