2013-11-06 14:21:12 +01:00
|
|
|
/* RetroArch - A frontend for libretro.
|
2014-01-01 01:50:59 +01:00
|
|
|
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
|
|
|
|
* Copyright (C) 2011-2014 - Daniel De Matteis
|
2013-11-06 14:21:12 +01:00
|
|
|
*
|
|
|
|
* RetroArch is free software: you can redistribute it and/or modify it under the terms
|
|
|
|
* of the GNU General Public License as published by the Free Software Found-
|
|
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
|
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
|
|
* PURPOSE. See the GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along with RetroArch.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __RARCH_MISCELLANEOUS_H
|
|
|
|
#define __RARCH_MISCELLANEOUS_H
|
|
|
|
|
2013-11-06 14:55:42 +01:00
|
|
|
#if defined(__CELLOS_LV2__) && !defined(__PSL1GHT__)
|
|
|
|
#include <sys/timer.h>
|
|
|
|
#elif defined(XENON)
|
|
|
|
#include <time/time.h>
|
2013-11-07 21:01:49 -05:00
|
|
|
#elif defined(GEKKO) || defined(__PSL1GHT__) || defined(__QNX__)
|
|
|
|
#include <unistd.h>
|
2014-02-11 16:10:40 +01:00
|
|
|
#elif defined(PSP)
|
|
|
|
#include <pspthreadman.h>
|
2013-11-06 14:55:42 +01:00
|
|
|
#else
|
|
|
|
#include <time.h>
|
|
|
|
#endif
|
|
|
|
|
2013-11-06 20:32:27 +01:00
|
|
|
#if defined(_WIN32) && !defined(_XBOX)
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
|
|
#include <windows.h>
|
2013-12-29 04:08:42 +01:00
|
|
|
#elif defined(_WIN32) && defined(_XBOX)
|
|
|
|
#include <Xtl.h>
|
2013-11-06 20:32:27 +01:00
|
|
|
#endif
|
2014-03-27 21:47:19 +01:00
|
|
|
#include "msvc/msvc_compat.h"
|
2013-11-06 20:32:27 +01:00
|
|
|
|
2013-11-06 14:21:12 +01:00
|
|
|
#include "retroarch_logger.h"
|
2014-09-28 23:21:27 +02:00
|
|
|
#include "endianness.h"
|
2014-03-27 21:44:41 +01:00
|
|
|
#include <limits.h>
|
2013-11-06 14:21:12 +01:00
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* Some platforms do not set this value.
|
|
|
|
* Just assume a value. It's usually 4KiB.
|
|
|
|
* Platforms with a known value (like Win32)
|
|
|
|
* set this value explicitly in platform specific headers.
|
|
|
|
*/
|
|
|
|
|
2013-11-06 22:01:02 +01:00
|
|
|
#ifndef PATH_MAX
|
|
|
|
#define PATH_MAX 4096
|
|
|
|
#endif
|
|
|
|
|
2013-11-06 14:21:12 +01:00
|
|
|
#ifndef max
|
|
|
|
#define max(a, b) ((a) > (b) ? (a) : (b))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef min
|
|
|
|
#define min(a, b) ((a) < (b) ? (a) : (b))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define rarch_assert(cond) do { \
|
2014-03-27 10:44:23 +01:00
|
|
|
if (!(cond)) { RARCH_ERR("Assertion failed at %s:%d.\n", __FILE__, __LINE__); abort(); } \
|
2013-11-06 14:21:12 +01:00
|
|
|
} while(0)
|
|
|
|
|
|
|
|
#define ARRAY_SIZE(a) (sizeof(a) / sizeof((a)[0]))
|
|
|
|
#define RARCH_SCALE_BASE 256
|
|
|
|
|
|
|
|
static inline void rarch_sleep(unsigned msec)
|
|
|
|
{
|
|
|
|
#if defined(__CELLOS_LV2__) && !defined(__PSL1GHT__)
|
|
|
|
sys_timer_usleep(1000 * msec);
|
|
|
|
#elif defined(PSP)
|
|
|
|
sceKernelDelayThread(1000 * msec);
|
|
|
|
#elif defined(_WIN32)
|
|
|
|
Sleep(msec);
|
|
|
|
#elif defined(XENON)
|
|
|
|
udelay(1000 * msec);
|
|
|
|
#elif defined(GEKKO) || defined(__PSL1GHT__) || defined(__QNX__)
|
|
|
|
usleep(1000 * msec);
|
|
|
|
#else
|
|
|
|
struct timespec tv = {0};
|
|
|
|
tv.tv_sec = msec / 1000;
|
|
|
|
tv.tv_nsec = (msec % 1000) * 1000000;
|
|
|
|
nanosleep(&tv, NULL);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint32_t next_pow2(uint32_t v)
|
|
|
|
{
|
|
|
|
v--;
|
|
|
|
v |= v >> 1;
|
|
|
|
v |= v >> 2;
|
|
|
|
v |= v >> 4;
|
|
|
|
v |= v >> 8;
|
|
|
|
v |= v >> 16;
|
|
|
|
v++;
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint32_t prev_pow2(uint32_t v)
|
|
|
|
{
|
|
|
|
v |= v >> 1;
|
|
|
|
v |= v >> 2;
|
|
|
|
v |= v >> 4;
|
|
|
|
v |= v >> 8;
|
|
|
|
v |= v >> 16;
|
|
|
|
return v - (v >> 1);
|
|
|
|
}
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* Helper macros and struct to keep track of many booleans.
|
|
|
|
* To check for multiple bits, use &&, not &.
|
|
|
|
* For OR, | can be used. */
|
2013-11-06 14:21:12 +01:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
uint32_t data[8];
|
|
|
|
} rarch_bits_t;
|
|
|
|
|
2014-10-16 22:25:09 +02:00
|
|
|
#define BIT_SET(a, bit) ((a)[(bit) >> 3] |= (1 << ((bit) & 7)))
|
|
|
|
#define BIT_CLEAR(a, bit) ((a)[(bit) >> 3] &= ~(1 << ((bit) & 7)))
|
|
|
|
#define BIT_GET(a, bit) ((a)[(bit) >> 3] & (1 << ((bit) & 7)))
|
2014-10-05 15:48:06 +02:00
|
|
|
|
2014-10-16 22:25:09 +02:00
|
|
|
#define BIT16_SET(a, bit) ((a) |= (1 << ((bit) & 15)))
|
|
|
|
#define BIT16_CLEAR(a, bit) ((a) &= ~(1 << ((bit) & 15)))
|
|
|
|
#define BIT16_GET(a, bit) (!!((a) & (1 << ((bit) & 15))))
|
|
|
|
#define BIT16_CLEAR_ALL(a) ((a) = 0)
|
2014-09-16 09:26:08 +02:00
|
|
|
|
2014-10-06 01:51:45 +02:00
|
|
|
#define BIT32_SET(a, bit) ((a) |= (1 << ((bit) & 31)))
|
|
|
|
#define BIT32_CLEAR(a, bit) ((a) &= ~(1 << ((bit) & 31)))
|
|
|
|
#define BIT32_GET(a, bit) (!!((a) & (1 << ((bit) & 31))))
|
|
|
|
#define BIT32_CLEAR_ALL(a) ((a) = 0)
|
|
|
|
|
2014-10-16 22:25:09 +02:00
|
|
|
#define BIT64_SET(a, bit) ((a) |= (1ULL << ((bit) & 63)))
|
|
|
|
#define BIT64_CLEAR(a, bit) ((a) &= ~(1ULL << ((bit) & 63)))
|
|
|
|
#define BIT64_GET(a, bit) (!!((a) & (1ULL << ((bit) & 63))))
|
|
|
|
#define BIT64_CLEAR_ALL(a) ((a) = 0)
|
|
|
|
|
|
|
|
#define BIT128_SET(a, bit) ((a).data[(bit) >> 5] |= (1 << ((bit) & 31))
|
|
|
|
#define BIT128_CLEAR(a, bit) ((a).data[(bit) >> 5] &= ~(1 << ((bit) & 31)))
|
|
|
|
#define BIT128_GET(a, bit) ((a).data[(bit) >> 5] & (1 << ((bit) & 31)))
|
|
|
|
#define BIT128_CLEAR_ALL(a) memset(&(a), 0, sizeof(a));
|
|
|
|
|
2014-10-05 15:58:06 +02:00
|
|
|
|
2013-11-06 14:21:12 +01:00
|
|
|
#endif
|