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
|
|
|
|
2016-04-30 16:34:51 +01:00
|
|
|
#if !defined(ARCHITECTURE_x86_64) && !defined(_M_ARM)
|
|
|
|
#include <cstdlib> // for exit
|
|
|
|
#endif
|
|
|
|
|
2014-12-04 20:57:00 +00:00
|
|
|
#include "common_types.h"
|
2013-09-05 01:17:46 +01:00
|
|
|
|
2013-09-06 04:04:04 +01:00
|
|
|
#define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
|
|
|
|
|
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)
|
|
|
|
#define DO_CONCAT2(x, y) x ## y
|
|
|
|
|
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
|
2015-08-12 03:45:15 +01:00
|
|
|
#define FORCE_INLINE __forceinline
|
2015-05-07 03:26:19 +01:00
|
|
|
#else
|
2015-08-12 03:45:15 +01:00
|
|
|
#define FORCE_INLINE inline __attribute__((always_inline))
|
2015-05-07 03:26:19 +01:00
|
|
|
#endif
|
|
|
|
|
2014-11-29 05:38:20 +00:00
|
|
|
#ifndef _MSC_VER
|
2013-09-05 01:17:46 +01:00
|
|
|
|
2015-08-15 03:29:08 +01:00
|
|
|
#ifdef ARCHITECTURE_x86_64
|
2015-01-21 01:16:47 +00:00
|
|
|
#define Crash() __asm__ __volatile__("int $3")
|
|
|
|
#elif defined(_M_ARM)
|
|
|
|
#define Crash() __asm__ __volatile__("trap")
|
|
|
|
#else
|
|
|
|
#define Crash() exit(1)
|
|
|
|
#endif
|
|
|
|
|
2013-09-05 01:17:46 +01:00
|
|
|
// GCC 4.8 defines all the rotate functions now
|
|
|
|
// Small issue with GCC's lrotl/lrotr intrinsics is they are still 32bit while we require 64bit
|
2015-08-16 07:41:40 +01:00
|
|
|
#ifdef _rotl
|
|
|
|
#define rotl _rotl
|
|
|
|
#else
|
|
|
|
inline u32 rotl(u32 x, int shift) {
|
2014-04-01 23:20:08 +01:00
|
|
|
shift &= 31;
|
|
|
|
if (!shift) return x;
|
|
|
|
return (x << shift) | (x >> (32 - shift));
|
2013-09-05 01:17:46 +01:00
|
|
|
}
|
2015-08-16 07:41:40 +01:00
|
|
|
#endif
|
2013-09-05 01:17:46 +01:00
|
|
|
|
2015-08-16 07:41:40 +01:00
|
|
|
#ifdef _rotr
|
|
|
|
#define rotr _rotr
|
|
|
|
#else
|
|
|
|
inline u32 rotr(u32 x, int shift) {
|
2014-04-01 23:20:08 +01:00
|
|
|
shift &= 31;
|
|
|
|
if (!shift) return x;
|
|
|
|
return (x >> shift) | (x << (32 - shift));
|
2013-09-05 01:17:46 +01:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
inline u64 _rotl64(u64 x, unsigned int shift){
|
2014-04-01 23:20:08 +01:00
|
|
|
unsigned int n = shift % 64;
|
|
|
|
return (x << n) | (x >> (64 - n));
|
2013-09-05 01:17:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
inline u64 _rotr64(u64 x, unsigned int shift){
|
2014-04-01 23:20:08 +01:00
|
|
|
unsigned int n = shift % 64;
|
|
|
|
return (x >> n) | (x << (64 - n));
|
2013-09-05 01:17:46 +01:00
|
|
|
}
|
|
|
|
|
2014-11-29 05:38:20 +00:00
|
|
|
#else // _MSC_VER
|
2016-05-27 10:40:01 +01:00
|
|
|
|
|
|
|
#if (_MSC_VER < 1900)
|
|
|
|
// Function Cross-Compatibility
|
|
|
|
#define snprintf _snprintf
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Locale Cross-Compatibility
|
|
|
|
#define locale_t _locale_t
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
__declspec(dllimport) void __stdcall DebugBreak(void);
|
|
|
|
}
|
|
|
|
#define Crash() {DebugBreak();}
|
|
|
|
|
|
|
|
// cstdlib provides these on MSVC
|
|
|
|
#define rotr _rotr
|
|
|
|
#define rotl _rotl
|
|
|
|
|
2014-11-29 05:38:20 +00:00
|
|
|
#endif // _MSC_VER ndef
|
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.
|
|
|
|
const char* GetLastErrorMsg();
|