59da940283
* [battle, damage] refactor damage formula to match gen5+ * [test] use exact values for dry skin, swarm tests * fixup: assume stats for dry-skin, swarm tests --------- Co-authored-by: sbird <sbird@no.tld>
81 lines
2.3 KiB
C
81 lines
2.3 KiB
C
#ifndef FPMATH_H_
|
|
#define FPMATH_H_
|
|
|
|
typedef s16 q4_12_t;
|
|
typedef u32 uq4_12_t;
|
|
|
|
#define Q_4_12_SHIFT (12)
|
|
#define UQ_4_12_SHIFT (12)
|
|
|
|
// Converts a number to Q8.8 fixed-point format
|
|
#define Q_8_8(n) ((s16)((n) * 256))
|
|
|
|
// Converts a number to Q4.12 fixed-point format
|
|
#define Q_4_12(n) ((q4_12_t)((n) * 4096))
|
|
#define UQ_4_12(n) ((uq4_12_t)((n) * 4096))
|
|
|
|
// Converts a number to Q24.8 fixed-point format
|
|
#define Q_24_8(n) ((s32)((n) << 8))
|
|
|
|
// Converts a Q8.8 fixed-point format number to a regular integer
|
|
#define Q_8_8_TO_INT(n) ((s32)((n) / 256))
|
|
|
|
// Converts a Q4.12 fixed-point format number to a regular integer
|
|
#define Q_4_12_TO_INT(n) ((s32)((n) / 4096))
|
|
#define UQ_4_12_TO_INT(n) ((u32)((n) / 4096))
|
|
|
|
// Converts a Q24.8 fixed-point format number to a regular integer
|
|
#define Q_24_8_TO_INT(n) ((s32)((n) >> 8))
|
|
|
|
// Rounding value for Q4.12 fixed-point format
|
|
#define Q_4_12_ROUND ((1) << (Q_4_12_SHIFT - 1))
|
|
#define UQ_4_12_ROUND ((1) << (UQ_4_12_SHIFT - 1))
|
|
|
|
// Basic arithmetic for fixed point number formats
|
|
// Consumers should use encapsulated functions where possible
|
|
|
|
// FP API does not provide sanity checks against overflows
|
|
|
|
static inline uq4_12_t uq4_12_add(uq4_12_t a, uq4_12_t b)
|
|
{
|
|
return a + b;
|
|
}
|
|
|
|
static inline uq4_12_t uq4_12_subtract(uq4_12_t a, uq4_12_t b)
|
|
{
|
|
return a - b;
|
|
}
|
|
|
|
static inline uq4_12_t uq4_12_multiply(uq4_12_t a, uq4_12_t b)
|
|
{
|
|
u32 product = (u32) a * b;
|
|
return (product + UQ_4_12_ROUND) >> UQ_4_12_SHIFT;
|
|
}
|
|
|
|
static inline uq4_12_t uq4_12_multiply_half_down(uq4_12_t a, uq4_12_t b)
|
|
{
|
|
u32 product = (u32) a * b;
|
|
return (product + UQ_4_12_ROUND - 1) >> UQ_4_12_SHIFT;
|
|
}
|
|
|
|
static inline uq4_12_t uq4_12_divide(uq4_12_t dividend, uq4_12_t divisor)
|
|
{
|
|
if (divisor == UQ_4_12(0.0)) return UQ_4_12(0);
|
|
return (dividend << UQ_4_12_SHIFT) / divisor;
|
|
}
|
|
|
|
// Multiplies value by the UQ_4_12 number modifier.
|
|
// Returns an integer, rounded to nearest (rounding down on n.5)
|
|
static inline u32 uq4_12_multiply_by_int_half_down(uq4_12_t modifier, u32 value)
|
|
{
|
|
return UQ_4_12_TO_INT((modifier * value) + UQ_4_12_ROUND - 1);
|
|
}
|
|
|
|
// Multiplies value by the UQ_4_12 number modifier.
|
|
// Returns an integer, rounded to nearest (rounding up on n.5)
|
|
static inline u32 uq4_12_multiply_by_int_half_up(uq4_12_t modifier, u32 value)
|
|
{
|
|
return UQ_4_12_TO_INT((modifier * value) + UQ_4_12_ROUND);
|
|
}
|
|
|
|
#endif // FPMATH_H_
|