2013-04-18 03:09:55 +00:00
|
|
|
// Copyright 2013 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2
|
|
|
|
// Refer to the license.txt file included.
|
2009-03-07 08:35:01 +00:00
|
|
|
|
2014-02-10 18:54:46 +00:00
|
|
|
#pragma once
|
2009-03-07 08:35:01 +00:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
#define SLEEP(x) Sleep(x)
|
|
|
|
#else
|
2012-04-07 19:45:12 +00:00
|
|
|
#include <unistd.h>
|
2009-03-07 08:35:01 +00:00
|
|
|
#define SLEEP(x) usleep(x*1000)
|
|
|
|
#endif
|
|
|
|
|
2014-08-08 17:26:26 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#include <libkern/OSByteOrder.h>
|
|
|
|
#endif
|
|
|
|
|
2014-02-19 18:46:47 +00:00
|
|
|
#include <clocale>
|
2013-09-12 00:19:36 +00:00
|
|
|
#include <cstddef>
|
2013-09-12 17:57:13 +00:00
|
|
|
#include <type_traits>
|
2014-02-19 18:46:47 +00:00
|
|
|
#include "Common/CommonTypes.h"
|
2013-09-12 00:19:36 +00:00
|
|
|
|
|
|
|
// Will fail to compile on a non-array:
|
2013-09-12 17:57:13 +00:00
|
|
|
// TODO: make this a function when constexpr is available
|
|
|
|
template <typename T>
|
|
|
|
struct ArraySizeImpl : public std::extent<T>
|
|
|
|
{ static_assert(std::is_array<T>::value, "is array"); };
|
|
|
|
|
|
|
|
#define ArraySize(x) ArraySizeImpl<decltype(x)>::value
|
2013-09-12 00:19:36 +00:00
|
|
|
|
2011-11-20 23:50:33 +00:00
|
|
|
#define b2(x) ( (x) | ( (x) >> 1) )
|
|
|
|
#define b4(x) ( b2(x) | ( b2(x) >> 2) )
|
|
|
|
#define b8(x) ( b4(x) | ( b4(x) >> 4) )
|
2013-10-29 05:23:17 +00:00
|
|
|
#define b16(x) ( b8(x) | ( b8(x) >> 8) )
|
2011-11-20 23:50:33 +00:00
|
|
|
#define b32(x) (b16(x) | (b16(x) >>16) )
|
2014-02-09 23:29:13 +00:00
|
|
|
#define ROUND_UP_POW2(x) (b32(x - 1) + 1)
|
2013-10-27 00:21:00 +00:00
|
|
|
|
2014-03-29 10:05:44 +00:00
|
|
|
#ifndef __GNUC_PREREQ
|
|
|
|
#define __GNUC_PREREQ(a, b) 0
|
2013-11-12 22:34:32 +00:00
|
|
|
#endif
|
|
|
|
|
2014-03-10 11:30:55 +00:00
|
|
|
#if (defined __GNUC__ && !__GNUC_PREREQ(4,9)) && \
|
2014-06-08 01:24:02 +00:00
|
|
|
!defined __SSSE3__ && defined _M_X86
|
2010-07-31 14:40:01 +00:00
|
|
|
#include <emmintrin.h>
|
|
|
|
static __inline __m128i __attribute__((__always_inline__))
|
|
|
|
_mm_shuffle_epi8(__m128i a, __m128i mask)
|
|
|
|
{
|
|
|
|
__m128i result;
|
|
|
|
__asm__("pshufb %1, %0"
|
|
|
|
: "=x" (result)
|
|
|
|
: "xm" (mask), "0" (a));
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-03-07 08:35:01 +00:00
|
|
|
#ifndef _WIN32
|
|
|
|
|
2010-07-22 03:29:35 +00:00
|
|
|
#include <errno.h>
|
2009-05-01 15:17:03 +00:00
|
|
|
#ifdef __linux__
|
2009-03-07 08:35:01 +00:00
|
|
|
#include <byteswap.h>
|
2011-08-27 18:42:11 +00:00
|
|
|
#elif defined __FreeBSD__
|
|
|
|
#include <sys/endian.h>
|
2009-05-01 15:17:03 +00:00
|
|
|
#endif
|
2010-07-22 03:29:35 +00:00
|
|
|
|
2009-03-07 08:35:01 +00:00
|
|
|
// go to debugger mode
|
2009-05-03 11:15:17 +00:00
|
|
|
#ifdef GEKKO
|
|
|
|
#define Crash()
|
2014-06-08 01:24:02 +00:00
|
|
|
#elif defined _M_X86
|
2009-05-03 11:15:17 +00:00
|
|
|
#define Crash() {asm ("int $3");}
|
2014-06-08 01:24:02 +00:00
|
|
|
#else
|
|
|
|
#define Crash() { exit(1); }
|
2009-05-03 11:15:17 +00:00
|
|
|
#endif
|
2013-09-12 00:19:36 +00:00
|
|
|
|
2013-04-03 15:52:06 +00:00
|
|
|
// GCC 4.8 defines all the rotate functions now
|
2013-04-03 17:42:58 +00:00
|
|
|
// Small issue with GCC's lrotl/lrotr intrinsics is they are still 32bit while we require 64bit
|
|
|
|
#ifndef _rotl
|
2014-08-30 20:14:56 +00:00
|
|
|
inline u32 _rotl(u32 x, int shift)
|
|
|
|
{
|
2013-03-20 01:51:12 +00:00
|
|
|
shift &= 31;
|
|
|
|
if (!shift) return x;
|
|
|
|
return (x << shift) | (x >> (32 - shift));
|
2009-03-07 08:35:01 +00:00
|
|
|
}
|
|
|
|
|
2014-08-30 20:14:56 +00:00
|
|
|
inline u32 _rotr(u32 x, int shift)
|
|
|
|
{
|
2013-03-20 01:51:12 +00:00
|
|
|
shift &= 31;
|
|
|
|
if (!shift) return x;
|
|
|
|
return (x >> shift) | (x << (32 - shift));
|
2009-03-07 08:35:01 +00:00
|
|
|
}
|
2013-04-03 17:42:58 +00:00
|
|
|
#endif
|
|
|
|
|
2014-08-30 20:14:56 +00:00
|
|
|
inline u64 _rotl64(u64 x, unsigned int shift)
|
|
|
|
{
|
2013-04-03 17:42:58 +00:00
|
|
|
unsigned int n = shift % 64;
|
|
|
|
return (x << n) | (x >> (64 - n));
|
|
|
|
}
|
2010-02-08 23:23:04 +00:00
|
|
|
|
2014-08-30 20:14:56 +00:00
|
|
|
inline u64 _rotr64(u64 x, unsigned int shift)
|
|
|
|
{
|
2010-02-08 23:23:04 +00:00
|
|
|
unsigned int n = shift % 64;
|
|
|
|
return (x >> n) | (x << (64 - n));
|
|
|
|
}
|
2010-07-31 14:40:01 +00:00
|
|
|
|
2009-03-07 08:35:01 +00:00
|
|
|
#else // WIN32
|
|
|
|
// Function Cross-Compatibility
|
|
|
|
#define strcasecmp _stricmp
|
|
|
|
#define strncasecmp _strnicmp
|
|
|
|
#define unlink _unlink
|
|
|
|
#define snprintf _snprintf
|
2009-11-08 08:54:09 +00:00
|
|
|
#define vscprintf _vscprintf
|
2013-10-29 05:23:17 +00:00
|
|
|
|
2013-06-15 11:21:57 +00:00
|
|
|
// Locale Cross-Compatibility
|
|
|
|
#define locale_t _locale_t
|
|
|
|
#define freelocale _free_locale
|
|
|
|
#define newlocale(mask, locale, base) _create_locale(mask, locale)
|
2013-10-29 05:23:17 +00:00
|
|
|
|
2014-02-17 04:51:41 +00:00
|
|
|
#define LC_GLOBAL_LOCALE ((locale_t)-1)
|
|
|
|
#define LC_ALL_MASK LC_ALL
|
|
|
|
#define LC_COLLATE_MASK LC_COLLATE
|
|
|
|
#define LC_CTYPE_MASK LC_CTYPE
|
|
|
|
#define LC_MONETARY_MASK LC_MONETARY
|
|
|
|
#define LC_NUMERIC_MASK LC_NUMERIC
|
|
|
|
#define LC_TIME_MASK LC_TIME
|
2013-10-29 05:23:17 +00:00
|
|
|
|
2013-06-15 11:21:57 +00:00
|
|
|
inline locale_t uselocale(locale_t new_locale)
|
|
|
|
{
|
|
|
|
// Retrieve the current per thread locale setting
|
|
|
|
bool bIsPerThread = (_configthreadlocale(0) == _ENABLE_PER_THREAD_LOCALE);
|
|
|
|
|
|
|
|
// Retrieve the current thread-specific locale
|
|
|
|
locale_t old_locale = bIsPerThread ? _get_current_locale() : LC_GLOBAL_LOCALE;
|
|
|
|
|
2014-03-10 11:30:55 +00:00
|
|
|
if (new_locale == LC_GLOBAL_LOCALE)
|
2013-06-15 11:21:57 +00:00
|
|
|
{
|
|
|
|
// Restore the global locale
|
|
|
|
_configthreadlocale(_DISABLE_PER_THREAD_LOCALE);
|
|
|
|
}
|
2014-03-10 11:30:55 +00:00
|
|
|
else if (new_locale != nullptr)
|
2013-06-15 11:21:57 +00:00
|
|
|
{
|
|
|
|
// Configure the thread to set the locale only for this thread
|
|
|
|
_configthreadlocale(_ENABLE_PER_THREAD_LOCALE);
|
|
|
|
|
|
|
|
// Set all locale categories
|
2014-03-10 11:30:55 +00:00
|
|
|
for (int i = LC_MIN; i <= LC_MAX; i++)
|
2013-06-15 11:21:57 +00:00
|
|
|
setlocale(i, new_locale->locinfo->lc_category[i].locale);
|
|
|
|
}
|
|
|
|
|
|
|
|
return old_locale;
|
|
|
|
}
|
2009-03-07 08:35:01 +00:00
|
|
|
|
|
|
|
// 64 bit offsets for windows
|
2010-12-04 03:50:55 +00:00
|
|
|
#define fseeko _fseeki64
|
|
|
|
#define ftello _ftelli64
|
2009-03-07 08:35:01 +00:00
|
|
|
#define atoll _atoi64
|
|
|
|
#define stat64 _stat64
|
2010-12-03 12:42:01 +00:00
|
|
|
#define fstat64 _fstat64
|
|
|
|
#define fileno _fileno
|
2009-03-07 08:35:01 +00:00
|
|
|
|
2014-08-03 18:42:06 +00:00
|
|
|
extern "C"
|
|
|
|
{
|
2009-03-07 08:35:01 +00:00
|
|
|
__declspec(dllimport) void __stdcall DebugBreak(void);
|
|
|
|
}
|
2014-08-03 18:42:06 +00:00
|
|
|
#define Crash() {DebugBreak();}
|
2014-08-31 12:52:21 +00:00
|
|
|
|
|
|
|
#if (_MSC_VER > 1800)
|
|
|
|
#error alignof compat can be removed
|
|
|
|
#else
|
|
|
|
#define alignof(x) __alignof(x)
|
|
|
|
#endif
|
2009-03-07 08:35:01 +00:00
|
|
|
#endif // WIN32 ndef
|
|
|
|
|
|
|
|
// 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();
|
|
|
|
|
|
|
|
namespace Common
|
|
|
|
{
|
|
|
|
inline u8 swap8(u8 _data) {return _data;}
|
2013-05-31 05:12:59 +00:00
|
|
|
inline u32 swap24(const u8* _data) {return (_data[0] << 16) | (_data[1] << 8) | _data[2];}
|
2009-03-07 08:35:01 +00:00
|
|
|
|
2013-02-26 19:49:00 +00:00
|
|
|
#ifdef ANDROID
|
|
|
|
#undef swap16
|
|
|
|
#undef swap32
|
|
|
|
#undef swap64
|
|
|
|
#endif
|
2013-07-11 13:58:10 +00:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
inline u16 swap16(u16 _data) {return _byteswap_ushort(_data);}
|
|
|
|
inline u32 swap32(u32 _data) {return _byteswap_ulong (_data);}
|
|
|
|
inline u64 swap64(u64 _data) {return _byteswap_uint64(_data);}
|
2014-03-29 10:05:44 +00:00
|
|
|
#elif _M_ARM_32
|
2013-10-29 05:23:17 +00:00
|
|
|
inline u16 swap16 (u16 _data) { u32 data = _data; __asm__ ("rev16 %0, %1\n" : "=l" (data) : "l" (data)); return (u16)data;}
|
|
|
|
inline u32 swap32 (u32 _data) {__asm__ ("rev %0, %1\n" : "=l" (_data) : "l" (_data)); return _data;}
|
2013-02-26 19:49:00 +00:00
|
|
|
inline u64 swap64(u64 _data) {return ((u64)swap32(_data) << 32) | swap32(_data >> 32);}
|
2009-03-07 08:35:01 +00:00
|
|
|
#elif __linux__
|
|
|
|
inline u16 swap16(u16 _data) {return bswap_16(_data);}
|
|
|
|
inline u32 swap32(u32 _data) {return bswap_32(_data);}
|
|
|
|
inline u64 swap64(u64 _data) {return bswap_64(_data);}
|
2010-02-16 08:46:21 +00:00
|
|
|
#elif __APPLE__
|
2010-07-18 08:30:40 +00:00
|
|
|
inline __attribute__((always_inline)) u16 swap16(u16 _data)
|
2014-08-08 17:26:26 +00:00
|
|
|
{return OSSwapInt16(_data);}
|
2010-07-18 08:30:40 +00:00
|
|
|
inline __attribute__((always_inline)) u32 swap32(u32 _data)
|
2014-08-08 17:26:26 +00:00
|
|
|
{return OSSwapInt32(_data);}
|
2010-07-18 08:30:40 +00:00
|
|
|
inline __attribute__((always_inline)) u64 swap64(u64 _data)
|
2014-08-08 17:26:26 +00:00
|
|
|
{return OSSwapInt64(_data);}
|
2011-08-27 18:42:11 +00:00
|
|
|
#elif __FreeBSD__
|
|
|
|
inline u16 swap16(u16 _data) {return bswap16(_data);}
|
|
|
|
inline u32 swap32(u32 _data) {return bswap32(_data);}
|
|
|
|
inline u64 swap64(u64 _data) {return bswap64(_data);}
|
2009-03-07 08:35:01 +00:00
|
|
|
#else
|
|
|
|
// Slow generic implementation.
|
|
|
|
inline u16 swap16(u16 data) {return (data >> 8) | (data << 8);}
|
|
|
|
inline u32 swap32(u32 data) {return (swap16(data) << 16) | swap16(data >> 16);}
|
|
|
|
inline u64 swap64(u64 data) {return ((u64)swap32(data) << 32) | swap32(data >> 32);}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
inline u16 swap16(const u8* _pData) {return swap16(*(const u16*)_pData);}
|
|
|
|
inline u32 swap32(const u8* _pData) {return swap32(*(const u32*)_pData);}
|
|
|
|
inline u64 swap64(const u8* _pData) {return swap64(*(const u64*)_pData);}
|
|
|
|
|
2013-02-21 02:20:53 +00:00
|
|
|
template <int count>
|
|
|
|
void swap(u8*);
|
|
|
|
|
|
|
|
template <>
|
|
|
|
inline void swap<1>(u8* data)
|
|
|
|
{}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
inline void swap<2>(u8* data)
|
|
|
|
{
|
|
|
|
*reinterpret_cast<u16*>(data) = swap16(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
inline void swap<4>(u8* data)
|
|
|
|
{
|
|
|
|
*reinterpret_cast<u32*>(data) = swap32(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
inline void swap<8>(u8* data)
|
|
|
|
{
|
|
|
|
*reinterpret_cast<u64*>(data) = swap64(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
inline T FromBigEndian(T data)
|
|
|
|
{
|
|
|
|
//static_assert(std::is_arithmetic<T>::value, "function only makes sense with arithmetic types");
|
2013-10-29 05:23:17 +00:00
|
|
|
|
2013-02-21 02:20:53 +00:00
|
|
|
swap<sizeof(data)>(reinterpret_cast<u8*>(&data));
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2009-03-28 08:57:34 +00:00
|
|
|
} // Namespace Common
|