2018-11-05 07:28:01 +00:00
|
|
|
// "Build Engine & Tools" Copyright (c) 1993-1997 Ken Silverman
|
|
|
|
// Ken Silverman's official web site: "http://www.advsys.net/ken"
|
|
|
|
// See the included license file "BUILDLIC.TXT" for license info.
|
|
|
|
//
|
2006-04-13 20:47:06 +00:00
|
|
|
// This file has been modified from Ken Silverman's original release
|
2012-03-12 04:47:04 +00:00
|
|
|
// by Jonathon Fowler (jf@jonof.id.au)
|
2018-11-05 07:28:01 +00:00
|
|
|
// by the EDuke32 team (development@voidpoint.com)
|
2006-04-13 20:47:06 +00:00
|
|
|
|
2014-11-22 12:32:56 +00:00
|
|
|
#ifndef pragmas_h_
|
|
|
|
#define pragmas_h_
|
2006-04-13 20:47:06 +00:00
|
|
|
|
2006-07-02 17:33:49 +00:00
|
|
|
|
2019-06-25 11:28:37 +00:00
|
|
|
#define EDUKE32_GENERATE_PRAGMAS \
|
|
|
|
EDUKE32_SCALER_PRAGMA(1) EDUKE32_SCALER_PRAGMA(2) EDUKE32_SCALER_PRAGMA(3) EDUKE32_SCALER_PRAGMA(4) \
|
|
|
|
EDUKE32_SCALER_PRAGMA(5) EDUKE32_SCALER_PRAGMA(6) EDUKE32_SCALER_PRAGMA(7) EDUKE32_SCALER_PRAGMA(8) \
|
|
|
|
EDUKE32_SCALER_PRAGMA(9) EDUKE32_SCALER_PRAGMA(10) EDUKE32_SCALER_PRAGMA(11) EDUKE32_SCALER_PRAGMA(12) \
|
|
|
|
EDUKE32_SCALER_PRAGMA(13) EDUKE32_SCALER_PRAGMA(14) EDUKE32_SCALER_PRAGMA(15) EDUKE32_SCALER_PRAGMA(16) \
|
|
|
|
EDUKE32_SCALER_PRAGMA(17) EDUKE32_SCALER_PRAGMA(18) EDUKE32_SCALER_PRAGMA(19) EDUKE32_SCALER_PRAGMA(20) \
|
|
|
|
EDUKE32_SCALER_PRAGMA(21) EDUKE32_SCALER_PRAGMA(22) EDUKE32_SCALER_PRAGMA(23) EDUKE32_SCALER_PRAGMA(24) \
|
|
|
|
EDUKE32_SCALER_PRAGMA(25) EDUKE32_SCALER_PRAGMA(26) EDUKE32_SCALER_PRAGMA(27) EDUKE32_SCALER_PRAGMA(28) \
|
2014-11-02 05:35:22 +00:00
|
|
|
EDUKE32_SCALER_PRAGMA(29) EDUKE32_SCALER_PRAGMA(30) EDUKE32_SCALER_PRAGMA(31)
|
2014-09-30 04:04:53 +00:00
|
|
|
|
2014-10-25 03:30:38 +00:00
|
|
|
extern int32_t reciptable[2048], fpuasm;
|
2006-04-13 20:47:06 +00:00
|
|
|
|
2009-09-09 07:19:14 +00:00
|
|
|
// break the C version of divscale out from the others
|
|
|
|
// because asm version overflows in drawmapview()
|
|
|
|
|
2014-11-02 05:35:22 +00:00
|
|
|
#define qw(x) ((int64_t)(x)) // quadword cast
|
|
|
|
#define dw(x) ((int32_t)(x)) // doubleword cast
|
|
|
|
#define wo(x) ((int16_t)(x)) // word cast
|
|
|
|
#define by(x) ((uint8_t)(x)) // byte cast
|
2009-09-09 07:19:14 +00:00
|
|
|
|
2014-10-25 03:29:21 +00:00
|
|
|
#define DIVTABLESIZE 16384
|
2014-08-23 10:28:17 +00:00
|
|
|
|
2019-06-25 11:29:50 +00:00
|
|
|
extern libdivide::libdivide_s64_t divtable64[DIVTABLESIZE];
|
|
|
|
extern libdivide::libdivide_s32_t divtable32[DIVTABLESIZE];
|
2015-09-27 21:18:12 +00:00
|
|
|
extern void initdivtables(void);
|
2014-10-25 03:29:21 +00:00
|
|
|
|
2018-10-25 23:33:52 +00:00
|
|
|
static inline uint32_t divideu32(uint32_t const n, uint32_t const d)
|
2012-05-01 12:37:32 +00:00
|
|
|
{
|
2019-06-25 11:29:50 +00:00
|
|
|
static libdivide::libdivide_u32_t udiv;
|
2014-10-25 03:29:21 +00:00
|
|
|
static uint32_t lastd;
|
|
|
|
|
|
|
|
if (d == lastd)
|
|
|
|
goto skip;
|
|
|
|
|
2019-06-25 11:29:50 +00:00
|
|
|
udiv = libdivide::libdivide_u32_gen((lastd = d));
|
2014-10-25 03:29:21 +00:00
|
|
|
skip:
|
2019-06-25 11:29:50 +00:00
|
|
|
return libdivide::libdivide_u32_do(n, &udiv);
|
2012-05-01 12:37:32 +00:00
|
|
|
}
|
|
|
|
|
2019-07-24 01:37:28 +00:00
|
|
|
static inline int64_t tabledivide64(int64_t const n, int64_t const d)
|
2014-10-25 03:29:21 +00:00
|
|
|
{
|
2019-06-25 11:29:50 +00:00
|
|
|
static libdivide::libdivide_s64_t sdiv;
|
2014-10-25 03:29:21 +00:00
|
|
|
static int32_t lastd;
|
2018-12-15 01:39:51 +00:00
|
|
|
auto const dptr = ((unsigned)d < DIVTABLESIZE) ? &divtable64[d] : &sdiv;
|
2012-05-01 12:37:32 +00:00
|
|
|
|
2014-10-25 03:29:21 +00:00
|
|
|
if (d == lastd || dptr != &sdiv)
|
|
|
|
goto skip;
|
|
|
|
|
2019-06-25 11:29:50 +00:00
|
|
|
sdiv = libdivide::libdivide_s64_gen((lastd = d));
|
2014-10-25 03:29:21 +00:00
|
|
|
skip:
|
2019-06-25 11:29:50 +00:00
|
|
|
return libdivide::libdivide_s64_do(n, dptr);
|
2014-10-25 03:29:21 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 23:33:52 +00:00
|
|
|
static inline int32_t tabledivide32(int32_t const n, int32_t const d)
|
2014-10-25 03:29:21 +00:00
|
|
|
{
|
2019-06-25 11:29:50 +00:00
|
|
|
static libdivide::libdivide_s32_t sdiv;
|
2014-10-25 03:29:21 +00:00
|
|
|
static int32_t lastd;
|
2018-12-15 01:39:51 +00:00
|
|
|
auto const dptr = ((unsigned)d < DIVTABLESIZE) ? &divtable32[d] : &sdiv;
|
2014-10-25 03:29:21 +00:00
|
|
|
|
|
|
|
if (d == lastd || dptr != &sdiv)
|
|
|
|
goto skip;
|
|
|
|
|
2019-06-25 11:29:50 +00:00
|
|
|
sdiv = libdivide::libdivide_s32_gen((lastd = d));
|
2014-10-25 03:29:21 +00:00
|
|
|
skip:
|
2019-06-25 11:29:50 +00:00
|
|
|
return libdivide::libdivide_s32_do(n, dptr);
|
2014-10-25 03:29:21 +00:00
|
|
|
}
|
2012-05-01 12:37:32 +00:00
|
|
|
|
2014-10-25 03:29:21 +00:00
|
|
|
extern uint32_t divideu32_noinline(uint32_t n, uint32_t d);
|
|
|
|
extern int32_t tabledivide32_noinline(int32_t n, int32_t d);
|
2019-07-24 01:37:28 +00:00
|
|
|
extern int64_t tabledivide64_noinline(int64_t n, int64_t d);
|
2009-09-09 07:19:14 +00:00
|
|
|
|
2014-10-25 03:29:21 +00:00
|
|
|
#ifdef GEKKO
|
2019-07-24 01:37:50 +00:00
|
|
|
static inline int32_t divscale(int32_t eax, int32_t ebx, int32_t ecx) { return dw(tabledivide64(ldexp(eax, ecx), ebx)); }
|
2014-10-25 03:29:21 +00:00
|
|
|
#else
|
2019-06-25 11:28:37 +00:00
|
|
|
static inline int32_t divscale(int32_t eax, int32_t ebx, int32_t ecx) { return dw(tabledivide64(qw(eax) << by(ecx), ebx)); }
|
2014-10-25 03:29:21 +00:00
|
|
|
#endif
|
|
|
|
|
2019-07-24 01:37:50 +00:00
|
|
|
static inline int64_t divscale64(int64_t eax, int64_t ebx, int64_t ecx) { return tabledivide64(eax << ecx, ebx); }
|
|
|
|
|
2018-10-25 23:33:52 +00:00
|
|
|
#define EDUKE32_SCALER_PRAGMA(a) \
|
2017-02-25 08:15:53 +00:00
|
|
|
static FORCE_INLINE int32_t divscale##a(int32_t eax, int32_t ebx) { return divscale(eax, ebx, a); }
|
2014-10-29 17:03:29 +00:00
|
|
|
EDUKE32_GENERATE_PRAGMAS EDUKE32_SCALER_PRAGMA(32)
|
|
|
|
#undef EDUKE32_SCALER_PRAGMA
|
2014-10-25 03:29:21 +00:00
|
|
|
|
|
|
|
static inline int32_t scale(int32_t eax, int32_t edx, int32_t ecx)
|
|
|
|
{
|
2019-06-25 11:28:37 +00:00
|
|
|
return dw(tabledivide64(qw(eax) * edx, ecx));
|
2014-10-25 03:29:21 +00:00
|
|
|
}
|
2009-09-09 07:19:14 +00:00
|
|
|
|
2019-06-25 11:28:37 +00:00
|
|
|
static FORCE_INLINE int32_t scaleadd(int32_t eax, int32_t edx, int32_t addend, int32_t ecx)
|
2017-07-29 20:39:53 +00:00
|
|
|
{
|
2019-06-25 11:28:37 +00:00
|
|
|
return dw(tabledivide64(qw(eax) * edx + addend, ecx));
|
2017-07-29 20:39:53 +00:00
|
|
|
}
|
|
|
|
|
2019-06-25 11:28:37 +00:00
|
|
|
static inline int32_t roundscale(int32_t eax, int32_t edx, int32_t ecx)
|
2017-07-29 20:39:53 +00:00
|
|
|
{
|
|
|
|
return scaleadd(eax, edx, ecx / 2, ecx);
|
|
|
|
}
|
|
|
|
|
2006-04-13 20:47:06 +00:00
|
|
|
|
2006-07-01 01:40:18 +00:00
|
|
|
//
|
|
|
|
// Generic C
|
|
|
|
//
|
|
|
|
|
2017-01-23 11:21:22 +00:00
|
|
|
#ifndef pragmas_have_mulscale
|
|
|
|
|
2018-10-25 23:33:52 +00:00
|
|
|
#define EDUKE32_SCALER_PRAGMA(a) \
|
|
|
|
static FORCE_INLINE CONSTEXPR int32_t mulscale##a(int32_t eax, int32_t edx) { return dw((qw(eax) * edx) >> by(a)); } \
|
|
|
|
static FORCE_INLINE CONSTEXPR int32_t dmulscale##a(int32_t eax, int32_t edx, int32_t esi, int32_t edi) \
|
|
|
|
{ \
|
|
|
|
return dw(((qw(eax) * edx) + (qw(esi) * edi)) >> by(a)); \
|
|
|
|
} \
|
|
|
|
static FORCE_INLINE CONSTEXPR int32_t tmulscale##a(int32_t eax, int32_t edx, int32_t ebx, int32_t ecx, int32_t esi, int32_t edi) \
|
|
|
|
{ \
|
|
|
|
return dw(((qw(eax) * edx) + (qw(ebx) * ecx) + (qw(esi) * edi)) >> by(a)); \
|
|
|
|
}
|
2014-11-02 05:35:22 +00:00
|
|
|
|
2014-10-29 17:03:29 +00:00
|
|
|
EDUKE32_GENERATE_PRAGMAS EDUKE32_SCALER_PRAGMA(32)
|
2014-09-30 04:04:53 +00:00
|
|
|
|
2014-10-29 17:03:29 +00:00
|
|
|
#undef EDUKE32_SCALER_PRAGMA
|
2006-07-01 01:40:18 +00:00
|
|
|
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
|
|
|
|
2017-06-24 09:20:46 +00:00
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
static FORCE_INLINE void swap(T * const a, T * const b)
|
|
|
|
{
|
|
|
|
T const t = *a;
|
|
|
|
*a = *b;
|
|
|
|
*b = t;
|
|
|
|
}
|
|
|
|
#define swapptr swap
|
|
|
|
|
2017-01-23 11:21:22 +00:00
|
|
|
#ifndef pragmas_have_swaps
|
2017-06-24 09:20:46 +00:00
|
|
|
#define swapchar swap
|
|
|
|
#define swapshort swap
|
|
|
|
#define swaplong swap
|
|
|
|
#define swapfloat swap
|
|
|
|
#define swapdouble swap
|
|
|
|
#define swap64bit swap
|
2019-09-08 19:19:23 +00:00
|
|
|
|
2017-06-24 09:20:46 +00:00
|
|
|
static FORCE_INLINE void swapchar2(void *a, void *b, int32_t s)
|
|
|
|
{
|
|
|
|
swapchar((char *)a, (char *)b);
|
|
|
|
swapchar((char *)a + 1, (char *)b + s);
|
|
|
|
}
|
|
|
|
#endif
|
2006-07-01 01:40:18 +00:00
|
|
|
|
2018-10-25 23:33:52 +00:00
|
|
|
static FORCE_INLINE CONSTEXPR char readpixel(void *s) { return *(char *)s; }
|
2019-09-09 22:03:43 +00:00
|
|
|
|
2006-07-01 01:40:18 +00:00
|
|
|
|
2017-01-23 11:21:22 +00:00
|
|
|
#ifndef pragmas_have_klabs
|
2019-08-01 06:50:13 +00:00
|
|
|
#if 0
|
|
|
|
static FORCE_INLINE int32_t klabs(int32_t const a)
|
2014-11-02 05:35:22 +00:00
|
|
|
{
|
2019-08-01 06:50:13 +00:00
|
|
|
uint32_t const m = a >> (sizeof(uint32_t) * CHAR_BIT - 1);
|
2014-11-02 05:35:22 +00:00
|
|
|
return (a ^ m) - m;
|
|
|
|
}
|
2019-08-01 06:50:13 +00:00
|
|
|
#else
|
|
|
|
#define klabs(x) abs(x)
|
|
|
|
#endif
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
|
|
|
#ifndef pragmas_have_ksgn
|
2019-08-01 06:50:17 +00:00
|
|
|
static FORCE_INLINE CONSTEXPR int ksgn(int32_t a) { return (a > 0) - (a < 0); }
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
2006-07-01 01:40:18 +00:00
|
|
|
|
2017-01-23 11:21:22 +00:00
|
|
|
#ifndef pragmas_have_mulscale
|
2018-10-25 23:33:52 +00:00
|
|
|
static FORCE_INLINE CONSTEXPR int32_t mulscale(int32_t eax, int32_t edx, int32_t ecx) { return dw((qw(eax) * edx) >> by(ecx)); }
|
|
|
|
static FORCE_INLINE CONSTEXPR int32_t dmulscale(int32_t eax, int32_t edx, int32_t esi, int32_t edi, int32_t ecx)
|
2014-11-02 05:35:22 +00:00
|
|
|
{
|
|
|
|
return dw(((qw(eax) * edx) + (qw(esi) * edi)) >> by(ecx));
|
|
|
|
}
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
2006-07-01 01:40:18 +00:00
|
|
|
|
2017-01-23 11:21:22 +00:00
|
|
|
#ifndef pragmas_have_qinterpolatedown16
|
2014-11-02 05:35:22 +00:00
|
|
|
void qinterpolatedown16(intptr_t bufptr, int32_t num, int32_t val, int32_t add);
|
|
|
|
void qinterpolatedown16short(intptr_t bufptr, int32_t num, int32_t val, int32_t add);
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
2006-07-01 01:40:18 +00:00
|
|
|
|
2017-01-23 11:21:22 +00:00
|
|
|
#ifndef pragmas_have_clearbuf
|
2014-11-02 05:35:22 +00:00
|
|
|
void clearbuf(void *d, int32_t c, int32_t a);
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
|
|
|
#ifndef pragmas_have_copybuf
|
2014-11-02 05:35:22 +00:00
|
|
|
void copybuf(const void *s, void *d, int32_t c);
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
|
|
|
#ifndef pragmas_have_swaps
|
2014-11-02 05:35:22 +00:00
|
|
|
void swapbuf4(void *a, void *b, int32_t c);
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
2006-07-01 01:40:18 +00:00
|
|
|
|
2017-01-23 11:21:22 +00:00
|
|
|
#ifndef pragmas_have_clearbufbyte
|
2009-01-09 09:29:17 +00:00
|
|
|
void clearbufbyte(void *D, int32_t c, int32_t a);
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
|
|
|
#ifndef pragmas_have_copybufbyte
|
2012-02-18 22:14:45 +00:00
|
|
|
void copybufbyte(const void *S, void *D, int32_t c);
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
|
|
|
#ifndef pragmas_have_copybufreverse
|
2012-02-18 22:14:45 +00:00
|
|
|
void copybufreverse(const void *S, void *D, int32_t c);
|
2017-01-23 11:21:22 +00:00
|
|
|
#endif
|
2006-04-13 20:47:06 +00:00
|
|
|
|
2017-01-23 11:21:22 +00:00
|
|
|
#ifndef pragmas_have_krecipasm
|
2014-10-25 03:30:38 +00:00
|
|
|
static inline int32_t krecipasm(int32_t i)
|
|
|
|
{
|
|
|
|
// Ken did this
|
2018-07-23 02:55:53 +00:00
|
|
|
union { int32_t i; float f; } x;
|
|
|
|
x.f = (float)i;
|
|
|
|
i = x.i;
|
2014-11-02 05:35:22 +00:00
|
|
|
return ((reciptable[(i >> 12) & 2047] >> (((i - 0x3f800000) >> 23) & 31)) ^ (i >> 31));
|
2014-10-25 03:30:38 +00:00
|
|
|
}
|
2006-04-13 20:47:06 +00:00
|
|
|
#endif
|
|
|
|
|
2014-09-30 04:04:53 +00:00
|
|
|
#undef qw
|
|
|
|
#undef dw
|
|
|
|
#undef wo
|
|
|
|
#undef by
|
|
|
|
|
2014-03-05 21:12:59 +00:00
|
|
|
static inline void swapbufreverse(void *s, void *d, int32_t c)
|
|
|
|
{
|
2014-11-02 05:35:22 +00:00
|
|
|
uint8_t *src = (uint8_t *)s, *dst = (uint8_t *)d;
|
2015-02-17 16:23:04 +00:00
|
|
|
Bassert(c >= 4);
|
|
|
|
|
2014-10-25 03:29:21 +00:00
|
|
|
do
|
|
|
|
{
|
|
|
|
swapchar(dst, src);
|
2014-11-02 05:35:22 +00:00
|
|
|
swapchar(dst + 1, src - 1);
|
|
|
|
swapchar(dst + 2, src - 2);
|
|
|
|
swapchar(dst + 3, src - 3);
|
2014-10-25 03:29:21 +00:00
|
|
|
dst += 4, src -= 4;
|
2015-02-17 16:23:04 +00:00
|
|
|
} while ((c -= 4) > 4);
|
|
|
|
|
|
|
|
while (c--)
|
|
|
|
swapchar(dst++, src--);
|
2014-03-05 21:12:59 +00:00
|
|
|
}
|
|
|
|
|
2006-07-02 17:33:49 +00:00
|
|
|
|
2014-11-22 12:32:56 +00:00
|
|
|
#endif // pragmas_h_
|