2014-12-17 05:38:14 +00:00
|
|
|
// Copyright 2013 Dolphin Emulator Project / 2014 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
2013-09-05 01:17:46 +01:00
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2014-08-17 18:45:50 +01:00
|
|
|
#pragma once
|
2013-09-05 01:17:46 +01:00
|
|
|
|
2018-07-19 14:03:30 +01:00
|
|
|
#include <string>
|
2017-05-28 00:14:10 +01:00
|
|
|
#include "common/common_types.h"
|
2013-09-05 01:17:46 +01:00
|
|
|
|
2015-01-11 15:32:31 +00:00
|
|
|
/// Textually concatenates two tokens. The double-expansion is required by the C preprocessor.
|
|
|
|
#define CONCAT2(x, y) DO_CONCAT2(x, y)
|
2016-09-18 01:38:01 +01:00
|
|
|
#define DO_CONCAT2(x, y) x##y
|
2015-01-11 15:32:31 +00:00
|
|
|
|
2015-02-13 04:57:02 +00:00
|
|
|
// helper macro to properly align structure members.
|
|
|
|
// Calling INSERT_PADDING_BYTES will add a new member variable with a name like "pad121",
|
|
|
|
// depending on the current source line to make sure variable names are unique.
|
2015-02-19 06:45:46 +00:00
|
|
|
#define INSERT_PADDING_BYTES(num_bytes) u8 CONCAT2(pad, __LINE__)[(num_bytes)]
|
|
|
|
#define INSERT_PADDING_WORDS(num_words) u32 CONCAT2(pad, __LINE__)[(num_words)]
|
2015-02-13 04:57:02 +00:00
|
|
|
|
2016-03-09 06:28:26 +00:00
|
|
|
// Inlining
|
2015-05-07 03:26:19 +01:00
|
|
|
#ifdef _WIN32
|
2016-09-18 01:38:01 +01:00
|
|
|
#define FORCE_INLINE __forceinline
|
2015-05-07 03:26:19 +01:00
|
|
|
#else
|
2016-09-18 01:38:01 +01:00
|
|
|
#define FORCE_INLINE inline __attribute__((always_inline))
|
2015-05-07 03:26:19 +01:00
|
|
|
#endif
|
|
|
|
|
2022-06-14 12:18:12 +01:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#define CITRA_NO_INLINE __declspec(noinline)
|
|
|
|
#else
|
|
|
|
#define CITRA_NO_INLINE __attribute__((noinline))
|
|
|
|
#endif
|
|
|
|
|
2023-05-01 20:38:58 +01:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#define CITRA_NO_RETURN __declspec(noreturn)
|
|
|
|
#else
|
|
|
|
#define CITRA_NO_RETURN __attribute__((noreturn))
|
|
|
|
#endif
|
|
|
|
|
2023-03-18 14:55:39 +00:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
extern "C" {
|
|
|
|
__declspec(dllimport) void __stdcall DebugBreak(void);
|
|
|
|
}
|
|
|
|
#define Crash() DebugBreak()
|
2015-01-21 01:16:47 +00:00
|
|
|
#else
|
2023-03-18 14:55:39 +00:00
|
|
|
#define Crash() __builtin_trap()
|
2015-01-21 01:16:47 +00:00
|
|
|
#endif
|
|
|
|
|
2023-03-18 14:55:39 +00:00
|
|
|
#ifdef _MSC_VER
|
2016-05-27 10:40:01 +01:00
|
|
|
#if (_MSC_VER < 1900)
|
2016-09-18 01:38:01 +01:00
|
|
|
// Function Cross-Compatibility
|
|
|
|
#define snprintf _snprintf
|
2016-05-27 10:40:01 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// Locale Cross-Compatibility
|
|
|
|
#define locale_t _locale_t
|
|
|
|
|
2023-03-18 14:55:39 +00:00
|
|
|
#endif // _MSC_VER
|
2013-09-05 01:17:46 +01:00
|
|
|
|
2023-03-27 12:29:17 +01:00
|
|
|
#define DECLARE_ENUM_FLAG_OPERATORS(type) \
|
|
|
|
[[nodiscard]] constexpr type operator|(type a, type b) noexcept { \
|
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<type>(static_cast<T>(a) | static_cast<T>(b)); \
|
|
|
|
} \
|
|
|
|
[[nodiscard]] constexpr type operator&(type a, type b) noexcept { \
|
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<type>(static_cast<T>(a) & static_cast<T>(b)); \
|
|
|
|
} \
|
|
|
|
[[nodiscard]] constexpr type operator^(type a, type b) noexcept { \
|
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<type>(static_cast<T>(a) ^ static_cast<T>(b)); \
|
|
|
|
} \
|
|
|
|
[[nodiscard]] constexpr type operator<<(type a, type b) noexcept { \
|
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<type>(static_cast<T>(a) << static_cast<T>(b)); \
|
|
|
|
} \
|
|
|
|
[[nodiscard]] constexpr type operator>>(type a, type b) noexcept { \
|
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<type>(static_cast<T>(a) >> static_cast<T>(b)); \
|
|
|
|
} \
|
|
|
|
constexpr type& operator|=(type& a, type b) noexcept { \
|
|
|
|
a = a | b; \
|
|
|
|
return a; \
|
|
|
|
} \
|
|
|
|
constexpr type& operator&=(type& a, type b) noexcept { \
|
|
|
|
a = a & b; \
|
|
|
|
return a; \
|
|
|
|
} \
|
|
|
|
constexpr type& operator^=(type& a, type b) noexcept { \
|
|
|
|
a = a ^ b; \
|
|
|
|
return a; \
|
|
|
|
} \
|
|
|
|
constexpr type& operator<<=(type& a, type b) noexcept { \
|
|
|
|
a = a << b; \
|
|
|
|
return a; \
|
|
|
|
} \
|
|
|
|
constexpr type& operator>>=(type& a, type b) noexcept { \
|
|
|
|
a = a >> b; \
|
|
|
|
return a; \
|
|
|
|
} \
|
|
|
|
[[nodiscard]] constexpr type operator~(type key) noexcept { \
|
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<type>(~static_cast<T>(key)); \
|
|
|
|
} \
|
|
|
|
[[nodiscard]] constexpr bool True(type key) noexcept { \
|
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<T>(key) != 0; \
|
|
|
|
} \
|
|
|
|
[[nodiscard]] constexpr bool False(type key) noexcept { \
|
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<T>(key) == 0; \
|
|
|
|
}
|
|
|
|
|
2013-09-05 01:17:46 +01:00
|
|
|
// Generic function to get last error message.
|
|
|
|
// Call directly after the command or use the error num.
|
|
|
|
// This function might change the error code.
|
|
|
|
// Defined in Misc.cpp.
|
2020-08-31 20:06:16 +01:00
|
|
|
[[nodiscard]] std::string GetLastErrorMsg();
|