2024-03-07 09:36:29 +00:00
|
|
|
// SPDX-FileCopyrightText: Copyright 2019 yuzu Emulator Project & 2024 suyu Emulator Project
|
2022-04-23 08:59:50 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2013-09-05 00:17:46 +00:00
|
|
|
|
2014-08-17 17:45:50 +00:00
|
|
|
#pragma once
|
2013-09-05 00:17:46 +00:00
|
|
|
|
2019-11-03 23:54:03 +00:00
|
|
|
#include <array>
|
2021-09-08 18:47:01 +00:00
|
|
|
#include <iterator>
|
2018-07-19 13:03:30 +00:00
|
|
|
|
2018-05-13 10:34:45 +00:00
|
|
|
#if !defined(ARCHITECTURE_x86_64)
|
2016-04-30 15:34:51 +00:00
|
|
|
#include <cstdlib> // for exit
|
|
|
|
#endif
|
2017-05-27 23:14:10 +00:00
|
|
|
#include "common/common_types.h"
|
2013-09-05 00:17:46 +00: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 00:38:01 +00:00
|
|
|
#define DO_CONCAT2(x, y) x##y
|
2015-01-11 15:32:31 +00:00
|
|
|
|
2019-11-03 23:54:03 +00:00
|
|
|
/// Helper macros to insert unused bytes or words to properly align structs. These values will be
|
|
|
|
/// zero-initialized.
|
2019-11-14 21:54:01 +00:00
|
|
|
#define INSERT_PADDING_BYTES(num_bytes) \
|
2022-07-15 09:34:38 +00:00
|
|
|
[[maybe_unused]] std::array<u8, num_bytes> CONCAT2(pad, __LINE__) {}
|
2019-11-14 21:54:01 +00:00
|
|
|
#define INSERT_PADDING_WORDS(num_words) \
|
2022-07-15 09:34:38 +00:00
|
|
|
[[maybe_unused]] std::array<u32, num_words> CONCAT2(pad, __LINE__) {}
|
2015-02-13 04:57:02 +00:00
|
|
|
|
2021-01-15 07:25:40 +00:00
|
|
|
/// These are similar to the INSERT_PADDING_* macros but do not zero-initialize the contents.
|
|
|
|
/// This keeps the structure trivial to construct.
|
2022-07-15 09:34:38 +00:00
|
|
|
#define INSERT_PADDING_BYTES_NOINIT(num_bytes) \
|
|
|
|
[[maybe_unused]] std::array<u8, num_bytes> CONCAT2(pad, __LINE__)
|
|
|
|
#define INSERT_PADDING_WORDS_NOINIT(num_words) \
|
|
|
|
[[maybe_unused]] std::array<u32, num_words> CONCAT2(pad, __LINE__)
|
2015-05-07 02:26:19 +00:00
|
|
|
|
2014-11-29 05:38:20 +00:00
|
|
|
#ifndef _MSC_VER
|
2013-09-05 00:17:46 +00:00
|
|
|
|
2022-11-06 21:45:36 +00:00
|
|
|
#if defined(ARCHITECTURE_x86_64)
|
2015-01-21 01:16:47 +00:00
|
|
|
#define Crash() __asm__ __volatile__("int $3")
|
2022-11-06 21:45:36 +00:00
|
|
|
#elif defined(ARCHITECTURE_arm64)
|
|
|
|
#define Crash() __asm__ __volatile__("brk #0")
|
2015-01-21 01:16:47 +00:00
|
|
|
#else
|
|
|
|
#define Crash() exit(1)
|
|
|
|
#endif
|
|
|
|
|
2023-10-12 13:16:22 +00:00
|
|
|
#define LTO_NOINLINE __attribute__((noinline))
|
|
|
|
|
2014-11-29 05:38:20 +00:00
|
|
|
#else // _MSC_VER
|
2016-05-27 09:40:01 +00:00
|
|
|
|
2023-10-12 13:16:22 +00:00
|
|
|
#define LTO_NOINLINE
|
|
|
|
|
2016-05-27 09:40:01 +00:00
|
|
|
// Locale Cross-Compatibility
|
|
|
|
#define locale_t _locale_t
|
|
|
|
|
|
|
|
extern "C" {
|
2016-09-18 00:38:01 +00:00
|
|
|
__declspec(dllimport) void __stdcall DebugBreak(void);
|
2016-05-27 09:40:01 +00:00
|
|
|
}
|
2016-09-19 01:01:46 +00:00
|
|
|
#define Crash() DebugBreak()
|
2016-05-27 09:40:01 +00:00
|
|
|
|
2014-11-29 05:38:20 +00:00
|
|
|
#endif // _MSC_VER ndef
|
2013-09-05 00:17:46 +00:00
|
|
|
|
2020-03-21 02:40:03 +00:00
|
|
|
#define DECLARE_ENUM_FLAG_OPERATORS(type) \
|
2020-08-14 13:38:45 +00:00
|
|
|
[[nodiscard]] constexpr type operator|(type a, type b) noexcept { \
|
2020-03-21 02:40:03 +00:00
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<type>(static_cast<T>(a) | static_cast<T>(b)); \
|
|
|
|
} \
|
2020-08-14 13:38:45 +00:00
|
|
|
[[nodiscard]] constexpr type operator&(type a, type b) noexcept { \
|
2020-03-21 02:40:03 +00:00
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<type>(static_cast<T>(a) & static_cast<T>(b)); \
|
|
|
|
} \
|
2020-08-24 08:39:26 +00:00
|
|
|
[[nodiscard]] constexpr type operator^(type a, type b) noexcept { \
|
2020-03-21 02:40:03 +00:00
|
|
|
using T = std::underlying_type_t<type>; \
|
2020-08-24 08:39:26 +00:00
|
|
|
return static_cast<type>(static_cast<T>(a) ^ static_cast<T>(b)); \
|
|
|
|
} \
|
2021-09-13 20:01:20 +00:00
|
|
|
[[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)); \
|
|
|
|
} \
|
2020-08-24 08:39:26 +00:00
|
|
|
constexpr type& operator|=(type& a, type b) noexcept { \
|
|
|
|
a = a | b; \
|
2020-03-21 02:40:03 +00:00
|
|
|
return a; \
|
|
|
|
} \
|
|
|
|
constexpr type& operator&=(type& a, type b) noexcept { \
|
2020-08-24 08:39:26 +00:00
|
|
|
a = a & b; \
|
|
|
|
return a; \
|
|
|
|
} \
|
|
|
|
constexpr type& operator^=(type& a, type b) noexcept { \
|
|
|
|
a = a ^ b; \
|
2020-03-21 02:40:03 +00:00
|
|
|
return a; \
|
|
|
|
} \
|
2021-09-13 20:01:20 +00:00
|
|
|
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; \
|
|
|
|
} \
|
2020-08-14 13:38:45 +00:00
|
|
|
[[nodiscard]] constexpr type operator~(type key) noexcept { \
|
2020-03-21 02:40:03 +00:00
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<type>(~static_cast<T>(key)); \
|
|
|
|
} \
|
2020-08-14 13:38:45 +00:00
|
|
|
[[nodiscard]] constexpr bool True(type key) noexcept { \
|
2020-03-21 02:40:03 +00:00
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<T>(key) != 0; \
|
|
|
|
} \
|
2020-08-14 13:38:45 +00:00
|
|
|
[[nodiscard]] constexpr bool False(type key) noexcept { \
|
2020-03-21 02:40:03 +00:00
|
|
|
using T = std::underlying_type_t<type>; \
|
|
|
|
return static_cast<T>(key) == 0; \
|
|
|
|
}
|
|
|
|
|
2024-03-07 09:36:29 +00:00
|
|
|
#define SUYU_NON_COPYABLE(cls) \
|
2021-03-31 21:19:26 +00:00
|
|
|
cls(const cls&) = delete; \
|
|
|
|
cls& operator=(const cls&) = delete
|
|
|
|
|
2024-03-07 09:36:29 +00:00
|
|
|
#define SUYU_NON_MOVEABLE(cls) \
|
2021-03-31 21:19:26 +00:00
|
|
|
cls(cls&&) = delete; \
|
|
|
|
cls& operator=(cls&&) = delete
|
|
|
|
|
2017-10-15 04:11:38 +00:00
|
|
|
namespace Common {
|
|
|
|
|
2020-08-14 13:38:45 +00:00
|
|
|
[[nodiscard]] constexpr u32 MakeMagic(char a, char b, char c, char d) {
|
2019-11-13 11:50:49 +00:00
|
|
|
return u32(a) | u32(b) << 8 | u32(c) << 16 | u32(d) << 24;
|
2017-10-15 04:11:38 +00:00
|
|
|
}
|
|
|
|
|
2023-11-26 07:36:08 +00:00
|
|
|
[[nodiscard]] constexpr u64 MakeMagic(char a, char b, char c, char d, char e, char f, char g,
|
|
|
|
char h) {
|
|
|
|
return u64(a) << 0 | u64(b) << 8 | u64(c) << 16 | u64(d) << 24 | u64(e) << 32 | u64(f) << 40 |
|
|
|
|
u64(g) << 48 | u64(h) << 56;
|
|
|
|
}
|
|
|
|
|
2021-04-02 06:04:54 +00:00
|
|
|
// std::size() does not support zero-size C arrays. We're fixing that.
|
|
|
|
template <class C>
|
|
|
|
constexpr auto Size(const C& c) -> decltype(c.size()) {
|
|
|
|
return std::size(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class C>
|
|
|
|
constexpr std::size_t Size(const C& c) {
|
|
|
|
if constexpr (sizeof(C) == 0) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return std::size(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-15 04:11:38 +00:00
|
|
|
} // namespace Common
|