1396 lines
31 KiB
C++
1396 lines
31 KiB
C++
// DO NOT REMOVE/DISABLE THESE MATH AND COMPILER SANITY TESTS. THEY EXIST FOR A REASON.
|
|
|
|
/* Mednafen - Multi-system Emulator
|
|
*
|
|
* This program 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 Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
// We really don't want NDEBUG defined ;)
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#undef NDEBUG
|
|
|
|
#include <stdio.h>
|
|
#include <exception>
|
|
#include <vector>
|
|
#include <algorithm>
|
|
#include "octoshock.h"
|
|
#include "math_ops.h"
|
|
#include "error.h"
|
|
|
|
#ifdef WANT_TEST_LEPACKER
|
|
#include "lepacker.h"
|
|
#endif
|
|
|
|
#ifdef WANT_TEST_HASHES
|
|
#include <mednafen/hash/sha1.h>
|
|
#include <mednafen/hash/sha256.h>
|
|
#endif
|
|
|
|
#ifdef WANT_TEST_ZLIB
|
|
#include <zlib.h>
|
|
#endif
|
|
|
|
#undef NDEBUG
|
|
#include <assert.h>
|
|
#include <math.h>
|
|
|
|
#include "psx/masmem.h"
|
|
|
|
#if defined(HAVE_FENV_H)
|
|
#include <fenv.h>
|
|
#endif
|
|
|
|
#define FATALME { printf("Math test failed: %s:%d\n", __FILE__, __LINE__); fprintf(stderr, "Math test failed: %s:%d\n", __FILE__, __LINE__); return(0); }
|
|
|
|
namespace MDFN_TESTS_CPP
|
|
{
|
|
|
|
// Don't define this static, and don't define it const. We want these tests to be done at run time, not compile time(although maybe we should do both...).
|
|
typedef struct
|
|
{
|
|
int bits;
|
|
uint32 negative_one;
|
|
uint32 mostneg;
|
|
int32 mostnegresult;
|
|
} MathTestEntry;
|
|
|
|
#define ADD_MTE(_bits) { _bits, ((uint32)1 << _bits) - 1, (uint32)1 << (_bits - 1), (int32)(0 - ((uint32)1 << (_bits - 1))) }
|
|
|
|
MathTestEntry math_test_vals[] =
|
|
{
|
|
{ 9, 0x01FF, 0x0100, -256 },
|
|
{ 10, 0x03FF, 0x0200, -512 },
|
|
{ 11, 0x07FF, 0x0400, -1024 },
|
|
{ 12, 0x0FFF, 0x0800, -2048 },
|
|
{ 13, 0x1FFF, 0x1000, -4096 },
|
|
{ 14, 0x3FFF, 0x2000, -8192 },
|
|
{ 15, 0x7FFF, 0x4000, -16384 },
|
|
|
|
ADD_MTE(17),
|
|
ADD_MTE(18),
|
|
ADD_MTE(19),
|
|
ADD_MTE(20),
|
|
ADD_MTE(21),
|
|
ADD_MTE(22),
|
|
ADD_MTE(23),
|
|
ADD_MTE(24),
|
|
ADD_MTE(25),
|
|
ADD_MTE(26),
|
|
ADD_MTE(27),
|
|
ADD_MTE(28),
|
|
ADD_MTE(29),
|
|
ADD_MTE(30),
|
|
ADD_MTE(31),
|
|
|
|
{ 0, 0, 0, 0 },
|
|
};
|
|
|
|
static bool DoSizeofTests(void)
|
|
{
|
|
const int SizePairs[][2] =
|
|
{
|
|
{ sizeof(uint8), 1 },
|
|
{ sizeof(int8), 1 },
|
|
|
|
{ sizeof(uint16), 2 },
|
|
{ sizeof(int16), 2 },
|
|
|
|
{ sizeof(uint32), 4 },
|
|
{ sizeof(int32), 4 },
|
|
|
|
{ sizeof(uint64), 8 },
|
|
{ sizeof(int64), 8 },
|
|
|
|
{ 0, 0 },
|
|
};
|
|
|
|
int i = -1;
|
|
|
|
while(SizePairs[++i][0])
|
|
{
|
|
if(SizePairs[i][0] != SizePairs[i][1])
|
|
FATALME;
|
|
}
|
|
|
|
assert(sizeof(char) == 1);
|
|
assert(sizeof(int) == 4);
|
|
assert(sizeof(long) >= 4);
|
|
|
|
assert(sizeof(char) == SIZEOF_CHAR);
|
|
assert(sizeof(short) == SIZEOF_SHORT);
|
|
assert(sizeof(int) == SIZEOF_INT);
|
|
assert(sizeof(long) == SIZEOF_LONG);
|
|
assert(sizeof(long long) == SIZEOF_LONG_LONG);
|
|
|
|
assert(sizeof(off_t) == SIZEOF_OFF_T);
|
|
|
|
return(1);
|
|
}
|
|
|
|
static void AntiNSOBugTest_Sub1_a(int *array) NO_INLINE;
|
|
static void AntiNSOBugTest_Sub1_a(int *array)
|
|
{
|
|
for(int value = 0; value < 127; value++)
|
|
array[value] += (int8)value * 15;
|
|
}
|
|
|
|
static void AntiNSOBugTest_Sub1_b(int *array) NO_INLINE;
|
|
static void AntiNSOBugTest_Sub1_b(int *array)
|
|
{
|
|
for(int value = 127; value < 256; value++)
|
|
array[value] += (int8)value * 15;
|
|
}
|
|
|
|
static void AntiNSOBugTest_Sub2(int *array) NO_INLINE;
|
|
static void AntiNSOBugTest_Sub2(int *array)
|
|
{
|
|
for(int value = 0; value < 256; value++)
|
|
array[value] += (int8)value * 15;
|
|
}
|
|
|
|
static void AntiNSOBugTest_Sub3(int *array) NO_INLINE;
|
|
static void AntiNSOBugTest_Sub3(int *array)
|
|
{
|
|
for(int value = 0; value < 256; value++)
|
|
{
|
|
if(value >= 128)
|
|
array[value] = (value - 256) * 15;
|
|
else
|
|
array[value] = value * 15;
|
|
}
|
|
}
|
|
|
|
static bool DoAntiNSOBugTest(void)
|
|
{
|
|
int array1[256], array2[256], array3[256];
|
|
|
|
memset(array1, 0, sizeof(array1));
|
|
memset(array2, 0, sizeof(array2));
|
|
memset(array3, 0, sizeof(array3));
|
|
|
|
AntiNSOBugTest_Sub1_a(array1);
|
|
AntiNSOBugTest_Sub1_b(array1);
|
|
AntiNSOBugTest_Sub2(array2);
|
|
AntiNSOBugTest_Sub3(array3);
|
|
|
|
for(int i = 0; i < 256; i++)
|
|
{
|
|
if((array1[i] != array2[i]) || (array2[i] != array3[i]))
|
|
{
|
|
printf("%d %d %d %d\n", i, array1[i], array2[i], array3[i]);
|
|
FATALME;
|
|
}
|
|
}
|
|
//for(int value = 0; value < 256; value++)
|
|
// printf("%d, %d\n", (int8)value, ((int8)value) * 15);
|
|
|
|
return(1);
|
|
}
|
|
|
|
//
|
|
// Related: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=61741
|
|
//
|
|
// Not found to be causing problems in Mednafen(unlike the earlier no-strict-overflow problem and associated test),
|
|
// but better safe than sorry.
|
|
//
|
|
static void DoAntiNSOBugTest2014_SubA(int a) NO_INLINE NO_CLONE;
|
|
static void DoAntiNSOBugTest2014_SubA(int a)
|
|
{
|
|
char c = 0;
|
|
|
|
for(; a; a--)
|
|
{
|
|
for(; c >= 0; c++)
|
|
{
|
|
|
|
}
|
|
}
|
|
|
|
assert(c == -128);
|
|
}
|
|
|
|
static int ANSOBT_CallCount;
|
|
static void DoAntiNSOBugTest2014_SubMx_F(void) NO_INLINE NO_CLONE;
|
|
static void DoAntiNSOBugTest2014_SubMx_F(void)
|
|
{
|
|
ANSOBT_CallCount++;
|
|
|
|
assert(ANSOBT_CallCount < 1000);
|
|
}
|
|
|
|
static void DoAntiNSOBugTest2014_SubM1(void) NO_INLINE NO_CLONE;
|
|
static void DoAntiNSOBugTest2014_SubM1(void)
|
|
{
|
|
char a;
|
|
|
|
for(a = 127 - 1; a >= 0; a++)
|
|
DoAntiNSOBugTest2014_SubMx_F();
|
|
}
|
|
|
|
static void DoAntiNSOBugTest2014_SubM3(void) NO_INLINE NO_CLONE;
|
|
static void DoAntiNSOBugTest2014_SubM3(void)
|
|
{
|
|
char a;
|
|
|
|
for(a = 127 - 3; a >= 0; a++)
|
|
DoAntiNSOBugTest2014_SubMx_F();
|
|
}
|
|
|
|
|
|
static void DoAntiNSOBugTest2014(void)
|
|
{
|
|
DoAntiNSOBugTest2014_SubA(1);
|
|
|
|
ANSOBT_CallCount = 0;
|
|
DoAntiNSOBugTest2014_SubM1();
|
|
assert(ANSOBT_CallCount == 2);
|
|
|
|
ANSOBT_CallCount = 0;
|
|
DoAntiNSOBugTest2014_SubM3();
|
|
assert(ANSOBT_CallCount == 4);
|
|
}
|
|
|
|
|
|
#ifdef WANT_TEST_LEPACKER
|
|
bool DoLEPackerTest(void)
|
|
{
|
|
MDFN::LEPacker mizer;
|
|
static const uint8 correct_result[24] = { 0xed, 0xfe, 0xed, 0xde, 0xaa, 0xca, 0xef, 0xbe, 0xbe, 0xba, 0xfe, 0xca, 0xad, 0xde, 0x01, 0x9a, 0x0c, 0xa7, 0xff, 0x00, 0xff, 0xff, 0x55, 0x7f };
|
|
|
|
uint64 u64_test = 0xDEADCAFEBABEBEEFULL;
|
|
uint32 u32_test = 0xDEEDFEED;
|
|
uint16 u16_test = 0xCAAA;
|
|
uint8 u8_test = 0x55;
|
|
int32 s32_test = -5829478;
|
|
int16 s16_test = -1;
|
|
int8 s8_test = 127;
|
|
|
|
bool bool_test0 = 0;
|
|
bool bool_test1 = 1;
|
|
|
|
mizer ^ u32_test;
|
|
mizer ^ u16_test;
|
|
mizer ^ u64_test;
|
|
mizer ^ bool_test1;
|
|
mizer ^ s32_test;
|
|
mizer ^ bool_test0;
|
|
mizer ^ s16_test;
|
|
mizer ^ u8_test;
|
|
mizer ^ s8_test;
|
|
|
|
if(mizer.size() != 24)
|
|
{
|
|
printf("Test failed: LEPacker data incorrect size.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
for(unsigned int i = 0; i < mizer.size(); i++)
|
|
if(mizer[i] != correct_result[i])
|
|
{
|
|
printf("Test failed: LEPacker packed data incorrect.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
u64_test = 0;
|
|
u32_test = 0;
|
|
u16_test = 0;
|
|
u8_test = 0;
|
|
s32_test = 0;
|
|
s16_test = 0;
|
|
s8_test = 0;
|
|
|
|
bool_test0 = 1;
|
|
bool_test1 = 0;
|
|
|
|
mizer.set_read_mode(TRUE);
|
|
|
|
mizer ^ u32_test;
|
|
mizer ^ u16_test;
|
|
mizer ^ u64_test;
|
|
mizer ^ bool_test1;
|
|
mizer ^ s32_test;
|
|
mizer ^ bool_test0;
|
|
mizer ^ s16_test;
|
|
mizer ^ u8_test;
|
|
mizer ^ s8_test;
|
|
|
|
|
|
if(u32_test != 0xDEEDFEED)
|
|
{
|
|
printf("Test failed: LEPacker u32 unpacking incorrect.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
if(u16_test != 0xCAAA)
|
|
{
|
|
printf("Test failed: LEPacker u16 unpacking incorrect.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
if(u64_test != 0xDEADCAFEBABEBEEFULL)
|
|
{
|
|
printf("%16llx\n", (unsigned long long)u64_test);
|
|
printf("Test failed: LEPacker u64 unpacking incorrect.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
if(u8_test != 0x55)
|
|
{
|
|
printf("Test failed: LEPacker u8 unpacking incorrect.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
if(s32_test != -5829478)
|
|
{
|
|
printf("Test failed: LEPacker s32 unpacking incorrect.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
if(s16_test != -1)
|
|
{
|
|
printf("Test failed: LEPacker s16 unpacking incorrect.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
if(s8_test != 127)
|
|
{
|
|
printf("Test failed: LEPacker s8 unpacking incorrect.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
if(bool_test0 != 0)
|
|
{
|
|
printf("Test failed: LEPacker bool unpacking incorrect.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
if(bool_test1 != 1)
|
|
{
|
|
printf("Test failed: LEPacker bool unpacking incorrect.\n");
|
|
return(FALSE);
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
#endif
|
|
|
|
struct MathTestTSOEntry
|
|
{
|
|
int32 a;
|
|
int32 b;
|
|
};
|
|
|
|
// Don't declare as static(though whopr might mess it up anyway)
|
|
MathTestTSOEntry MathTestTSOTests[] =
|
|
{
|
|
{ 0x7FFFFFFF, 2 },
|
|
{ 0x7FFFFFFE, 0x7FFFFFFF },
|
|
{ 0x7FFFFFFF, 0x7FFFFFFF },
|
|
{ 0x7FFFFFFE, 0x7FFFFFFE },
|
|
};
|
|
|
|
volatile int32 MDFNTestsCPP_SLS_Var = (int32)0xDEADBEEF;
|
|
volatile int8 MDFNTestsCPP_SLS_Var8 = (int8)0xEF;
|
|
volatile int16 MDFNTestsCPP_SLS_Var16 = (int16)0xBEEF;
|
|
int32 MDFNTestsCPP_SLS_Var_NT = (int32)0xDEADBEEF;
|
|
int32 MDFNTestsCPP_SLS_Var_NT2 = (int32)0x7EADBEEF;
|
|
|
|
static uint64 NO_INLINE NO_CLONE Mul_U16U16U32U64_Proper(uint16 a, uint16 b) // For reference
|
|
{
|
|
return (uint32)a * (uint32)b;
|
|
}
|
|
|
|
static uint64 NO_INLINE NO_CLONE Mul_U16U16U32U64(uint16 a, uint16 b)
|
|
{
|
|
return (uint32)(a * b);
|
|
}
|
|
|
|
static void TestSignedOverflow(void)
|
|
{
|
|
assert(Mul_U16U16U32U64_Proper(65535, 65535) == 0xfffe0001ULL);
|
|
assert(Mul_U16U16U32U64(65535, 65535) == 0xfffe0001ULL);
|
|
|
|
for(unsigned int i = 0; i < sizeof(MathTestTSOTests) / sizeof(MathTestTSOEntry); i++)
|
|
{
|
|
int32 a = MathTestTSOTests[i].a;
|
|
int32 b = MathTestTSOTests[i].b;
|
|
|
|
assert((a + b) < a && (a + b) < b);
|
|
|
|
assert((a + 0x7FFFFFFE) < a);
|
|
assert((b + 0x7FFFFFFE) < b);
|
|
|
|
assert((a + 0x7FFFFFFF) < a);
|
|
assert((b + 0x7FFFFFFF) < b);
|
|
|
|
assert((int32)(a + 0x80000000) < a);
|
|
assert((int32)(b + 0x80000000) < b);
|
|
|
|
assert((int32)(a ^ 0x80000000) < a);
|
|
assert((int32)(b ^ 0x80000000) < b);
|
|
}
|
|
|
|
for(unsigned i = 0; i < 64; i++)
|
|
{
|
|
MDFNTestsCPP_SLS_Var = (MDFNTestsCPP_SLS_Var << 1) ^ ((MDFNTestsCPP_SLS_Var << 2) + 0x7FFFFFFF) ^ ((MDFNTestsCPP_SLS_Var >> 31) & 0x3);
|
|
MDFNTestsCPP_SLS_Var8 = (MDFNTestsCPP_SLS_Var8 << 1) ^ ((MDFNTestsCPP_SLS_Var8 << 2) + 0x7F) ^ ((MDFNTestsCPP_SLS_Var8 >> 7) & 0x3);
|
|
MDFNTestsCPP_SLS_Var16 = (MDFNTestsCPP_SLS_Var16 << 1) ^ ((MDFNTestsCPP_SLS_Var16 << 2) + 0x7FFF) ^ ((MDFNTestsCPP_SLS_Var16 >> 15) & 0x3);
|
|
}
|
|
|
|
{
|
|
int8 a = MDFNTestsCPP_SLS_Var8;
|
|
int16 b = MDFNTestsCPP_SLS_Var16;
|
|
int32 c = MDFNTestsCPP_SLS_Var;
|
|
int64 d = (int64)MDFNTestsCPP_SLS_Var * (int64)MDFNTestsCPP_SLS_Var;
|
|
int32 e = c;
|
|
int64 f = c;
|
|
|
|
for(int i = 0; i < 64; i++)
|
|
{
|
|
a += a * i + b;
|
|
b += b * i + c;
|
|
c += c * i + d;
|
|
d += d * i + a;
|
|
|
|
e += e * i + c;
|
|
f += f * i + c;
|
|
}
|
|
//printf("%08x %16llx - %02x %04x %08x %16llx\n", (uint32)e, (uint64)f, (uint8)a, (uint16)b, (uint32)c, (uint64)d);
|
|
assert((uint32)e == (uint32)f && (uint32)e == 0x00c37de2 && (uint64)f == 0x5d17261900c37de2);
|
|
assert((uint8)a == 0xbf);
|
|
assert((uint16)b == 0xb77c);
|
|
assert((uint32)c == 0xb4244622U);
|
|
assert((uint64)d == 0xa966e02ed95c83fULL);
|
|
}
|
|
|
|
|
|
//printf("%02x %04x %08x\n", (uint8)MDFNTestsCPP_SLS_Var8, (uint16)MDFNTestsCPP_SLS_Var16, (uint32)MDFNTestsCPP_SLS_Var);
|
|
assert((uint8)MDFNTestsCPP_SLS_Var8 == 0x04);
|
|
assert((uint16)MDFNTestsCPP_SLS_Var16 == 0xa7d8);
|
|
assert((uint32)MDFNTestsCPP_SLS_Var == 0x4ef11a23);
|
|
|
|
for(signed i = 1; i != 0; i =~-i); // Not really signed overflow, but meh!
|
|
for(signed i = -1; i != 0; i <<= 1);
|
|
for(signed i = 1; i >= 0; i *= 3);
|
|
|
|
if(MDFNTestsCPP_SLS_Var_NT < 0)
|
|
assert((MDFNTestsCPP_SLS_Var_NT << 2) > 0);
|
|
|
|
if(MDFNTestsCPP_SLS_Var_NT2 > 0)
|
|
assert((MDFNTestsCPP_SLS_Var_NT2 << 2) < 0);
|
|
}
|
|
|
|
unsigned MDFNTests_OverShiftAmounts[3] = { 8, 16, 32};
|
|
uint32 MDFNTests_OverShiftTV = 0xBEEFD00D;
|
|
static void TestDefinedOverShift(void)
|
|
{
|
|
//for(unsigned sa = 0; sa < 4; sa++)
|
|
{
|
|
for(unsigned i = 0; i < 2; i++)
|
|
{
|
|
uint8 v8 = MDFNTests_OverShiftTV;
|
|
uint16 v16 = MDFNTests_OverShiftTV;
|
|
uint32 v32 = MDFNTests_OverShiftTV;
|
|
|
|
int8 iv8 = MDFNTests_OverShiftTV;
|
|
int16 iv16 = MDFNTests_OverShiftTV;
|
|
int32 iv32 = MDFNTests_OverShiftTV;
|
|
|
|
if(i == 1)
|
|
{
|
|
v8 >>= MDFNTests_OverShiftAmounts[0];
|
|
v16 >>= MDFNTests_OverShiftAmounts[1];
|
|
v32 = (uint64)v32 >> MDFNTests_OverShiftAmounts[2];
|
|
|
|
iv8 >>= MDFNTests_OverShiftAmounts[0];
|
|
iv16 >>= MDFNTests_OverShiftAmounts[1];
|
|
iv32 = (int64)iv32 >> MDFNTests_OverShiftAmounts[2];
|
|
}
|
|
else
|
|
{
|
|
v8 <<= MDFNTests_OverShiftAmounts[0];
|
|
v16 <<= MDFNTests_OverShiftAmounts[1];
|
|
v32 = (uint64)v32 << MDFNTests_OverShiftAmounts[2];
|
|
|
|
iv8 <<= MDFNTests_OverShiftAmounts[0];
|
|
iv16 <<= MDFNTests_OverShiftAmounts[1];
|
|
iv32 = (int64)iv32 << MDFNTests_OverShiftAmounts[2];
|
|
}
|
|
|
|
assert(v8 == 0);
|
|
assert(v16 == 0);
|
|
assert(v32 == 0);
|
|
|
|
assert(iv8 == 0);
|
|
assert(iv16 == -(int)i);
|
|
assert(iv32 == -(int)i);
|
|
}
|
|
}
|
|
}
|
|
|
|
static uint8 BoolConvSupportFunc(void) MDFN_COLD NO_INLINE;
|
|
static uint8 BoolConvSupportFunc(void)
|
|
{
|
|
return 0xFF;
|
|
}
|
|
|
|
static bool BoolConv0(void) MDFN_COLD NO_INLINE;
|
|
static bool BoolConv0(void)
|
|
{
|
|
return BoolConvSupportFunc() & 1;
|
|
}
|
|
|
|
static void BoolTestThing(unsigned val) MDFN_COLD NO_INLINE;
|
|
static void BoolTestThing(unsigned val)
|
|
{
|
|
if(val != 1)
|
|
printf("%u\n", val);
|
|
|
|
assert(val == 1);
|
|
}
|
|
|
|
static void TestBoolConv(void)
|
|
{
|
|
BoolTestThing(BoolConv0());
|
|
}
|
|
|
|
static void TestNarrowConstFold(void) NO_INLINE MDFN_COLD;
|
|
static void TestNarrowConstFold(void)
|
|
{
|
|
unsigned sa = 8;
|
|
uint8 za[1] = { 0 };
|
|
int a;
|
|
|
|
a = za[0] < (uint8)(1 << sa);
|
|
|
|
assert(a == 0);
|
|
}
|
|
|
|
|
|
unsigned MDFNTests_ModTern_a = 2;
|
|
unsigned MDFNTests_ModTern_b = 0;
|
|
static void ModTernTestEval(unsigned v) NO_INLINE MDFN_COLD;
|
|
static void ModTernTestEval(unsigned v)
|
|
{
|
|
assert(v == 0);
|
|
}
|
|
|
|
static void TestModTern(void) NO_INLINE MDFN_COLD;
|
|
static void TestModTern(void)
|
|
{
|
|
if(!MDFNTests_ModTern_b)
|
|
{
|
|
MDFNTests_ModTern_b = MDFNTests_ModTern_a;
|
|
|
|
if(1 % (MDFNTests_ModTern_a ? MDFNTests_ModTern_a : 2))
|
|
MDFNTests_ModTern_b = 0;
|
|
}
|
|
ModTernTestEval(MDFNTests_ModTern_b);
|
|
}
|
|
|
|
static int TestBWNotMask31GTZ_Sub(int a) NO_INLINE NO_CLONE;
|
|
static int TestBWNotMask31GTZ_Sub(int a)
|
|
{
|
|
a = (((~a) & 0x80000000LL) > 0) + 1;
|
|
return a;
|
|
}
|
|
|
|
static void TestBWNotMask31GTZ(void)
|
|
{
|
|
assert(TestBWNotMask31GTZ_Sub(0) == 2);
|
|
}
|
|
|
|
int MDFN_tests_TestTernary_val = 0;
|
|
static void NO_INLINE NO_CLONE TestTernary_Sub(void)
|
|
{
|
|
MDFN_tests_TestTernary_val++;
|
|
}
|
|
|
|
static void TestTernary(void)
|
|
{
|
|
int a = ((MDFN_tests_TestTernary_val++) ? (MDFN_tests_TestTernary_val = 20) : (TestTernary_Sub(), MDFN_tests_TestTernary_val));
|
|
|
|
assert(a == 2);
|
|
}
|
|
|
|
size_t TestLLVM15470_Counter;
|
|
void NO_INLINE NO_CLONE TestLLVM15470_Sub2(size_t x)
|
|
{
|
|
assert(x == TestLLVM15470_Counter);
|
|
TestLLVM15470_Counter++;
|
|
}
|
|
|
|
void NO_INLINE NO_CLONE TestLLVM15470_Sub(size_t m)
|
|
{
|
|
size_t m2 = ~(size_t)0;
|
|
|
|
for(size_t i = 1; i <= 4; i *= m)
|
|
m2++;
|
|
|
|
for(size_t a = 0; a < 2; a++)
|
|
{
|
|
for(size_t b = 1; b <= 2; b++)
|
|
{
|
|
TestLLVM15470_Sub2(a * m2 + b);
|
|
}
|
|
}
|
|
}
|
|
|
|
void NO_INLINE NO_CLONE TestLLVM15470(void)
|
|
{
|
|
TestLLVM15470_Counter = 1;
|
|
TestLLVM15470_Sub(2);
|
|
}
|
|
|
|
int NO_INLINE NO_CLONE TestGCC60196_Sub(const int16* data, int count)
|
|
{
|
|
int ret = 0;
|
|
|
|
for(int i = 0; i < count; i++)
|
|
ret += i * data[i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
void NO_INLINE NO_CLONE TestGCC60196(void)
|
|
{
|
|
int16 ta[16];
|
|
|
|
for(unsigned i = 0; i < 16; i++)
|
|
ta[i] = 1;
|
|
|
|
assert(TestGCC60196_Sub(ta, sizeof(ta) / sizeof(ta[0])) == 120);
|
|
}
|
|
|
|
template<typename A, typename B>
|
|
void NO_INLINE NO_CLONE TestSUCompare_Sub(A a, B b)
|
|
{
|
|
assert(a < b);
|
|
}
|
|
|
|
int16 TestSUCompare_x0 = 256;
|
|
|
|
void NO_INLINE NO_CLONE TestSUCompare(void)
|
|
{
|
|
int8 a = 1;
|
|
uint8 b = 255;
|
|
int16 c = 1;
|
|
uint16 d = 65535;
|
|
int32 e = 1;
|
|
uint32 f = ~0U;
|
|
int64 g = ~(uint32)0;
|
|
uint64 h = ~(uint64)0;
|
|
|
|
assert(a < b);
|
|
assert(c < d);
|
|
assert((uint32)e < f);
|
|
assert((uint64)g < h);
|
|
|
|
TestSUCompare_Sub<int8, uint8>(1, 255);
|
|
TestSUCompare_Sub<int16, uint16>(1, 65535);
|
|
|
|
TestSUCompare_Sub<int8, uint8>(TestSUCompare_x0, 255);
|
|
}
|
|
|
|
static void DoAlignmentChecks(void)
|
|
{
|
|
uint8 padding0[3];
|
|
alignas(16) uint8 aligned0[7];
|
|
alignas(4) uint8 aligned1[2];
|
|
alignas(16) uint32 aligned2[2];
|
|
uint8 padding1[3];
|
|
|
|
static uint8 g_padding0[3];
|
|
alignas(16) static uint8 g_aligned0[7];
|
|
alignas(4) static uint8 g_aligned1[2];
|
|
alignas(16) static uint32 g_aligned2[2];
|
|
static uint8 g_padding1[3];
|
|
|
|
// Make sure compiler doesn't removing padding vars
|
|
assert((&padding0[1] - &padding0[0]) == 1);
|
|
assert((&padding1[1] - &padding1[0]) == 1);
|
|
assert((&g_padding0[1] - &g_padding0[0]) == 1);
|
|
assert((&g_padding1[1] - &g_padding1[0]) == 1);
|
|
|
|
|
|
assert( (((unsigned long long)&aligned0[0]) & 0xF) == 0);
|
|
assert( (((unsigned long long)&aligned1[0]) & 0x3) == 0);
|
|
assert( (((unsigned long long)&aligned2[0]) & 0xF) == 0);
|
|
|
|
assert(((uint8 *)&aligned0[1] - (uint8 *)&aligned0[0]) == 1);
|
|
assert(((uint8 *)&aligned1[1] - (uint8 *)&aligned1[0]) == 1);
|
|
assert(((uint8 *)&aligned2[1] - (uint8 *)&aligned2[0]) == 4);
|
|
|
|
|
|
assert( (((unsigned long long)&g_aligned0[0]) & 0xF) == 0);
|
|
assert( (((unsigned long long)&g_aligned1[0]) & 0x3) == 0);
|
|
assert( (((unsigned long long)&g_aligned2[0]) & 0xF) == 0);
|
|
|
|
assert(((uint8 *)&g_aligned0[1] - (uint8 *)&g_aligned0[0]) == 1);
|
|
assert(((uint8 *)&g_aligned1[1] - (uint8 *)&g_aligned1[0]) == 1);
|
|
assert(((uint8 *)&g_aligned2[1] - (uint8 *)&g_aligned2[0]) == 4);
|
|
}
|
|
|
|
static uint32 NO_INLINE NO_CLONE RunMASMemTests_DoomAndGloom(uint32 offset)
|
|
{
|
|
MultiAccessSizeMem<4, false> mt0;
|
|
|
|
mt0.WriteU32(offset, 4);
|
|
mt0.WriteU16(offset, 0);
|
|
mt0.WriteU32(offset, mt0.ReadU32(offset) + 1);
|
|
|
|
return mt0.ReadU32(offset);
|
|
}
|
|
|
|
static void RunMASMemTests(void)
|
|
{
|
|
// Little endian:
|
|
{
|
|
MultiAccessSizeMem<4, false> mt0;
|
|
|
|
mt0.WriteU16(0, 0xDEAD);
|
|
mt0.WriteU32(0, 0xCAFEBEEF);
|
|
mt0.WriteU16(2, mt0.ReadU16(0));
|
|
mt0.WriteU8(1, mt0.ReadU8(0));
|
|
mt0.WriteU16(2, mt0.ReadU16(0));
|
|
mt0.WriteU32(0, mt0.ReadU32(0) + 0x13121111);
|
|
|
|
assert(mt0.ReadU16(0) == 0x0100 && mt0.ReadU16(2) == 0x0302);
|
|
assert(mt0.ReadU32(0) == 0x03020100);
|
|
|
|
mt0.WriteU32(0, 0xB0B0AA55);
|
|
mt0.WriteU24(0, 0xDEADBEEF);
|
|
assert(mt0.ReadU32(0) == 0xB0ADBEEF);
|
|
assert(mt0.ReadU24(1) == 0x00B0ADBE);
|
|
}
|
|
|
|
// Big endian:
|
|
{
|
|
MultiAccessSizeMem<4, true> mt0;
|
|
|
|
mt0.WriteU16(2, 0xDEAD);
|
|
mt0.WriteU32(0, 0xCAFEBEEF);
|
|
mt0.WriteU16(0, mt0.ReadU16(2));
|
|
mt0.WriteU8(2, mt0.ReadU8(3));
|
|
mt0.WriteU16(0, mt0.ReadU16(2));
|
|
mt0.WriteU32(0, mt0.ReadU32(0) + 0x13121111);
|
|
|
|
assert(mt0.ReadU16(2) == 0x0100 && mt0.ReadU16(0) == 0x0302);
|
|
assert(mt0.ReadU32(0) == 0x03020100);
|
|
|
|
mt0.WriteU32(0, 0xB0B0AA55);
|
|
mt0.WriteU24(1, 0xDEADBEEF);
|
|
assert(mt0.ReadU32(0) == 0xB0ADBEEF);
|
|
assert(mt0.ReadU24(0) == 0x00B0ADBE);
|
|
}
|
|
|
|
assert(RunMASMemTests_DoomAndGloom(0) == 1);
|
|
}
|
|
|
|
#ifdef WANT_TEST_EXCEPTION
|
|
static void NO_INLINE NO_CLONE ExceptionTestSub(int v, int n, int* y)
|
|
{
|
|
if(n)
|
|
{
|
|
if(n & 1)
|
|
{
|
|
try
|
|
{
|
|
ExceptionTestSub(v + n, n - 1, y);
|
|
}
|
|
catch(const std::exception &e)
|
|
{
|
|
(*y)++;
|
|
throw;
|
|
}
|
|
}
|
|
else
|
|
ExceptionTestSub(v + n, n - 1, y);
|
|
}
|
|
else
|
|
throw MDFN_Error(v, "%d", v);
|
|
}
|
|
|
|
static void RunExceptionTests(void)
|
|
{
|
|
int y = 0;
|
|
int z = 0;
|
|
|
|
for(int x = -8; x < 8; x++)
|
|
{
|
|
try
|
|
{
|
|
ExceptionTestSub(x, x & 3, &y);
|
|
}
|
|
catch(const MDFN_Error &e)
|
|
{
|
|
int epv = x;
|
|
|
|
for(unsigned i = x & 3; i; i--)
|
|
epv += i;
|
|
|
|
z += epv;
|
|
|
|
assert(e.GetErrno() == epv);
|
|
assert(atoi(e.what()) == epv);
|
|
continue;
|
|
}
|
|
catch(...)
|
|
{
|
|
abort();
|
|
}
|
|
abort();
|
|
}
|
|
|
|
assert(y == 16);
|
|
assert(z == 32);
|
|
}
|
|
#endif
|
|
|
|
std::vector<int> stltests_vec[2];
|
|
|
|
static void NO_INLINE NO_CLONE RunSTLTests_Sub0(int v)
|
|
{
|
|
stltests_vec[0].assign(v, v);
|
|
}
|
|
|
|
static void RunSTLTests(void)
|
|
{
|
|
assert(stltests_vec[0] == stltests_vec[1]);
|
|
RunSTLTests_Sub0(0);
|
|
assert(stltests_vec[0] == stltests_vec[1]);
|
|
RunSTLTests_Sub0(1);
|
|
RunSTLTests_Sub0(0);
|
|
assert(stltests_vec[0] == stltests_vec[1]);
|
|
}
|
|
|
|
static void LZCount_Test(void)
|
|
{
|
|
for(uint32 i = 0, x = 0; i < 33; i++, x = (x << 1) + 1)
|
|
{
|
|
assert(MDFN_lzcount32(x) == 32 - i);
|
|
}
|
|
|
|
for(uint32 i = 0, x = 0; i < 33; i++, x = (x ? (x << 1) : 1))
|
|
{
|
|
assert(MDFN_lzcount32(x) == 32 - i);
|
|
}
|
|
|
|
for(uint64 i = 0, x = 0; i < 65; i++, x = (x << 1) + 1)
|
|
{
|
|
assert(MDFN_lzcount64(x) == 64 - i);
|
|
}
|
|
|
|
for(uint64 i = 0, x = 0; i < 65; i++, x = (x ? (x << 1) : 1))
|
|
{
|
|
assert(MDFN_lzcount64(x) == 64 - i);
|
|
}
|
|
|
|
uint32 tv = 0;
|
|
for(uint32 i = 0, x = 1; i < 200; i++, x = (x * 9) + MDFN_lzcount32(x) + MDFN_lzcount32(x >> (x & 31)))
|
|
{
|
|
tv += x;
|
|
}
|
|
assert(tv == 0x397d920f);
|
|
|
|
uint64 tv64 = 0;
|
|
for(uint64 i = 0, x = 1; i < 200; i++, x = (x * 9) + MDFN_lzcount64(x) + MDFN_lzcount64(x >> (x & 63)))
|
|
{
|
|
tv64 += x;
|
|
}
|
|
assert(tv64 == 0x7b8263de01922c29);
|
|
}
|
|
|
|
|
|
// don't make this static, and don't make it local scope. Whole-program optimization might defeat the purpose of this, though...
|
|
unsigned int mdfn_shifty_test[4] =
|
|
{
|
|
0, 8, 16, 32
|
|
};
|
|
|
|
|
|
// Don't make static.
|
|
double mdfn_fptest0_sub(double x, double n) MDFN_COLD NO_INLINE;
|
|
double mdfn_fptest0_sub(double x, double n)
|
|
{
|
|
double u = x / (n * n);
|
|
|
|
return(u);
|
|
}
|
|
|
|
static void fptest0(void)
|
|
{
|
|
assert(mdfn_fptest0_sub(36, 2) == 9);
|
|
}
|
|
|
|
volatile double mdfn_fptest1_v;
|
|
static void fptest1(void)
|
|
{
|
|
mdfn_fptest1_v = 1.0;
|
|
|
|
for(int i = 0; i < 128; i++)
|
|
mdfn_fptest1_v *= 2;
|
|
|
|
assert(mdfn_fptest1_v == 340282366920938463463374607431768211456.0);
|
|
}
|
|
|
|
#if defined(HAVE_FENV_H) && defined(HAVE_NEARBYINTF)
|
|
// For advisory/debug purposes, don't error out on failure.
|
|
static void libc_rounding_test(void)
|
|
{
|
|
unsigned old_rm = fegetround();
|
|
float tv = 4118966.75;
|
|
float goodres = 4118967.0;
|
|
float res;
|
|
|
|
fesetround(FE_TONEAREST);
|
|
|
|
if((res = nearbyintf(tv)) != goodres)
|
|
fprintf(stderr, "\n***** Buggy libc nearbyintf() detected(%f != %f). *****\n\n", res, goodres);
|
|
|
|
fesetround(old_rm);
|
|
}
|
|
#else
|
|
static void libc_rounding_test(void)
|
|
{
|
|
|
|
}
|
|
#endif
|
|
|
|
static int pow_test_sub_a(int y, double z) NO_INLINE NO_CLONE;
|
|
static int pow_test_sub_a(int y, double z)
|
|
{
|
|
return std::min<int>(floor(pow(10, z)), std::min<int>(floor(pow(10, y)), (int)pow(10, y)));
|
|
}
|
|
|
|
static int pow_test_sub_b(int y) NO_INLINE NO_CLONE;
|
|
static int pow_test_sub_b(int y)
|
|
{
|
|
return std::min<int>(floor(pow(2, y)), (int)pow(2, y));
|
|
}
|
|
|
|
static void pow_test(void)
|
|
{
|
|
unsigned muller10 = 1;
|
|
unsigned muller2 = 1;
|
|
|
|
for(int y = 0; y < 10; y++, muller10 *= 10, muller2 <<= 1)
|
|
{
|
|
unsigned res10 = pow_test_sub_a(y, y);
|
|
unsigned res2 = pow_test_sub_b(y);
|
|
|
|
//printf("%u %u\n", res10, res2);
|
|
|
|
assert(res10 == muller10);
|
|
assert(res2 == muller2);
|
|
}
|
|
}
|
|
|
|
static void RunFPTests(void)
|
|
{
|
|
fptest0();
|
|
fptest1();
|
|
|
|
libc_rounding_test();
|
|
pow_test();
|
|
}
|
|
|
|
#if 0
|
|
static void NO_CLONE NO_INLINE ThreadSub(int tv)
|
|
{
|
|
throw MDFN_Error(tv, "%d\n", tv);
|
|
}
|
|
|
|
|
|
static int ThreadTestEntry(void* data)
|
|
{
|
|
const uint32 st = *(uint32*)data;
|
|
|
|
while(MDFND_GetTime() < st)
|
|
{
|
|
try
|
|
{
|
|
ThreadSub(rand());
|
|
}
|
|
catch(MDFN_Error &e)
|
|
{
|
|
assert(e.GetErrno() == atoi(e.what()));
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void RunThreadTests(void)
|
|
{
|
|
MDFN_Thread *a, *b, *c, *d;
|
|
uint32 t = MDFND_GetTime() + 5000;
|
|
|
|
a = MDFND_CreateThread(ThreadTestEntry, &t);
|
|
b = MDFND_CreateThread(ThreadTestEntry, &t);
|
|
c = MDFND_CreateThread(ThreadTestEntry, &t);
|
|
d = MDFND_CreateThread(ThreadTestEntry, &t);
|
|
|
|
MDFND_WaitThread(a, NULL);
|
|
MDFND_WaitThread(b, NULL);
|
|
MDFND_WaitThread(c, NULL);
|
|
MDFND_WaitThread(d, NULL);
|
|
}
|
|
#endif
|
|
|
|
#ifdef WANT_TEST_ZLIB
|
|
static void zlib_test(void)
|
|
{
|
|
auto cfl = zlibCompileFlags();
|
|
|
|
assert((2 << ((cfl >> 0) & 0x3)) == sizeof(uInt));
|
|
assert((2 << ((cfl >> 2) & 0x3)) == sizeof(uLong));
|
|
assert((2 << ((cfl >> 4) & 0x3)) == sizeof(voidpf));
|
|
|
|
#ifdef Z_LARGE64
|
|
if((2 << ((cfl >> 6) & 0x3)) != sizeof(z_off_t))
|
|
{
|
|
assert(sizeof(z_off64_t) == 8);
|
|
assert(&gztell == &gztell64);
|
|
}
|
|
#else
|
|
assert((2 << ((cfl >> 6) & 0x3)) == sizeof(z_off_t));
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
const char* MDFN_tests_stringA = "AB\0C";
|
|
const char* MDFN_tests_stringB = "AB\0CD";
|
|
const char* MDFN_tests_stringC = "AB\0X";
|
|
|
|
}
|
|
|
|
using namespace MDFN_TESTS_CPP;
|
|
|
|
bool MDFN_RunMathTests(void)
|
|
{
|
|
MathTestEntry *itoo = math_test_vals;
|
|
|
|
if(!DoSizeofTests())
|
|
return(0);
|
|
|
|
assert(MDFN_tests_stringA != MDFN_tests_stringB && MDFN_tests_stringA[3] == 'C' && MDFN_tests_stringB[4] == 'D');
|
|
assert(MDFN_tests_stringA != MDFN_tests_stringC && MDFN_tests_stringB != MDFN_tests_stringC && MDFN_tests_stringC[3] == 'X');
|
|
|
|
// Make sure the "char" type is signed(pass -fsigned-char to gcc). New code in Mednafen shouldn't be written with the
|
|
// assumption that "char" is signed, but there likely is at least some code that does.
|
|
{
|
|
char tmp = 255;
|
|
assert(tmp < 0);
|
|
}
|
|
|
|
#if 0
|
|
// TODO(except for 32-bit >> 32 test)
|
|
{
|
|
uint8 test_cow8 = (uint8)0xFF >> mdfn_shifty_test[1];
|
|
uint16 test_cow16 = (uint16)0xFFFF >> mdfn_shifty_test[2];
|
|
uint32 test_cow32 = (uint32)0xFFFFFFFF >> mdfn_shifty_test[3];
|
|
uint32 test_cow32_2 = (uint32)0xFFFFFFFF >> mdfn_shifty_test[0];
|
|
|
|
printf("%08x\n", test_cow32);
|
|
|
|
assert(test_cow8 == 0);
|
|
assert(test_cow16 == 0);
|
|
assert(test_cow32 == 0);
|
|
assert(test_cow32_2 == 0xFFFFFFFF);
|
|
}
|
|
#endif
|
|
|
|
{
|
|
int32 meow;
|
|
|
|
meow = 1;
|
|
meow >>= 1;
|
|
assert(meow == 0);
|
|
|
|
meow = 5;
|
|
meow >>= 1;
|
|
assert(meow == 2);
|
|
|
|
meow = -1;
|
|
meow >>= 1;
|
|
assert(meow == -1);
|
|
|
|
meow = -5;
|
|
meow >>= 1;
|
|
assert(meow == -3);
|
|
|
|
meow = 1;
|
|
meow /= 2;
|
|
assert(meow == 0);
|
|
|
|
meow = 5;
|
|
meow /= 2;
|
|
assert(meow == 2);
|
|
|
|
meow = -1;
|
|
meow /= 2;
|
|
assert(meow == 0);
|
|
|
|
meow = -5;
|
|
meow /= 2;
|
|
assert(meow == -2);
|
|
|
|
meow = -5;
|
|
meow = (int32)(meow + ((uint32)meow >> 31)) >> 1;
|
|
assert(meow == -2);
|
|
|
|
#if 0
|
|
meow = 1 << 30;
|
|
meow <<= 1;
|
|
assert(meow == -2147483648);
|
|
|
|
meow = 1 << 31;
|
|
meow <<= 1;
|
|
assert(meow == 0);
|
|
#endif
|
|
}
|
|
|
|
|
|
// New tests added May 22, 2010 to detect MSVC compiler(and possibly other compilers) bad code generation.
|
|
{
|
|
uint32 test_tab[4] = { 0x2000 | 0x1000, 0x2000, 0x1000, 0x0000 };
|
|
const uint32 result_tab[4][2] = { { 0xE, 0x7 }, { 0xE, 0x0 }, { 0x0, 0x7 }, { 0x0, 0x0 } };
|
|
|
|
for(int i = 0; i < 4; i++)
|
|
{
|
|
uint32 hflip_xor;
|
|
uint32 vflip_xor;
|
|
uint32 bgsc;
|
|
|
|
bgsc = test_tab[i];
|
|
|
|
hflip_xor = ((int32)(bgsc << 18) >> 30) & 0xE;
|
|
vflip_xor = ((int32)(bgsc << 19) >> 31) & 0x7;
|
|
|
|
assert(hflip_xor == result_tab[i][0]);
|
|
assert(vflip_xor == result_tab[i][1]);
|
|
|
|
//printf("%d %d\n", hflip_xor, result_tab[i][0]);
|
|
//printf("%d %d\n", vflip_xor, result_tab[i][1]);
|
|
}
|
|
|
|
uint32 lfsr = 1;
|
|
|
|
// quick and dirty RNG(to also test non-constant-expression evaluation, at least until compilers are extremely advanced :b)
|
|
for(int i = 0; i < 256; i++)
|
|
{
|
|
int feedback = ((lfsr >> 7) & 1) ^ ((lfsr >> 14) & 1);
|
|
lfsr = ((lfsr << 1) & 0x7FFF) | feedback;
|
|
|
|
uint32 hflip_xor;
|
|
uint32 vflip_xor;
|
|
uint32 hflip_xor_alt;
|
|
uint32 vflip_xor_alt;
|
|
uint32 bgsc;
|
|
|
|
bgsc = lfsr;
|
|
|
|
hflip_xor = ((int32)(bgsc << 18) >> 30) & 0xE;
|
|
vflip_xor = ((int32)(bgsc << 19) >> 31) & 0x7;
|
|
|
|
hflip_xor_alt = bgsc & 0x2000 ? 0xE : 0;
|
|
vflip_xor_alt = bgsc & 0x1000 ? 0x7 : 0;
|
|
|
|
assert(hflip_xor == hflip_xor_alt);
|
|
assert(vflip_xor == vflip_xor_alt);
|
|
}
|
|
|
|
}
|
|
|
|
DoAlignmentChecks();
|
|
TestSignedOverflow();
|
|
TestDefinedOverShift();
|
|
TestBoolConv();
|
|
TestNarrowConstFold();
|
|
|
|
TestGCC60196();
|
|
|
|
TestModTern();
|
|
TestBWNotMask31GTZ();
|
|
TestTernary();
|
|
TestLLVM15470();
|
|
|
|
TestSUCompare();
|
|
|
|
if(sign_9_to_s16(itoo->negative_one) != -1 || sign_9_to_s16(itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_10_to_s16(itoo->negative_one) != -1 || sign_10_to_s16(itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_11_to_s16(itoo->negative_one) != -1 || sign_11_to_s16(itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_12_to_s16(itoo->negative_one) != -1 || sign_12_to_s16(itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_13_to_s16(itoo->negative_one) != -1 || sign_13_to_s16(itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_14_to_s16(itoo->negative_one) != -1 || sign_14_to_s16(itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_15_to_s16(itoo->negative_one) != -1 || sign_15_to_s16(itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(17, itoo->negative_one) != -1 || sign_x_to_s32(17, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(18, itoo->negative_one) != -1 || sign_x_to_s32(18, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(19, itoo->negative_one) != -1 || sign_x_to_s32(19, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(20, itoo->negative_one) != -1 || sign_x_to_s32(20, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(21, itoo->negative_one) != -1 || sign_x_to_s32(21, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(22, itoo->negative_one) != -1 || sign_x_to_s32(22, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(23, itoo->negative_one) != -1 || sign_x_to_s32(23, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(24, itoo->negative_one) != -1 || sign_x_to_s32(24, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(25, itoo->negative_one) != -1 || sign_x_to_s32(25, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(26, itoo->negative_one) != -1 || sign_x_to_s32(26, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(27, itoo->negative_one) != -1 || sign_x_to_s32(27, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(28, itoo->negative_one) != -1 || sign_x_to_s32(28, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(29, itoo->negative_one) != -1 || sign_x_to_s32(29, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(30, itoo->negative_one) != -1 || sign_x_to_s32(30, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sign_x_to_s32(31, itoo->negative_one) != -1 || sign_x_to_s32(31, itoo->mostneg) != itoo->mostnegresult)
|
|
FATALME;
|
|
itoo++;
|
|
|
|
if(sizeof(int8) != 1 || sizeof(uint8) != 1)
|
|
FATALME;
|
|
|
|
|
|
if(!DoAntiNSOBugTest())
|
|
return(0);
|
|
|
|
DoAntiNSOBugTest2014();
|
|
|
|
#ifdef WANT_TEST_LEPACKER
|
|
if(!DoLEPackerTest())
|
|
return(0);
|
|
#endif
|
|
|
|
assert(uilog2(0) == 0);
|
|
assert(uilog2(1) == 0);
|
|
assert(uilog2(3) == 1);
|
|
assert(uilog2(4095) == 11);
|
|
assert(uilog2(0xFFFFFFFF) == 31);
|
|
|
|
RunFPTests();
|
|
|
|
RunMASMemTests();
|
|
|
|
#ifdef WANT_TEST_EXCEPTION
|
|
RunExceptionTests();
|
|
#endif
|
|
|
|
//RunThreadTests();
|
|
|
|
RunSTLTests();
|
|
|
|
LZCount_Test();
|
|
|
|
#ifdef WANT_TEST_HASHES
|
|
sha1_test();
|
|
sha256_test();
|
|
#endif
|
|
|
|
#ifdef WANT_TEST_ZLIB
|
|
zlib_test();
|
|
#endif
|
|
|
|
#if 0
|
|
// Not really a math test.
|
|
const char *test_paths[] = { "/meow", "/meow/cow", "\\meow", "\\meow\\cow", "\\\\meow", "\\\\meow\\cow",
|
|
"/meow.", "/me.ow/cow.", "\\meow.", "\\me.ow\\cow.", "\\\\meow.", "\\\\meow\\cow.",
|
|
"/meow.txt", "/me.ow/cow.txt", "\\meow.txt", "\\me.ow\\cow.txt", "\\\\meow.txt", "\\\\meow\\cow.txt"
|
|
|
|
"/meow", "/meow\\cow", "\\meow", "\\meow/cow", "\\\\meow", "\\\\meow/cow",
|
|
"/meow.", "\\me.ow/cow.", "\\meow.", "/me.ow\\cow.", "\\\\meow.", "\\\\meow/cow.",
|
|
"/meow.txt", "/me.ow\\cow.txt", "\\meow.txt", "\\me.ow/cow.txt", "\\\\meow.txt", "\\\\meow/cow.txt",
|
|
"/bark///dog", "\\bark\\\\\\dog" };
|
|
|
|
for(unsigned i = 0; i < sizeof(test_paths) / sizeof(const char *); i++)
|
|
{
|
|
std::string file_path = std::string(test_paths[i]);
|
|
std::string dir_path;
|
|
std::string file_base;
|
|
std::string file_ext;
|
|
|
|
MDFN_GetFilePathComponents(file_path, &dir_path, &file_base, &file_ext);
|
|
|
|
printf("%s ------ dir=%s --- base=%s --- ext=%s\n", file_path.c_str(), dir_path.c_str(), file_base.c_str(), file_ext.c_str());
|
|
|
|
}
|
|
#endif
|
|
|
|
|
|
return(1);
|
|
}
|