Commit 036947f6 authored by NickFengIBM's avatar NickFengIBM Committed by Feng Xiao

re-write int128 long division to avoid license impact from stackoverflow references (#4633)

* rewrite int128 long divison to avoid stackoverflow hit

Protobuf was showing Stackoverflow hits in the code base, primarily code written to calculate long division. This code was copied from a stackoverflow post, which means it would be licensed under CC BY-SA 3.0. Due to this license, IBM Legal did not want to include this OSS in our products and advised us to re-write this particular piece of code to avoid the license restriction. We have re-written the code for our own distribution, and are willing to merge it into the main code base for others who want to avoid the stackoverflow license issues to benefit as well.
parent fb33d88e
...@@ -76,52 +76,36 @@ static inline int Fls128(uint128 n) { ...@@ -76,52 +76,36 @@ static inline int Fls128(uint128 n) {
return Fls64(Uint128Low64(n)); return Fls64(Uint128Low64(n));
} }
// Long division/modulo for uint128 implemented using the shift-subtract
// division algorithm adapted from:
// http://stackoverflow.com/questions/5386377/division-without-using
void uint128::DivModImpl(uint128 dividend, uint128 divisor, void uint128::DivModImpl(uint128 dividend, uint128 divisor,
uint128* quotient_ret, uint128* remainder_ret) { uint128* quotient_ret, uint128* remainder_ret) {
if (divisor == 0) { if (divisor == 0) {
GOOGLE_LOG(FATAL) << "Division or mod by zero: dividend.hi=" << dividend.hi_ GOOGLE_LOG(FATAL) << "Division or mod by zero: dividend.hi=" << dividend.hi_
<< ", lo=" << dividend.lo_; << ", lo=" << dividend.lo_;
} } else if (dividend < divisor) {
if (divisor > dividend) {
*quotient_ret = 0; *quotient_ret = 0;
*remainder_ret = dividend; *remainder_ret = dividend;
return; return;
} } else {
int dividend_bit_length = Fls128(dividend);
if (divisor == dividend) { int divisor_bit_length = Fls128(divisor);
*quotient_ret = 1; int difference = dividend_bit_length - divisor_bit_length;
*remainder_ret = 0;
return;
}
uint128 denominator = divisor;
uint128 position = 1;
uint128 quotient = 0; uint128 quotient = 0;
while (difference >= 0) {
// Left aligns the MSB of the denominator and the dividend. quotient <<= 1;
int shift = Fls128(dividend) - Fls128(denominator); uint128 shifted_divisor = divisor << difference;
denominator <<= shift; if (shifted_divisor <= dividend) {
position <<= shift; dividend -= shifted_divisor;
quotient += 1;
// Uses shift-subtract algorithm to divide dividend by denominator. The
// remainder will be left in dividend.
while (position > 0) {
if (dividend >= denominator) {
dividend -= denominator;
quotient |= position;
} }
position >>= 1; difference -= 1;
denominator >>= 1;
} }
//record the final quotient and remainder
*quotient_ret = quotient; *quotient_ret = quotient;
*remainder_ret = dividend; *remainder_ret = dividend;
}
} }
uint128& uint128::operator/=(const uint128& divisor) { uint128& uint128::operator/=(const uint128& divisor) {
uint128 quotient = 0; uint128 quotient = 0;
uint128 remainder = 0; uint128 remainder = 0;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment