2019-02-22 16:48:26 +00:00
|
|
|
/*
|
|
|
|
Copyright (c) 2019 tevador
|
|
|
|
|
|
|
|
This file is part of RandomX.
|
|
|
|
|
|
|
|
RandomX is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
RandomX is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with RandomX. If not, see<http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "reciprocal.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
Calculates rcp = 2**x / divisor for highest integer x such that rcp < 2**64.
|
2019-05-10 14:43:24 +00:00
|
|
|
divisor must not be 0 or a power of 2
|
2019-02-22 16:48:26 +00:00
|
|
|
|
|
|
|
Equivalent x86 assembly (divisor in rcx):
|
|
|
|
|
|
|
|
mov edx, 1
|
|
|
|
mov r8, rcx
|
|
|
|
xor eax, eax
|
|
|
|
bsr rcx, rcx
|
|
|
|
shl rdx, cl
|
|
|
|
div r8
|
|
|
|
ret
|
|
|
|
|
|
|
|
*/
|
2019-04-20 14:53:06 +00:00
|
|
|
uint64_t randomx_reciprocal(uint64_t divisor) {
|
2019-02-22 16:48:26 +00:00
|
|
|
|
|
|
|
const uint64_t p2exp63 = 1ULL << 63;
|
|
|
|
|
|
|
|
uint64_t quotient = p2exp63 / divisor, remainder = p2exp63 % divisor;
|
|
|
|
|
|
|
|
unsigned bsr = 0; //highest set bit in divisor
|
|
|
|
|
|
|
|
for (uint64_t bit = divisor; bit > 0; bit >>= 1)
|
|
|
|
bsr++;
|
|
|
|
|
|
|
|
for (unsigned shift = 0; shift < bsr; shift++) {
|
|
|
|
if (remainder >= divisor - remainder) {
|
|
|
|
quotient = quotient * 2 + 1;
|
|
|
|
remainder = remainder * 2 - divisor;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
quotient = quotient * 2;
|
|
|
|
remainder = remainder * 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return quotient;
|
2019-05-04 17:40:25 +00:00
|
|
|
}
|