1
0
mirror of https://gitlab.com/80486DX2-66/gists synced 2025-01-10 17:32:05 +05:30
gists/c-programming/math/opt_int_div.h

49 lines
2.5 KiB
C
Raw Normal View History

2024-02-24 20:05:25 +03:00
/*
* opt_int_div.h
*
* "Optimized integer division": A header-only library
2024-02-25 01:09:36 +03:00
*
2024-04-22 00:08:49 +03:00
* NOTE: This code will work only on a computer that uses bits.
*
2024-02-24 20:05:25 +03:00
* Author: Intel A80486DX2-66
2024-04-25 23:16:39 +03:00
* License: Unlicense
2024-02-24 20:05:25 +03:00
*/
#include <math.h>
#include <stdint.h>
// helper functions
#define INT_BIN_DIV(a, b) ((a) >> (uintmax_t) log2l((b))) /* NOTE: log2l may
slow things down */
#define INT_DIV_NEG_RESULT_SIGN(a, b) \
/* the sign is negative only if one of the numbers is negative */ \
(((a) < 0) != ((b) < 0)) /* 1 if sign is negative else 0 */
2024-02-24 20:05:25 +03:00
#define INT_ABS(x) ((x) < 0 ? -(x) : (x))
#define OPT_INT_DIV_TEST(b) \
/* check if b is a power of 2 */ \
/* */ \
/* condition: ( X & (X - 1) ) == 0 */ \
((b) & ((b) - 1)) == 0
2024-02-24 20:05:25 +03:00
// the main macro
#define OPT_INT_DIV(a, b) \
( /* beginning */ \
/* check for equality of abs(a) and abs(b) */ \
2024-02-24 20:05:25 +03:00
INT_ABS((a)) == INT_ABS((b)) ? \
/* the result is +/-1 */ \
2024-02-24 20:05:25 +03:00
(INT_DIV_NEG_RESULT_SIGN(a, b) ? -1 : 1) : ( \
\
/* check if abs(a) < abs(b) */ \
INT_ABS((a)) < INT_ABS((b)) ? \
/* if abs(a) < abs(b), then the result is less than 1 (i.e., 0) */\
0 : ( \
\
OPT_INT_DIV_TEST((b)) ? \
(INT_DIV_NEG_RESULT_SIGN(a, b) ? \
-INT_BIN_DIV(INT_ABS((a)), INT_ABS((b))) \
2024-02-24 20:05:25 +03:00
: \
INT_BIN_DIV(INT_ABS((a)), INT_ABS((b)))) \
2024-02-24 20:05:25 +03:00
: \
((a) / (b)) )) \
2024-02-24 20:05:25 +03:00
) /* end */