mirror of
https://github.com/mangosfour/server.git
synced 2025-12-17 16:37:00 +00:00
Initial Mangos Three Commit
This commit is contained in:
parent
bb91aa5933
commit
7665a09232
2444 changed files with 625144 additions and 0 deletions
193
dep/tomlib/Math/src/bn_mp_ilogb.c
Normal file
193
dep/tomlib/Math/src/bn_mp_ilogb.c
Normal file
|
|
@ -0,0 +1,193 @@
|
|||
#include "tommath_private.h"
|
||||
#ifdef BN_MP_ILOGB_C
|
||||
/* LibTomMath, multiple-precision integer library -- Tom St Denis */
|
||||
/* SPDX-License-Identifier: Unlicense */
|
||||
|
||||
/* Compute log_{base}(a) */
|
||||
static mp_word s_pow(mp_word base, mp_word exponent)
|
||||
{
|
||||
mp_word result = 1uLL;
|
||||
while (exponent != 0u) {
|
||||
if ((exponent & 1u) == 1u) {
|
||||
result *= base;
|
||||
}
|
||||
exponent >>= 1;
|
||||
base *= base;
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
static mp_digit s_digit_ilogb(mp_digit base, mp_digit n)
|
||||
{
|
||||
mp_word bracket_low = 1uLL, bracket_mid, bracket_high, N;
|
||||
mp_digit ret, high = 1uL, low = 0uL, mid;
|
||||
|
||||
if (n < base) {
|
||||
return (mp_digit)0uL;
|
||||
}
|
||||
if (n == base) {
|
||||
return (mp_digit)1uL;
|
||||
}
|
||||
|
||||
bracket_high = (mp_word) base ;
|
||||
N = (mp_word) n;
|
||||
|
||||
while (bracket_high < N) {
|
||||
low = high;
|
||||
bracket_low = bracket_high;
|
||||
high <<= 1;
|
||||
bracket_high *= bracket_high;
|
||||
}
|
||||
|
||||
while (((mp_digit)(high - low)) > 1uL) {
|
||||
mid = (low + high) >> 1;
|
||||
bracket_mid = bracket_low * s_pow(base, mid - low) ;
|
||||
|
||||
if (N < bracket_mid) {
|
||||
high = mid ;
|
||||
bracket_high = bracket_mid ;
|
||||
}
|
||||
if (N > bracket_mid) {
|
||||
low = mid ;
|
||||
bracket_low = bracket_mid ;
|
||||
}
|
||||
if (N == bracket_mid) {
|
||||
return (mp_digit) mid;
|
||||
}
|
||||
}
|
||||
|
||||
if (bracket_high == N) {
|
||||
ret = high;
|
||||
} else {
|
||||
ret = low;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* TODO: output could be "int" because the output of mp_radix_size is int, too,
|
||||
as is the output of mp_bitcount.
|
||||
With the same problem: max size is INT_MAX * MP_DIGIT not INT_MAX only!
|
||||
*/
|
||||
int mp_ilogb(mp_int *a, mp_digit base, mp_int *c)
|
||||
{
|
||||
int err, cmp;
|
||||
unsigned int high, low, mid;
|
||||
mp_int bracket_low, bracket_high, bracket_mid, t, bi_base;
|
||||
mp_digit tmp;
|
||||
|
||||
err = MP_OKAY;
|
||||
if (a->sign == MP_NEG) {
|
||||
return MP_VAL;
|
||||
}
|
||||
if (MP_IS_ZERO(a)) {
|
||||
return MP_VAL;
|
||||
}
|
||||
|
||||
if (base < 2u) {
|
||||
return MP_VAL;
|
||||
}
|
||||
if (base == 2u) {
|
||||
cmp = mp_count_bits(a) - 1;
|
||||
if ((err = mp_set_int(c, (unsigned long)cmp)) != MP_OKAY) {
|
||||
goto LBL_ERR;
|
||||
}
|
||||
return err;
|
||||
}
|
||||
if (a->used == 1) {
|
||||
tmp = s_digit_ilogb(base, a->dp[0]);
|
||||
mp_set(c, tmp);
|
||||
return err;
|
||||
}
|
||||
|
||||
cmp = mp_cmp_d(a, base);
|
||||
|
||||
if (cmp == MP_LT) {
|
||||
mp_zero(c);
|
||||
return err;
|
||||
}
|
||||
if (cmp == MP_EQ) {
|
||||
mp_set(c, (mp_digit)1uL);
|
||||
return err;
|
||||
}
|
||||
|
||||
if ((err =
|
||||
mp_init_multi(&bracket_low, &bracket_high,
|
||||
&bracket_mid, &t, &bi_base, NULL)) != MP_OKAY) {
|
||||
return err;
|
||||
}
|
||||
|
||||
low = 0u;
|
||||
mp_set(&bracket_low, 1uL);
|
||||
high = 1u;
|
||||
|
||||
mp_set(&bracket_high, base);
|
||||
|
||||
/*
|
||||
A kind of Giant-step/baby-step algorithm.
|
||||
Idea shamelessly stolen from https://programmingpraxis.com/2010/05/07/integer-logarithms/2/
|
||||
The effect is asymptotic, hence needs benchmarks to test if the Giant-step should be skipped
|
||||
for small n.
|
||||
*/
|
||||
while (mp_cmp(&bracket_high, a) == MP_LT) {
|
||||
low = high;
|
||||
if ((err = mp_copy(&bracket_high, &bracket_low)) != MP_OKAY) {
|
||||
goto LBL_ERR;
|
||||
}
|
||||
high <<= 1;
|
||||
if ((err = mp_sqr(&bracket_high, &bracket_high)) != MP_OKAY) {
|
||||
goto LBL_ERR;
|
||||
}
|
||||
}
|
||||
mp_set(&bi_base, base);
|
||||
|
||||
while ((high - low) > 1u) {
|
||||
mid = (high + low) >> 1;
|
||||
/* Difference can be larger then the type behind mp_digit can hold */
|
||||
if ((mid - low) > (unsigned int)(MP_MASK)) {
|
||||
err = MP_VAL;
|
||||
goto LBL_ERR;
|
||||
}
|
||||
if ((err = mp_expt_d(&bi_base, (mp_digit)(mid - low), &t)) != MP_OKAY) {
|
||||
goto LBL_ERR;
|
||||
}
|
||||
if ((err = mp_mul(&bracket_low, &t, &bracket_mid)) != MP_OKAY) {
|
||||
goto LBL_ERR;
|
||||
}
|
||||
cmp = mp_cmp(a, &bracket_mid);
|
||||
if (cmp == MP_LT) {
|
||||
high = mid;
|
||||
mp_exch(&bracket_mid, &bracket_high);
|
||||
}
|
||||
if (cmp == MP_GT) {
|
||||
low = mid;
|
||||
mp_exch(&bracket_mid, &bracket_low);
|
||||
}
|
||||
if (cmp == MP_EQ) {
|
||||
if ((err = mp_set_int(c, (unsigned long)mid)) != MP_OKAY) {
|
||||
goto LBL_ERR;
|
||||
}
|
||||
goto LBL_END;
|
||||
}
|
||||
}
|
||||
|
||||
if (mp_cmp(&bracket_high, a) == MP_EQ) {
|
||||
if ((err = mp_set_int(c, (unsigned long)high)) != MP_OKAY) {
|
||||
goto LBL_ERR;
|
||||
}
|
||||
} else {
|
||||
if ((err = mp_set_int(c, (unsigned long)low)) != MP_OKAY) {
|
||||
goto LBL_ERR;
|
||||
}
|
||||
}
|
||||
|
||||
LBL_END:
|
||||
LBL_ERR:
|
||||
mp_clear_multi(&bracket_low, &bracket_high, &bracket_mid,
|
||||
&t, &bi_base, NULL);
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
#endif
|
||||
Loading…
Add table
Add a link
Reference in a new issue