2020-11-29 12:03:50 +00:00
|
|
|
#pragma once // No BOM and only basic ASCII in this header, or a neko will die
|
2015-01-12 18:12:06 +00:00
|
|
|
|
2020-12-12 12:01:29 +00:00
|
|
|
#include "util/types.hpp"
|
2016-08-14 00:22:19 +00:00
|
|
|
|
2016-05-13 14:01:48 +00:00
|
|
|
#include <string>
|
2016-02-01 21:55:43 +00:00
|
|
|
|
|
|
|
namespace fmt
|
|
|
|
{
|
2020-12-18 07:39:54 +00:00
|
|
|
template <typename CharT, usz N, typename... Args>
|
2020-02-04 18:37:00 +00:00
|
|
|
static std::string format(const CharT(&)[N], const Args&...);
|
2022-03-16 18:59:20 +00:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
2022-03-16 19:44:47 +00:00
|
|
|
// Get a string for a windows error (DWORD). Optionally a module HANDLE can be passed.
|
|
|
|
std::string win_error_to_string(unsigned long error, void* module_handle = nullptr);
|
2022-03-16 18:59:20 +00:00
|
|
|
#endif
|
2016-08-03 20:51:05 +00:00
|
|
|
}
|
|
|
|
|
2020-12-22 11:05:05 +00:00
|
|
|
template <typename T, typename = void>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_unveil
|
|
|
|
{
|
2017-05-14 20:29:33 +00:00
|
|
|
static_assert(sizeof(T) > 0, "fmt_unveil<> error: incomplete type");
|
2016-08-03 20:51:05 +00:00
|
|
|
|
|
|
|
using type = T;
|
|
|
|
|
|
|
|
static inline u64 get(const T& arg)
|
2015-01-18 22:54:56 +00:00
|
|
|
{
|
2020-12-12 13:29:55 +00:00
|
|
|
return reinterpret_cast<uptr>(&arg);
|
2016-04-25 10:49:12 +00:00
|
|
|
}
|
2016-08-07 13:59:46 +00:00
|
|
|
|
|
|
|
// Temporary value container (can possibly be created by other fmt_unveil<> specializations)
|
|
|
|
struct u64_wrapper
|
|
|
|
{
|
|
|
|
T arg;
|
|
|
|
|
|
|
|
// Allow implicit conversion
|
|
|
|
operator u64() const
|
|
|
|
{
|
2020-12-12 13:29:55 +00:00
|
|
|
return reinterpret_cast<uptr>(&arg);
|
2016-08-07 13:59:46 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// This overload resolution takes the precedence
|
|
|
|
static inline u64_wrapper get(T&& arg)
|
|
|
|
{
|
2016-08-16 15:46:24 +00:00
|
|
|
return u64_wrapper{std::move(arg)};
|
2016-08-07 13:59:46 +00:00
|
|
|
}
|
2016-08-03 20:51:05 +00:00
|
|
|
};
|
|
|
|
|
2016-08-13 13:36:04 +00:00
|
|
|
template <typename T>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_unveil<T, std::enable_if_t<std::is_integral<T>::value && sizeof(T) <= 8 && alignof(T) <= 8>>
|
|
|
|
{
|
|
|
|
using type = T;
|
2015-01-18 22:54:56 +00:00
|
|
|
|
2016-08-03 20:51:05 +00:00
|
|
|
static inline u64 get(T arg)
|
2015-01-18 22:54:56 +00:00
|
|
|
{
|
2016-08-03 20:51:05 +00:00
|
|
|
return static_cast<T>(arg);
|
|
|
|
}
|
|
|
|
};
|
2015-01-18 22:54:56 +00:00
|
|
|
|
2016-08-13 13:36:04 +00:00
|
|
|
template <typename T>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_unveil<T, std::enable_if_t<std::is_floating_point<T>::value && sizeof(T) <= 8 && alignof(T) <= 8>>
|
|
|
|
{
|
|
|
|
using type = T;
|
2015-01-18 22:54:56 +00:00
|
|
|
|
2017-05-14 20:29:33 +00:00
|
|
|
// Convert FP to f64 and reinterpret as u64
|
|
|
|
static inline u64 get(const f64& arg)
|
2015-01-18 22:54:56 +00:00
|
|
|
{
|
2019-07-07 08:53:07 +00:00
|
|
|
return std::bit_cast<u64>(arg);
|
2016-08-03 20:51:05 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-08-13 13:36:04 +00:00
|
|
|
template <typename T>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_unveil<T, std::enable_if_t<std::is_enum<T>::value>>
|
|
|
|
{
|
|
|
|
using type = T;
|
|
|
|
|
|
|
|
static inline u64 get(T arg)
|
|
|
|
{
|
|
|
|
return static_cast<std::underlying_type_t<T>>(arg);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-08-13 13:36:04 +00:00
|
|
|
template <typename T>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_unveil<T*, void>
|
|
|
|
{
|
2018-04-21 20:55:01 +00:00
|
|
|
using type = std::add_const_t<T>*;
|
2016-08-03 20:51:05 +00:00
|
|
|
|
2018-04-21 20:55:01 +00:00
|
|
|
static inline u64 get(type arg)
|
2016-08-03 20:51:05 +00:00
|
|
|
{
|
2020-12-12 13:29:55 +00:00
|
|
|
return reinterpret_cast<uptr>(arg);
|
2016-08-03 20:51:05 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-07-17 08:36:27 +00:00
|
|
|
namespace fmt
|
|
|
|
{
|
|
|
|
template <typename T>
|
|
|
|
concept CharT = (std::is_same_v<const T, const char> || std::is_same_v<const T, const char8_t>);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <fmt::CharT T, usz N>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_unveil<T[N], void>
|
|
|
|
{
|
2018-04-21 20:55:01 +00:00
|
|
|
using type = std::add_const_t<T>*;
|
2016-08-03 20:51:05 +00:00
|
|
|
|
2018-04-21 20:55:01 +00:00
|
|
|
static inline u64 get(type arg)
|
2016-08-03 20:51:05 +00:00
|
|
|
{
|
2020-12-12 13:29:55 +00:00
|
|
|
return reinterpret_cast<uptr>(arg);
|
2016-08-03 20:51:05 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-12-18 07:39:54 +00:00
|
|
|
template <typename T, bool Se, usz Align>
|
2020-12-13 13:34:45 +00:00
|
|
|
struct fmt_unveil<se_t<T, Se, Align>, void>
|
|
|
|
{
|
|
|
|
using type = typename fmt_unveil<T>::type;
|
|
|
|
|
|
|
|
static inline auto get(const se_t<T, Se, Align>& arg)
|
|
|
|
{
|
|
|
|
return fmt_unveil<T>::get(arg);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-08-03 20:51:05 +00:00
|
|
|
// String type format provider, also type classifier (format() called if an argument is formatted as "%s")
|
2016-08-13 13:36:04 +00:00
|
|
|
template <typename T, typename = void>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_class_string
|
|
|
|
{
|
|
|
|
// Formatting function (must be explicitly specialized)
|
|
|
|
static void format(std::string& out, u64 arg);
|
|
|
|
|
|
|
|
// Helper typedef (visible in format())
|
|
|
|
using type = T;
|
|
|
|
|
|
|
|
// Helper function (converts arg to object reference)
|
2021-02-09 09:33:50 +00:00
|
|
|
static FORCE_INLINE SAFE_BUFFERS(const T&) get_object(u64 arg)
|
2016-08-03 20:51:05 +00:00
|
|
|
{
|
2020-12-12 13:29:55 +00:00
|
|
|
return *reinterpret_cast<const T*>(static_cast<uptr>(arg));
|
2016-08-03 20:51:05 +00:00
|
|
|
}
|
|
|
|
|
2017-05-14 20:29:33 +00:00
|
|
|
// Enum -> string function type
|
|
|
|
using convert_t = const char*(*)(T value);
|
|
|
|
|
2016-08-03 20:51:05 +00:00
|
|
|
// Helper function (safely converts arg to enum value)
|
2021-02-09 09:33:50 +00:00
|
|
|
static FORCE_INLINE SAFE_BUFFERS(void) format_enum(std::string& out, u64 arg, convert_t convert)
|
2016-08-03 20:51:05 +00:00
|
|
|
{
|
|
|
|
const auto value = static_cast<std::underlying_type_t<T>>(arg);
|
|
|
|
|
|
|
|
// Check narrowing
|
|
|
|
if (static_cast<u64>(value) == arg)
|
|
|
|
{
|
2017-05-14 20:29:33 +00:00
|
|
|
if (const char* str = convert(static_cast<T>(value)))
|
2016-08-03 20:51:05 +00:00
|
|
|
{
|
|
|
|
out += str;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fallback to underlying type formatting
|
|
|
|
fmt_class_string<std::underlying_type_t<T>>::format(out, static_cast<u64>(value));
|
|
|
|
}
|
|
|
|
|
2016-08-07 19:01:27 +00:00
|
|
|
// Helper function (bitset formatting)
|
2021-02-09 09:33:50 +00:00
|
|
|
static FORCE_INLINE SAFE_BUFFERS(void) format_bitset(std::string& out, u64 arg, const char* prefix, const char* delim, const char* suffix, void (*fmt)(std::string&, u64))
|
2016-08-07 19:01:27 +00:00
|
|
|
{
|
|
|
|
// Start from raw value
|
|
|
|
fmt_class_string<u64>::format(out, arg);
|
|
|
|
|
|
|
|
out += prefix;
|
|
|
|
|
2017-02-15 15:07:42 +00:00
|
|
|
for (u64 i = 0; i < 63; i++)
|
2016-08-07 19:01:27 +00:00
|
|
|
{
|
|
|
|
const u64 mask = 1ull << i;
|
|
|
|
|
|
|
|
if (arg & mask)
|
|
|
|
{
|
|
|
|
fmt(out, i);
|
|
|
|
|
2017-02-15 15:07:42 +00:00
|
|
|
if (arg >> (i + 1))
|
2016-08-07 19:01:27 +00:00
|
|
|
{
|
|
|
|
out += delim;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-15 15:07:42 +00:00
|
|
|
if (arg & (1ull << 63))
|
|
|
|
{
|
|
|
|
fmt(out, 63);
|
|
|
|
}
|
|
|
|
|
2016-08-07 19:01:27 +00:00
|
|
|
out += suffix;
|
|
|
|
}
|
|
|
|
|
2016-08-03 20:51:05 +00:00
|
|
|
// Helper constant (may be used in format_enum as lambda return value)
|
|
|
|
static constexpr const char* unknown = nullptr;
|
|
|
|
};
|
|
|
|
|
2016-08-13 13:36:04 +00:00
|
|
|
template <>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_class_string<const void*, void>
|
|
|
|
{
|
|
|
|
static void format(std::string& out, u64 arg);
|
|
|
|
};
|
|
|
|
|
2016-08-13 13:36:04 +00:00
|
|
|
template <typename T>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_class_string<T*, void> : fmt_class_string<const void*, void>
|
|
|
|
{
|
|
|
|
// Classify all pointers as const void*
|
|
|
|
};
|
|
|
|
|
2016-08-13 13:36:04 +00:00
|
|
|
template <>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_class_string<const char*, void>
|
|
|
|
{
|
|
|
|
static void format(std::string& out, u64 arg);
|
|
|
|
};
|
|
|
|
|
2016-08-13 13:36:04 +00:00
|
|
|
template <>
|
2016-08-03 20:51:05 +00:00
|
|
|
struct fmt_class_string<char*, void> : fmt_class_string<const char*>
|
|
|
|
{
|
|
|
|
// Classify char* as const char*
|
|
|
|
};
|
|
|
|
|
2021-07-10 13:03:43 +00:00
|
|
|
template <>
|
|
|
|
struct fmt_class_string<const char8_t*, void> : fmt_class_string<const char*>
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct fmt_class_string<char8_t*, void> : fmt_class_string<const char8_t*>
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
2022-05-07 12:55:33 +00:00
|
|
|
template <>
|
|
|
|
struct fmt_class_string<const wchar_t*, void>
|
|
|
|
{
|
|
|
|
static void format(std::string& out, u64 arg);
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct fmt_class_string<wchar_t*, void> : fmt_class_string<const wchar_t*>
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
2021-07-17 08:36:27 +00:00
|
|
|
namespace fmt
|
|
|
|
{
|
|
|
|
// Both uchar and std::byte are allowed
|
|
|
|
template <typename T>
|
2021-07-17 17:30:21 +00:00
|
|
|
concept ByteArray = requires (T& t) { const_cast<std::conditional_t<std::is_same_v<decltype(std::as_const(t[0])), const std::byte&>, std::byte, uchar>&>(std::data(t)[0]); };
|
2021-07-17 08:36:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template <fmt::ByteArray T>
|
|
|
|
struct fmt_class_string<T, void>
|
|
|
|
{
|
|
|
|
static FORCE_INLINE SAFE_BUFFERS(const T&) get_object(u64 arg)
|
|
|
|
{
|
|
|
|
return *reinterpret_cast<const T*>(static_cast<uptr>(arg));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void format(std::string& out, u64 arg)
|
|
|
|
{
|
|
|
|
const auto& obj = get_object(arg);
|
2021-12-30 16:39:18 +00:00
|
|
|
|
2021-07-17 08:36:27 +00:00
|
|
|
void format_byte_array(std::string&, const uchar*, usz);
|
|
|
|
format_byte_array(out, reinterpret_cast<const uchar*>(std::data(obj)), std::size(obj));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-08-05 16:49:45 +00:00
|
|
|
struct fmt_type_info
|
2016-08-03 20:51:05 +00:00
|
|
|
{
|
2016-08-05 16:49:45 +00:00
|
|
|
decltype(&fmt_class_string<int>::format) fmt_string;
|
2016-08-03 20:51:05 +00:00
|
|
|
|
2016-08-13 13:36:04 +00:00
|
|
|
template <typename T>
|
2016-08-05 16:49:45 +00:00
|
|
|
static constexpr fmt_type_info make()
|
2016-08-03 20:51:05 +00:00
|
|
|
{
|
2016-08-05 16:49:45 +00:00
|
|
|
return fmt_type_info
|
2016-08-03 20:51:05 +00:00
|
|
|
{
|
2016-08-05 16:49:45 +00:00
|
|
|
&fmt_class_string<T>::format,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-08-25 09:06:57 +00:00
|
|
|
// Argument array type (each element generated via fmt_unveil<>)
|
|
|
|
template <typename... Args>
|
|
|
|
using fmt_args_t = const u64(&&)[sizeof...(Args) + 1];
|
|
|
|
|
2020-12-22 11:05:05 +00:00
|
|
|
template <typename Arg>
|
|
|
|
using fmt_unveil_t = typename fmt_unveil<Arg>::type;
|
|
|
|
|
2016-08-05 16:49:45 +00:00
|
|
|
namespace fmt
|
|
|
|
{
|
2018-03-22 19:48:38 +00:00
|
|
|
// Base-57 format helper
|
|
|
|
struct base57
|
|
|
|
{
|
|
|
|
const uchar* data;
|
2020-12-18 07:39:54 +00:00
|
|
|
usz size;
|
2018-03-22 19:48:38 +00:00
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
base57(const T& arg)
|
|
|
|
: data(reinterpret_cast<const uchar*>(&arg))
|
|
|
|
, size(sizeof(T))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-12-18 07:39:54 +00:00
|
|
|
base57(const uchar* data, usz size)
|
2018-03-22 19:48:38 +00:00
|
|
|
: data(data)
|
|
|
|
, size(size)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-08-24 12:09:05 +00:00
|
|
|
template <typename... Args>
|
|
|
|
constexpr const fmt_type_info type_info_v[sizeof...(Args) + 1]{fmt_type_info::make<fmt_unveil_t<Args>>()...};
|
|
|
|
|
2016-08-03 20:51:05 +00:00
|
|
|
// Internal formatting function
|
2016-08-05 16:49:45 +00:00
|
|
|
void raw_append(std::string& out, const char*, const fmt_type_info*, const u64*) noexcept;
|
2016-08-03 20:51:05 +00:00
|
|
|
|
|
|
|
// Formatting function
|
2020-12-18 07:39:54 +00:00
|
|
|
template <typename CharT, usz N, typename... Args>
|
2021-02-09 09:33:50 +00:00
|
|
|
FORCE_INLINE SAFE_BUFFERS(void) append(std::string& out, const CharT(&fmt)[N], const Args&... args)
|
2016-08-03 20:51:05 +00:00
|
|
|
{
|
2021-05-19 07:12:57 +00:00
|
|
|
raw_append(out, reinterpret_cast<const char*>(fmt), type_info_v<Args...>, fmt_args_t<Args...>{fmt_unveil<Args>::get(args)...});
|
2016-08-03 20:51:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Formatting function
|
2020-12-18 07:39:54 +00:00
|
|
|
template <typename CharT, usz N, typename... Args>
|
2021-02-09 09:33:50 +00:00
|
|
|
FORCE_INLINE SAFE_BUFFERS(std::string) format(const CharT(&fmt)[N], const Args&... args)
|
2016-02-01 21:55:43 +00:00
|
|
|
{
|
2016-08-03 20:51:05 +00:00
|
|
|
std::string result;
|
2020-02-04 18:37:00 +00:00
|
|
|
append(result, fmt, args...);
|
2016-02-01 21:55:43 +00:00
|
|
|
return result;
|
|
|
|
}
|
2016-08-03 20:51:05 +00:00
|
|
|
|
2016-08-08 16:01:06 +00:00
|
|
|
// Internal exception message formatting template, must be explicitly specialized or instantiated in cpp to minimize code bloat
|
2020-12-09 15:04:52 +00:00
|
|
|
[[noreturn]] void raw_throw_exception(const src_loc&, const char*, const fmt_type_info*, const u64*);
|
2016-08-03 20:51:05 +00:00
|
|
|
|
2016-08-08 16:01:06 +00:00
|
|
|
// Throw exception with formatting
|
2020-12-18 07:39:54 +00:00
|
|
|
template <typename CharT, usz N, typename... Args>
|
2020-12-09 15:04:52 +00:00
|
|
|
struct throw_exception
|
2016-08-03 20:51:05 +00:00
|
|
|
{
|
2021-02-09 09:33:50 +00:00
|
|
|
[[noreturn]] FORCE_INLINE SAFE_BUFFERS() throw_exception(const CharT(&fmt)[N], const Args&... args,
|
2020-12-09 15:04:52 +00:00
|
|
|
u32 line = __builtin_LINE(),
|
|
|
|
u32 col = __builtin_COLUMN(),
|
|
|
|
const char* file = __builtin_FILE(),
|
|
|
|
const char* func = __builtin_FUNCTION())
|
|
|
|
{
|
2021-05-19 07:12:57 +00:00
|
|
|
raw_throw_exception({line, col, file, func}, reinterpret_cast<const char*>(fmt), type_info_v<Args...>, fmt_args_t<Args...>{fmt_unveil<Args>::get(args)...});
|
2020-12-09 15:04:52 +00:00
|
|
|
}
|
2021-01-06 20:33:09 +00:00
|
|
|
|
2021-01-18 17:30:29 +00:00
|
|
|
#ifndef _MSC_VER
|
2021-01-18 10:49:16 +00:00
|
|
|
[[noreturn]] ~throw_exception();
|
|
|
|
#endif
|
2020-12-09 15:04:52 +00:00
|
|
|
};
|
|
|
|
|
2020-12-18 07:39:54 +00:00
|
|
|
template <typename CharT, usz N, typename... Args>
|
2020-12-09 15:04:52 +00:00
|
|
|
throw_exception(const CharT(&)[N], const Args&...) -> throw_exception<CharT, N, Args...>;
|
2021-06-09 16:05:25 +00:00
|
|
|
|
|
|
|
// Helper template: pack format variables
|
|
|
|
template <typename Arg = void, typename... Args>
|
|
|
|
struct tie
|
|
|
|
{
|
|
|
|
// Universal reference
|
|
|
|
std::add_rvalue_reference_t<Arg> arg;
|
|
|
|
|
|
|
|
tie<Args...> next;
|
|
|
|
|
|
|
|
// Store only references, unveil op is postponed
|
|
|
|
tie(Arg&& arg, Args&&... args) noexcept
|
|
|
|
: arg(std::forward<Arg>(arg))
|
|
|
|
, next(std::forward<Args>(args)...)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
using type = std::remove_cvref_t<Arg>;
|
|
|
|
|
|
|
|
// Storage for fmt_unveil (deferred initialization)
|
|
|
|
decltype(fmt_unveil<type>::get(std::declval<Arg>())) value;
|
|
|
|
|
|
|
|
void init(u64 to[])
|
|
|
|
{
|
|
|
|
value = fmt_unveil<type>::get(arg);
|
|
|
|
to[0] = value;
|
|
|
|
next.init(to + 1);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct tie<void>
|
|
|
|
{
|
|
|
|
void init(u64 to[]) const
|
|
|
|
{
|
|
|
|
// Isn't really null terminated, this value has no meaning
|
|
|
|
to[0] = 0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename... Args>
|
|
|
|
tie(Args&&... args) -> tie<Args...>;
|
|
|
|
|
|
|
|
// Ensure with formatting
|
|
|
|
template <typename T, typename CharT, usz N, typename... Args>
|
|
|
|
decltype(auto) ensure(T&& arg, const CharT(&fmt)[N], tie<Args...> args,
|
|
|
|
u32 line = __builtin_LINE(),
|
|
|
|
u32 col = __builtin_COLUMN(),
|
|
|
|
const char* file = __builtin_FILE(),
|
|
|
|
const char* func = __builtin_FUNCTION()) noexcept
|
|
|
|
{
|
|
|
|
if (std::forward<T>(arg)) [[likely]]
|
|
|
|
{
|
|
|
|
return std::forward<T>(arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prepare u64 array
|
|
|
|
u64 data[sizeof...(Args) + 1];
|
|
|
|
args.init(data);
|
|
|
|
|
|
|
|
raw_throw_exception({line, col, file, func}, reinterpret_cast<const char*>(fmt), type_info_v<std::remove_cvref_t<Args>...>, +data);
|
|
|
|
}
|
2014-04-01 00:33:55 +00:00
|
|
|
}
|