Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in / Register
Toggle navigation
R
rapidjson
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Packages
Packages
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
submodule
rapidjson
Commits
6cc3910a
Unverified
Commit
6cc3910a
authored
Jun 17, 2018
by
Milo Yip
Committed by
GitHub
Jun 17, 2018
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #1290 from abolz/fix-strtod
Fix strtod
parents
01c71740
7101911d
Expand all
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
95 additions
and
52 deletions
+95
-52
biginteger.h
include/rapidjson/internal/biginteger.h
+1
-1
diyfp.h
include/rapidjson/internal/diyfp.h
+23
-11
strtod.h
include/rapidjson/internal/strtod.h
+59
-38
reader.h
include/rapidjson/reader.h
+7
-2
bigintegertest.cpp
test/unittest/bigintegertest.cpp
+5
-0
readertest.cpp
test/unittest/readertest.cpp
+0
-0
No files found.
include/rapidjson/internal/biginteger.h
View file @
6cc3910a
...
@@ -133,7 +133,7 @@ public:
...
@@ -133,7 +133,7 @@ public:
RAPIDJSON_ASSERT
(
count_
+
offset
<=
kCapacity
);
RAPIDJSON_ASSERT
(
count_
+
offset
<=
kCapacity
);
if
(
interShift
==
0
)
{
if
(
interShift
==
0
)
{
std
::
memmove
(
&
digits_
[
count_
-
1
+
offset
],
&
digits_
[
count_
-
1
]
,
count_
*
sizeof
(
Type
));
std
::
memmove
(
digits_
+
offset
,
digits_
,
count_
*
sizeof
(
Type
));
count_
+=
offset
;
count_
+=
offset
;
}
}
else
{
else
{
...
...
include/rapidjson/internal/diyfp.h
View file @
6cc3910a
// Tencent is pleased to support the open source community by making RapidJSON available.
// Tencent is pleased to support the open source community by making RapidJSON available.
//
//
// Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
// Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
//
//
// Licensed under the MIT License (the "License"); you may not use this file except
// Licensed under the MIT License (the "License"); you may not use this file except
...
@@ -7,9 +7,9 @@
...
@@ -7,9 +7,9 @@
//
//
// http://opensource.org/licenses/MIT
// http://opensource.org/licenses/MIT
//
//
// Unless required by applicable law or agreed to in writing, software distributed
// Unless required by applicable law or agreed to in writing, software distributed
// under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
// under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
// CONDITIONS OF ANY KIND, either express or implied. See the License for the
// CONDITIONS OF ANY KIND, either express or implied. See the License for the
// specific language governing permissions and limitations under the License.
// specific language governing permissions and limitations under the License.
// This is a C++ header-only implementation of Grisu2 algorithm from the publication:
// This is a C++ header-only implementation of Grisu2 algorithm from the publication:
...
@@ -20,6 +20,7 @@
...
@@ -20,6 +20,7 @@
#define RAPIDJSON_DIYFP_H_
#define RAPIDJSON_DIYFP_H_
#include "../rapidjson.h"
#include "../rapidjson.h"
#include <limits>
#if defined(_MSC_VER) && defined(_M_AMD64) && !defined(__INTEL_COMPILER)
#if defined(_MSC_VER) && defined(_M_AMD64) && !defined(__INTEL_COMPILER)
#include <intrin.h>
#include <intrin.h>
...
@@ -56,7 +57,7 @@ struct DiyFp {
...
@@ -56,7 +57,7 @@ struct DiyFp {
if
(
biased_e
!=
0
)
{
if
(
biased_e
!=
0
)
{
f
=
significand
+
kDpHiddenBit
;
f
=
significand
+
kDpHiddenBit
;
e
=
biased_e
-
kDpExponentBias
;
e
=
biased_e
-
kDpExponentBias
;
}
}
else
{
else
{
f
=
significand
;
f
=
significand
;
e
=
kDpMinExponent
+
1
;
e
=
kDpMinExponent
+
1
;
...
@@ -141,7 +142,16 @@ struct DiyFp {
...
@@ -141,7 +142,16 @@ struct DiyFp {
double
d
;
double
d
;
uint64_t
u64
;
uint64_t
u64
;
}
u
;
}
u
;
const
uint64_t
be
=
(
e
==
kDpDenormalExponent
&&
(
f
&
kDpHiddenBit
)
==
0
)
?
0
:
RAPIDJSON_ASSERT
(
f
<=
kDpHiddenBit
+
kDpSignificandMask
);
if
(
e
<
kDpDenormalExponent
)
{
// Underflow.
return
0
.
0
;
}
if
(
e
>=
kDpMaxExponent
)
{
// Overflow.
return
std
::
numeric_limits
<
double
>::
infinity
();
}
const
uint64_t
be
=
(
e
==
kDpDenormalExponent
&&
(
f
&
kDpHiddenBit
)
==
0
)
?
0
:
static_cast
<
uint64_t
>
(
e
+
kDpExponentBias
);
static_cast
<
uint64_t
>
(
e
+
kDpExponentBias
);
u
.
u64
=
(
f
&
kDpSignificandMask
)
|
(
be
<<
kDpSignificandSize
);
u
.
u64
=
(
f
&
kDpSignificandMask
)
|
(
be
<<
kDpSignificandSize
);
return
u
.
d
;
return
u
.
d
;
...
@@ -220,9 +230,10 @@ inline DiyFp GetCachedPowerByIndex(size_t index) {
...
@@ -220,9 +230,10 @@ inline DiyFp GetCachedPowerByIndex(size_t index) {
641
,
667
,
694
,
720
,
747
,
774
,
800
,
827
,
853
,
880
,
641
,
667
,
694
,
720
,
747
,
774
,
800
,
827
,
853
,
880
,
907
,
933
,
960
,
986
,
1013
,
1039
,
1066
907
,
933
,
960
,
986
,
1013
,
1039
,
1066
};
};
RAPIDJSON_ASSERT
(
index
<
87
);
return
DiyFp
(
kCachedPowers_F
[
index
],
kCachedPowers_E
[
index
]);
return
DiyFp
(
kCachedPowers_F
[
index
],
kCachedPowers_E
[
index
]);
}
}
inline
DiyFp
GetCachedPower
(
int
e
,
int
*
K
)
{
inline
DiyFp
GetCachedPower
(
int
e
,
int
*
K
)
{
//int k = static_cast<int>(ceil((-61 - e) * 0.30102999566398114)) + 374;
//int k = static_cast<int>(ceil((-61 - e) * 0.30102999566398114)) + 374;
...
@@ -238,10 +249,11 @@ inline DiyFp GetCachedPower(int e, int* K) {
...
@@ -238,10 +249,11 @@ inline DiyFp GetCachedPower(int e, int* K) {
}
}
inline
DiyFp
GetCachedPower10
(
int
exp
,
int
*
outExp
)
{
inline
DiyFp
GetCachedPower10
(
int
exp
,
int
*
outExp
)
{
unsigned
index
=
(
static_cast
<
unsigned
>
(
exp
)
+
348u
)
/
8u
;
RAPIDJSON_ASSERT
(
exp
>=
-
348
);
*
outExp
=
-
348
+
static_cast
<
int
>
(
index
)
*
8
;
unsigned
index
=
static_cast
<
unsigned
>
(
exp
+
348
)
/
8u
;
return
GetCachedPowerByIndex
(
index
);
*
outExp
=
-
348
+
static_cast
<
int
>
(
index
)
*
8
;
}
return
GetCachedPowerByIndex
(
index
);
}
#ifdef __GNUC__
#ifdef __GNUC__
RAPIDJSON_DIAG_POP
RAPIDJSON_DIAG_POP
...
...
include/rapidjson/internal/strtod.h
View file @
6cc3910a
...
@@ -19,6 +19,8 @@
...
@@ -19,6 +19,8 @@
#include "biginteger.h"
#include "biginteger.h"
#include "diyfp.h"
#include "diyfp.h"
#include "pow10.h"
#include "pow10.h"
#include <climits>
#include <limits>
RAPIDJSON_NAMESPACE_BEGIN
RAPIDJSON_NAMESPACE_BEGIN
namespace
internal
{
namespace
internal
{
...
@@ -126,20 +128,20 @@ inline bool StrtodFast(double d, int p, double* result) {
...
@@ -126,20 +128,20 @@ inline bool StrtodFast(double d, int p, double* result) {
}
}
// Compute an approximation and see if it is within 1/2 ULP
// Compute an approximation and see if it is within 1/2 ULP
inline
bool
StrtodDiyFp
(
const
char
*
decimals
,
size_t
length
,
size_t
decimalPosition
,
int
e
xp
,
double
*
result
)
{
inline
bool
StrtodDiyFp
(
const
char
*
decimals
,
int
dLen
,
int
dE
xp
,
double
*
result
)
{
uint64_t
significand
=
0
;
uint64_t
significand
=
0
;
size_
t
i
=
0
;
// 2^64 - 1 = 18446744073709551615, 1844674407370955161 = 0x1999999999999999
in
t
i
=
0
;
// 2^64 - 1 = 18446744073709551615, 1844674407370955161 = 0x1999999999999999
for
(;
i
<
length
;
i
++
)
{
for
(;
i
<
dLen
;
i
++
)
{
if
(
significand
>
RAPIDJSON_UINT64_C2
(
0x19999999
,
0x99999999
)
||
if
(
significand
>
RAPIDJSON_UINT64_C2
(
0x19999999
,
0x99999999
)
||
(
significand
==
RAPIDJSON_UINT64_C2
(
0x19999999
,
0x99999999
)
&&
decimals
[
i
]
>
'5'
))
(
significand
==
RAPIDJSON_UINT64_C2
(
0x19999999
,
0x99999999
)
&&
decimals
[
i
]
>
'5'
))
break
;
break
;
significand
=
significand
*
10u
+
static_cast
<
unsigned
>
(
decimals
[
i
]
-
'0'
);
significand
=
significand
*
10u
+
static_cast
<
unsigned
>
(
decimals
[
i
]
-
'0'
);
}
}
if
(
i
<
length
&&
decimals
[
i
]
>=
'5'
)
// Rounding
if
(
i
<
dLen
&&
decimals
[
i
]
>=
'5'
)
// Rounding
significand
++
;
significand
++
;
size_t
remaining
=
length
-
i
;
int
remaining
=
dLen
-
i
;
const
int
kUlpShift
=
3
;
const
int
kUlpShift
=
3
;
const
int
kUlp
=
1
<<
kUlpShift
;
const
int
kUlp
=
1
<<
kUlpShift
;
int64_t
error
=
(
remaining
==
0
)
?
0
:
kUlp
/
2
;
int64_t
error
=
(
remaining
==
0
)
?
0
:
kUlp
/
2
;
...
@@ -148,24 +150,24 @@ inline bool StrtodDiyFp(const char* decimals, size_t length, size_t decimalPosit
...
@@ -148,24 +150,24 @@ inline bool StrtodDiyFp(const char* decimals, size_t length, size_t decimalPosit
v
=
v
.
Normalize
();
v
=
v
.
Normalize
();
error
<<=
-
v
.
e
;
error
<<=
-
v
.
e
;
const
int
dExp
=
static_cast
<
int
>
(
decimalPosition
)
-
static_cast
<
int
>
(
i
)
+
exp
;
dExp
+=
remaining
;
int
actualExp
;
int
actualExp
;
DiyFp
cachedPower
=
GetCachedPower10
(
dExp
,
&
actualExp
);
DiyFp
cachedPower
=
GetCachedPower10
(
dExp
,
&
actualExp
);
if
(
actualExp
!=
dExp
)
{
if
(
actualExp
!=
dExp
)
{
static
const
DiyFp
kPow10
[]
=
{
static
const
DiyFp
kPow10
[]
=
{
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xa0000000
,
00000000
),
-
60
),
// 10^1
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xa0000000
,
0
x0
0000000
),
-
60
),
// 10^1
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xc8000000
,
00000000
),
-
57
),
// 10^2
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xc8000000
,
0
x0
0000000
),
-
57
),
// 10^2
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xfa000000
,
00000000
),
-
54
),
// 10^3
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xfa000000
,
0
x0
0000000
),
-
54
),
// 10^3
DiyFp
(
RAPIDJSON_UINT64_C2
(
0x9c400000
,
00000000
),
-
50
),
// 10^4
DiyFp
(
RAPIDJSON_UINT64_C2
(
0x9c400000
,
0
x0
0000000
),
-
50
),
// 10^4
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xc3500000
,
00000000
),
-
47
),
// 10^5
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xc3500000
,
0
x0
0000000
),
-
47
),
// 10^5
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xf4240000
,
00000000
),
-
44
),
// 10^6
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xf4240000
,
0
x0
0000000
),
-
44
),
// 10^6
DiyFp
(
RAPIDJSON_UINT64_C2
(
0x98968000
,
00000000
),
-
40
)
// 10^7
DiyFp
(
RAPIDJSON_UINT64_C2
(
0x98968000
,
0
x0
0000000
),
-
40
)
// 10^7
};
};
int
adjustment
=
dExp
-
actualExp
-
1
;
int
adjustment
=
dExp
-
actualExp
;
RAPIDJSON_ASSERT
(
adjustment
>=
0
&&
adjustment
<
7
);
RAPIDJSON_ASSERT
(
adjustment
>=
1
&&
adjustment
<
8
);
v
=
v
*
kPow10
[
adjustment
];
v
=
v
*
kPow10
[
adjustment
-
1
];
if
(
length
+
static_cast
<
unsigned
>
(
adjustment
)
>
19u
)
// has more digits than decimal digits in 64-bit
if
(
dLen
+
adjustment
>
19
)
// has more digits than decimal digits in 64-bit
error
+=
kUlp
/
2
;
error
+=
kUlp
/
2
;
}
}
...
@@ -203,9 +205,9 @@ inline bool StrtodDiyFp(const char* decimals, size_t length, size_t decimalPosit
...
@@ -203,9 +205,9 @@ inline bool StrtodDiyFp(const char* decimals, size_t length, size_t decimalPosit
return
halfWay
-
static_cast
<
unsigned
>
(
error
)
>=
precisionBits
||
precisionBits
>=
halfWay
+
static_cast
<
unsigned
>
(
error
);
return
halfWay
-
static_cast
<
unsigned
>
(
error
)
>=
precisionBits
||
precisionBits
>=
halfWay
+
static_cast
<
unsigned
>
(
error
);
}
}
inline
double
StrtodBigInteger
(
double
approx
,
const
char
*
decimals
,
size_t
length
,
size_t
decimalPosition
,
int
e
xp
)
{
inline
double
StrtodBigInteger
(
double
approx
,
const
char
*
decimals
,
int
dLen
,
int
dE
xp
)
{
const
BigInteger
dInt
(
decimals
,
length
);
RAPIDJSON_ASSERT
(
dLen
>=
0
);
const
int
dExp
=
static_cast
<
int
>
(
decimalPosition
)
-
static_cast
<
int
>
(
length
)
+
exp
;
const
BigInteger
dInt
(
decimals
,
static_cast
<
unsigned
>
(
dLen
))
;
Double
a
(
approx
);
Double
a
(
approx
);
int
cmp
=
CheckWithinHalfULP
(
a
.
Value
(),
dInt
,
dExp
);
int
cmp
=
CheckWithinHalfULP
(
a
.
Value
(),
dInt
,
dExp
);
if
(
cmp
<
0
)
if
(
cmp
<
0
)
...
@@ -225,42 +227,61 @@ inline double StrtodFullPrecision(double d, int p, const char* decimals, size_t
...
@@ -225,42 +227,61 @@ inline double StrtodFullPrecision(double d, int p, const char* decimals, size_t
RAPIDJSON_ASSERT
(
d
>=
0
.
0
);
RAPIDJSON_ASSERT
(
d
>=
0
.
0
);
RAPIDJSON_ASSERT
(
length
>=
1
);
RAPIDJSON_ASSERT
(
length
>=
1
);
double
result
;
double
result
=
0
.
0
;
if
(
StrtodFast
(
d
,
p
,
&
result
))
if
(
StrtodFast
(
d
,
p
,
&
result
))
return
result
;
return
result
;
RAPIDJSON_ASSERT
(
length
<=
INT_MAX
);
int
dLen
=
static_cast
<
int
>
(
length
);
RAPIDJSON_ASSERT
(
length
>=
decimalPosition
);
RAPIDJSON_ASSERT
(
length
-
decimalPosition
<=
INT_MAX
);
int
dExpAdjust
=
static_cast
<
int
>
(
length
-
decimalPosition
);
RAPIDJSON_ASSERT
(
exp
>=
INT_MIN
+
dExpAdjust
);
int
dExp
=
exp
-
dExpAdjust
;
// Make sure length+dExp does not overflow
RAPIDJSON_ASSERT
(
dExp
<=
INT_MAX
-
dLen
);
// Trim leading zeros
// Trim leading zeros
while
(
*
decimals
==
'0'
&&
length
>
1
)
{
while
(
dLen
>
0
&&
*
decimals
==
'0'
)
{
length
--
;
dLen
--
;
decimals
++
;
decimals
++
;
decimalPosition
--
;
}
}
// Trim trailing zeros
// Trim trailing zeros
while
(
decimals
[
length
-
1
]
==
'0'
&&
length
>
1
)
{
while
(
dLen
>
0
&&
decimals
[
dLen
-
1
]
==
'0'
)
{
length
--
;
dLen
--
;
decimalPosition
--
;
dExp
++
;
exp
++
;
}
if
(
dLen
==
0
)
{
// Buffer only contains zeros.
return
0
.
0
;
}
}
// Trim right-most digits
// Trim right-most digits
const
int
kMaxDecimalDigit
=
780
;
const
int
kMaxDecimalDigit
=
767
+
1
;
if
(
static_cast
<
int
>
(
length
)
>
kMaxDecimalDigit
)
{
if
(
dLen
>
kMaxDecimalDigit
)
{
int
delta
=
(
static_cast
<
int
>
(
length
)
-
kMaxDecimalDigit
);
dExp
+=
dLen
-
kMaxDecimalDigit
;
exp
+=
delta
;
dLen
=
kMaxDecimalDigit
;
decimalPosition
-=
static_cast
<
unsigned
>
(
delta
);
length
=
kMaxDecimalDigit
;
}
}
// If too small, underflow to zero
// If too small, underflow to zero.
if
(
int
(
length
)
+
exp
<
-
324
)
// Any x <= 10^-324 is interpreted as zero.
if
(
dLen
+
dExp
<=
-
324
)
return
0
.
0
;
return
0
.
0
;
if
(
StrtodDiyFp
(
decimals
,
length
,
decimalPosition
,
exp
,
&
result
))
// If too large, overflow to infinity.
// Any x >= 10^309 is interpreted as +infinity.
if
(
dLen
+
dExp
>
309
)
return
std
::
numeric_limits
<
double
>::
infinity
();
if
(
StrtodDiyFp
(
decimals
,
dLen
,
dExp
,
&
result
))
return
result
;
return
result
;
// Use approximation from StrtodDiyFp and make adjustment with BigInteger comparison
// Use approximation from StrtodDiyFp and make adjustment with BigInteger comparison
return
StrtodBigInteger
(
result
,
decimals
,
length
,
decimalPosition
,
e
xp
);
return
StrtodBigInteger
(
result
,
decimals
,
dLen
,
dE
xp
);
}
}
}
// namespace internal
}
// namespace internal
...
...
include/rapidjson/reader.h
View file @
6cc3910a
...
@@ -1561,8 +1561,6 @@ private:
...
@@ -1561,8 +1561,6 @@ private:
// Force double for big integer
// Force double for big integer
if
(
useDouble
)
{
if
(
useDouble
)
{
while
(
RAPIDJSON_LIKELY
(
s
.
Peek
()
>=
'0'
&&
s
.
Peek
()
<=
'9'
))
{
while
(
RAPIDJSON_LIKELY
(
s
.
Peek
()
>=
'0'
&&
s
.
Peek
()
<=
'9'
))
{
if
(
RAPIDJSON_UNLIKELY
(
d
>=
1.7976931348623157e307
))
// DBL_MAX / 10.0
RAPIDJSON_PARSE_ERROR
(
kParseErrorNumberTooBig
,
startOffset
);
d
=
d
*
10
+
(
s
.
TakePush
()
-
'0'
);
d
=
d
*
10
+
(
s
.
TakePush
()
-
'0'
);
}
}
}
}
...
@@ -1702,6 +1700,13 @@ private:
...
@@ -1702,6 +1700,13 @@ private:
else
else
d
=
internal
::
StrtodNormalPrecision
(
d
,
p
);
d
=
internal
::
StrtodNormalPrecision
(
d
,
p
);
// Use > max, instead of == inf, to fix bogus warning -Wfloat-equal
if
(
d
>
std
::
numeric_limits
<
double
>::
max
())
{
// Overflow
// TODO: internal::StrtodX should report overflow (or underflow)
RAPIDJSON_PARSE_ERROR
(
kParseErrorNumberTooBig
,
startOffset
);
}
cont
=
handler
.
Double
(
minus
?
-
d
:
d
);
cont
=
handler
.
Double
(
minus
?
-
d
:
d
);
}
}
else
if
(
useNanOrInf
)
{
else
if
(
useNanOrInf
)
{
...
...
test/unittest/bigintegertest.cpp
View file @
6cc3910a
...
@@ -120,6 +120,11 @@ TEST(BigInteger, LeftShift) {
...
@@ -120,6 +120,11 @@ TEST(BigInteger, LeftShift) {
EXPECT_TRUE
(
BIGINTEGER_LITERAL
(
"4537899042132549697536"
)
==
a
);
EXPECT_TRUE
(
BIGINTEGER_LITERAL
(
"4537899042132549697536"
)
==
a
);
a
<<=
99
;
a
<<=
99
;
EXPECT_TRUE
(
BIGINTEGER_LITERAL
(
"2876235222267216943024851750785644982682875244576768"
)
==
a
);
EXPECT_TRUE
(
BIGINTEGER_LITERAL
(
"2876235222267216943024851750785644982682875244576768"
)
==
a
);
a
=
1
;
a
<<=
64
;
// a.count_ != 1
a
<<=
256
;
// interShift == 0
EXPECT_TRUE
(
BIGINTEGER_LITERAL
(
"2135987035920910082395021706169552114602704522356652769947041607822219725780640550022962086936576"
)
==
a
);
}
}
TEST
(
BigInteger
,
Compare
)
{
TEST
(
BigInteger
,
Compare
)
{
...
...
test/unittest/readertest.cpp
View file @
6cc3910a
This diff is collapsed.
Click to expand it.
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment