2024-02-24 22:35:25 +05:30
|
|
|
/*
|
|
|
|
* opt_int_div.h
|
|
|
|
*
|
2024-02-25 03:39:36 +05:30
|
|
|
* "Optimized integer division"
|
|
|
|
*
|
2024-02-24 22:35:25 +05:30
|
|
|
* Author: Intel A80486DX2-66
|
|
|
|
* License: Creative Commons Zero 1.0 Universal
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <math.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
// parameters
|
2024-03-09 21:59:11 +05:30
|
|
|
#define OPT_INT_DIV_IS_TEST_PRECISION_LIMITED 0 /* true (1) or false (0) */
|
|
|
|
#define OPT_INT_DIV_TEST_PRECISION 8 /* only if the above parameter is 1 */
|
2024-02-24 22:35:25 +05:30
|
|
|
|
|
|
|
// constants
|
|
|
|
#define OPT_INT_DIV_ROUNDING \
|
2024-03-09 19:20:35 +05:30
|
|
|
/* formula: 10^OPT_INT_DIV_TEST_PRECISION */ \
|
2024-03-09 19:19:33 +05:30
|
|
|
powl(10.l, (long double) OPT_INT_DIV_TEST_PRECISION)
|
2024-02-24 22:35:25 +05:30
|
|
|
|
|
|
|
// helper functions
|
|
|
|
#define INT_BIN_DIV(a, b) ((a) >> (uintmax_t) log2l((b)))
|
2024-03-09 20:18:05 +05:30
|
|
|
#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 22:35:25 +05:30
|
|
|
#define INT_ABS(x) ((x) < 0 ? -(x) : (x))
|
2024-03-09 21:59:11 +05:30
|
|
|
#define LOG2_DEC_PORTION(b) fmodl(log2l(INT_ABS((b))), 1.l)
|
|
|
|
#define OPT_INT_DIV_TEST(b) \
|
|
|
|
/* check if b is a power of 2 */ \
|
|
|
|
/* */ \
|
|
|
|
/* formula: round(log_2(b) % 1, OPT_INT_DIV_TEST_PRECISION digits */ \
|
|
|
|
/* after point) == 0 */ \
|
|
|
|
OPT_INT_DIV_IS_TEST_PRECISION_LIMITED ? \
|
|
|
|
roundl(LOG2_DEC_PORTION((b)) * OPT_INT_DIV_ROUNDING) == 0.l \
|
|
|
|
: \
|
|
|
|
LOG2_DEC_PORTION((b)) == 0.l
|
2024-02-24 22:35:25 +05:30
|
|
|
|
|
|
|
// the main macro
|
|
|
|
#define OPT_INT_DIV(a, b) \
|
|
|
|
( /* beginning */ \
|
|
|
|
/* check for equality of a and b */ \
|
|
|
|
INT_ABS((a)) == INT_ABS((b)) ? \
|
2024-03-09 20:18:05 +05:30
|
|
|
/* the result is +/-1 */ \
|
2024-02-24 22:35:25 +05:30
|
|
|
(INT_DIV_NEG_RESULT_SIGN(a, b) ? -1 : 1) : ( \
|
|
|
|
\
|
2024-03-09 20:21:22 +05:30
|
|
|
/* check if abs(a) < abs(b) */ \
|
|
|
|
INT_ABS((a)) < INT_ABS((b)) ? \
|
|
|
|
0 : ( \
|
|
|
|
\
|
2024-03-09 21:59:11 +05:30
|
|
|
OPT_INT_DIV_TEST((b)) ? \
|
|
|
|
(INT_DIV_NEG_RESULT_SIGN(a, b) ? \
|
2024-03-09 20:25:41 +05:30
|
|
|
-INT_BIN_DIV(INT_ABS((a)), INT_ABS((b))) \
|
2024-02-24 22:35:25 +05:30
|
|
|
: \
|
2024-03-09 20:25:41 +05:30
|
|
|
INT_BIN_DIV(INT_ABS((a)), INT_ABS((b)))) \
|
2024-02-24 22:35:25 +05:30
|
|
|
: \
|
2024-03-09 20:21:22 +05:30
|
|
|
((a) / (b)) )) \
|
2024-02-24 22:35:25 +05:30
|
|
|
) /* end */
|