Commit 475b2420 authored by Milo Yip's avatar Milo Yip

Minor refactoring before optimization trial

parent 5171775d
...@@ -29,7 +29,7 @@ ...@@ -29,7 +29,7 @@
namespace rapidjson { namespace rapidjson {
namespace internal { namespace internal {
inline double StrtodFastPath(double significand, int exp) { inline double FastPath(double significand, int exp) {
if (exp < -308) if (exp < -308)
return 0.0; return 0.0;
else if (exp >= 0) else if (exp >= 0)
...@@ -38,14 +38,14 @@ inline double StrtodFastPath(double significand, int exp) { ...@@ -38,14 +38,14 @@ inline double StrtodFastPath(double significand, int exp) {
return significand / internal::Pow10(-exp); return significand / internal::Pow10(-exp);
} }
inline double NormalPrecision(double d, int p) { inline double StrtodNormalPrecision(double d, int p) {
if (p < -308) { if (p < -308) {
// Prevent expSum < -308, making Pow10(p) = 0 // Prevent expSum < -308, making Pow10(p) = 0
d = StrtodFastPath(d, -308); d = FastPath(d, -308);
d = StrtodFastPath(d, p + 308); d = FastPath(d, p + 308);
} }
else else
d = StrtodFastPath(d, p); d = FastPath(d, p);
return d; return d;
} }
...@@ -124,25 +124,24 @@ inline int CheckWithinHalfULP(double b, const BigInteger& d, int dExp, bool* adj ...@@ -124,25 +124,24 @@ inline int CheckWithinHalfULP(double b, const BigInteger& d, int dExp, bool* adj
return cmp; return cmp;
} }
inline double FullPrecision(double d, int p, const char* decimals, size_t length, size_t decimalPosition, int exp) { inline bool StrtodFast(double d, int p, double* result) {
RAPIDJSON_ASSERT(d >= 0.0);
RAPIDJSON_ASSERT(length >= 1);
// Use fast path for string-to-double conversion if possible // Use fast path for string-to-double conversion if possible
// see http://www.exploringbinary.com/fast-path-decimal-to-floating-point-conversion/ // see http://www.exploringbinary.com/fast-path-decimal-to-floating-point-conversion/
if (p > 22) { if (p > 22 && p < 22 + 16) {
if (p < 22 + 16) {
// Fast Path Cases In Disguise // Fast Path Cases In Disguise
d *= internal::Pow10(p - 22); d *= internal::Pow10(p - 22);
p = 22; p = 22;
} }
}
if (p >= -22 && p <= 22 && d <= 9007199254740991.0) // 2^53 - 1 if (p >= -22 && p <= 22 && d <= 9007199254740991.0) { // 2^53 - 1
return StrtodFastPath(d, p); *result = FastPath(d, p);
return true;
// Use slow-path with BigInteger comparison }
else
return false;
}
inline double StrtodBigInteger(double d, int p, const char* decimals, size_t length, size_t decimalPosition, int exp) {
// Trim leading zeros // Trim leading zeros
while (*decimals == '0' && length > 1) { while (*decimals == '0' && length > 1) {
length--; length--;
...@@ -170,7 +169,7 @@ inline double FullPrecision(double d, int p, const char* decimals, size_t length ...@@ -170,7 +169,7 @@ inline double FullPrecision(double d, int p, const char* decimals, size_t length
const BigInteger dInt(decimals, length); const BigInteger dInt(decimals, length);
const int dExp = (int)decimalPosition - (int)length + exp; const int dExp = (int)decimalPosition - (int)length + exp;
Double approx = NormalPrecision(d, p); Double approx = StrtodNormalPrecision(d, p);
for (int i = 0; i < 10; i++) { for (int i = 0; i < 10; i++) {
bool adjustToNegative; bool adjustToNegative;
int cmp = CheckWithinHalfULP(approx.Value(), dInt, dExp, &adjustToNegative); int cmp = CheckWithinHalfULP(approx.Value(), dInt, dExp, &adjustToNegative);
...@@ -191,6 +190,17 @@ inline double FullPrecision(double d, int p, const char* decimals, size_t length ...@@ -191,6 +190,17 @@ inline double FullPrecision(double d, int p, const char* decimals, size_t length
return approx.Value(); return approx.Value();
} }
inline double StrtodFullPrecision(double d, int p, const char* decimals, size_t length, size_t decimalPosition, int exp) {
RAPIDJSON_ASSERT(d >= 0.0);
RAPIDJSON_ASSERT(length >= 1);
double result;
if (StrtodFast(d, p, &result))
return result;
return StrtodBigInteger(d, p, decimals, length, decimalPosition, exp);
}
} // namespace internal } // namespace internal
} // namespace rapidjson } // namespace rapidjson
......
...@@ -941,9 +941,9 @@ private: ...@@ -941,9 +941,9 @@ private:
if (useDouble) { if (useDouble) {
int p = exp + expFrac; int p = exp + expFrac;
if (parseFlags & kParseFullPrecisionFlag) if (parseFlags & kParseFullPrecisionFlag)
d = internal::FullPrecision(d, p, decimal, length, decimalPosition, exp); d = internal::StrtodFullPrecision(d, p, decimal, length, decimalPosition, exp);
else else
d = internal::NormalPrecision(d, p); d = internal::StrtodNormalPrecision(d, p);
cont = handler.Double(minus ? -d : d); cont = handler.Double(minus ? -d : d);
} }
......
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