/*
* 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