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
b4e2d58c
Commit
b4e2d58c
authored
Oct 31, 2014
by
miloyip
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Temp commit
parent
faa877ff
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
87 additions
and
34 deletions
+87
-34
biginteger.h
include/rapidjson/internal/biginteger.h
+1
-1
diyfp.h
include/rapidjson/internal/diyfp.h
+45
-23
ieee754.h
include/rapidjson/internal/ieee754.h
+9
-0
strtod.h
include/rapidjson/internal/strtod.h
+29
-9
readertest.cpp
test/unittest/readertest.cpp
+3
-1
No files found.
include/rapidjson/internal/biginteger.h
View file @
b4e2d58c
...
...
@@ -221,7 +221,7 @@ private:
if
(
IsZero
())
*
this
=
u
;
else
{
unsigned
exp
=
end
-
begin
;
unsigned
exp
=
static_cast
<
unsigned
>
(
end
-
begin
)
;
(
MultiplyPow5
(
exp
)
<<=
exp
)
+=
u
;
// *this = *this * 10^exp + u
}
}
...
...
include/rapidjson/internal/diyfp.h
View file @
b4e2d58c
...
...
@@ -45,7 +45,7 @@ struct DiyFp {
DiyFp
(
uint64_t
f
,
int
e
)
:
f
(
f
),
e
(
e
)
{}
DiyFp
(
double
d
)
{
explicit
DiyFp
(
double
d
)
{
union
{
double
d
;
uint64_t
u64
;
...
...
@@ -102,17 +102,21 @@ struct DiyFp {
unsigned
long
index
;
_BitScanReverse64
(
&
index
,
f
);
return
DiyFp
(
f
<<
(
63
-
index
),
e
-
(
63
-
index
));
#elif defined(__GNUC__)
#elif
0//
defined(__GNUC__)
int
s
=
__builtin_clzll
(
f
)
+
1
;
return
DiyFp
(
f
<<
s
,
e
-
s
);
#else
DiyFp
res
=
*
this
;
while
(
!
(
res
.
f
&
kDpHiddenBit
))
{
while
(
!
(
res
.
f
&
(
static_cast
<
uint64_t
>
(
1
)
<<
63
)
))
{
res
.
f
<<=
1
;
res
.
e
--
;
}
res
.
f
<<=
(
kDiySignificandSize
-
kDpSignificandSize
-
1
);
res
.
e
=
res
.
e
-
(
kDiySignificandSize
-
kDpSignificandSize
-
1
);
// while (!(res.f & kDpHiddenBit)) {
// res.f <<= 1;
// res.e--;
// }
// res.f <<= (kDiySignificandSize - kDpSignificandSize - 1);
// res.e = res.e - (kDiySignificandSize - kDpSignificandSize - 1);
return
res
;
#endif
}
...
...
@@ -143,10 +147,39 @@ struct DiyFp {
*
minus
=
mi
;
}
double
ToDouble
()
const
{
union
{
double
d
;
uint64_t
u64
;
}
u
;
uint64_t
significand
=
f
;
int
exponent
=
e
;
while
(
significand
>
kDpHiddenBit
+
kDpSignificandMask
)
{
significand
>>=
1
;
exponent
++
;
}
while
(
exponent
>
kDpDenormalExponent
&&
(
significand
&
kDpHiddenBit
)
==
0
)
{
significand
<<=
1
;
exponent
--
;
}
if
(
exponent
>=
kDpMaxExponent
)
{
u
.
u64
=
kDpExponentMask
;
// Infinity
return
u
.
d
;
}
else
if
(
exponent
<
kDpDenormalExponent
)
return
0
.
0
;
const
uint64_t
be
=
(
exponent
==
kDpDenormalExponent
&&
(
significand
&
kDpHiddenBit
)
==
0
)
?
0
:
static_cast
<
uint64_t
>
(
exponent
+
kDpExponentBias
);
u
.
u64
=
(
significand
&
kDpSignificandMask
)
|
(
be
<<
kDpSignificandSize
);
return
u
.
d
;
}
static
const
int
kDiySignificandSize
=
64
;
static
const
int
kDpSignificandSize
=
52
;
static
const
int
kDpExponentBias
=
0x3FF
+
kDpSignificandSize
;
static
const
int
kDpMaxExponent
=
0x7FF
-
kDpExponentBias
;
static
const
int
kDpMinExponent
=
-
kDpExponentBias
;
static
const
int
kDpDenormalExponent
=
-
kDpExponentBias
-
1
;
static
const
uint64_t
kDpExponentMask
=
RAPIDJSON_UINT64_C2
(
0x7FF00000
,
0x00000000
);
static
const
uint64_t
kDpSignificandMask
=
RAPIDJSON_UINT64_C2
(
0x000FFFFF
,
0xFFFFFFFF
);
static
const
uint64_t
kDpHiddenBit
=
RAPIDJSON_UINT64_C2
(
0x00100000
,
0x00000000
);
...
...
@@ -155,7 +188,7 @@ struct DiyFp {
int
e
;
};
inline
uint64_t
GetCachedPowerF
(
size_t
index
)
{
inline
DiyFp
GetCachedPowerByIndex
(
size_t
index
)
{
// 10^-348, 10^-340, ..., 10^340
static
const
uint64_t
kCachedPowers_F
[]
=
{
RAPIDJSON_UINT64_C2
(
0xfa8fd5a0
,
0x081c0288
),
RAPIDJSON_UINT64_C2
(
0xbaaee17f
,
0xa23ebf76
),
...
...
@@ -203,10 +236,6 @@ inline uint64_t GetCachedPowerF(size_t index) {
RAPIDJSON_UINT64_C2
(
0x9e19db92
,
0xb4e31ba9
),
RAPIDJSON_UINT64_C2
(
0xeb96bf6e
,
0xbadf77d9
),
RAPIDJSON_UINT64_C2
(
0xaf87023b
,
0x9bf0ee6b
)
};
return
kCachedPowers_F
[
index
];
}
inline
DiyFp
GetCachedPower
(
int
e
,
int
*
K
)
{
static
const
int16_t
kCachedPowers_E
[]
=
{
-
1220
,
-
1193
,
-
1166
,
-
1140
,
-
1113
,
-
1087
,
-
1060
,
-
1034
,
-
1007
,
-
980
,
-
954
,
-
927
,
-
901
,
-
874
,
-
847
,
-
821
,
-
794
,
-
768
,
-
741
,
-
715
,
...
...
@@ -218,6 +247,10 @@ inline DiyFp GetCachedPower(int e, int* K) {
641
,
667
,
694
,
720
,
747
,
774
,
800
,
827
,
853
,
880
,
907
,
933
,
960
,
986
,
1013
,
1039
,
1066
};
return
DiyFp
(
kCachedPowers_F
[
index
],
kCachedPowers_E
[
index
]);
}
inline
DiyFp
GetCachedPower
(
int
e
,
int
*
K
)
{
//int k = static_cast<int>(ceil((-61 - e) * 0.30102999566398114)) + 374;
double
dk
=
(
-
61
-
e
)
*
0
.
30102999566398114
+
347
;
// dk must be positive, so can do ceiling in positive
...
...
@@ -228,24 +261,13 @@ inline DiyFp GetCachedPower(int e, int* K) {
unsigned
index
=
static_cast
<
unsigned
>
((
k
>>
3
)
+
1
);
*
K
=
-
(
-
348
+
static_cast
<
int
>
(
index
<<
3
));
// decimal exponent no need lookup table
return
DiyFp
(
GetCachedPowerF
(
index
),
kCachedPowers_E
[
index
]
);
return
GetCachedPowerByIndex
(
index
);
}
inline
DiyFp
GetCachedPower10
(
int
exp
,
int
*
outExp
)
{
// static const int16_t kCachedPowers_E10[] = {
// -348, -340, -332, -324, -316, -308, -300, -292, -284, -276,
// -268, -260, -252, -244, -236, -228, -220, -212, -204, -196,
// -188, -180, -172, -164, -156, -148, -140, -132, -124, -116,
// -108, -100, -92, -84, -76, -68, -60, -52, -44, -36,
// -28, -20, -12, -4, 4, 12, 20, 28, 36, 44,
// 52, 60, 68, 76, 84, 92, 100, 108, 116, 124,
// 132, 140, 148, 156, 164, 172, 180, 188, 196, 204,
// 212, 220, 228, 236, 244, 252, 260, 268, 276, 284,
// 292, 300, 308, 316, 324, 332, 340
// };
unsigned
index
=
(
exp
+
348
)
/
8
;
*
outExp
=
-
348
+
index
*
8
;
return
GetCachedPower
F
(
index
);
return
GetCachedPower
ByIndex
(
index
);
}
#ifdef __GNUC__
...
...
include/rapidjson/internal/ieee754.h
View file @
b4e2d58c
...
...
@@ -60,6 +60,15 @@ public:
int
IntegerExponent
()
const
{
return
(
IsNormal
()
?
Exponent
()
:
kDenormalExponent
)
-
kSignificandSize
;
}
uint64_t
ToBias
()
const
{
return
(
u
&
kSignMask
)
?
~
u
+
1
:
u
|
kSignMask
;
}
static
unsigned
EffectiveSignificandSize
(
int
order
)
{
if
(
order
>=
kDenormalExponent
+
kSignificandSize
)
return
kSignificandSize
;
else
if
(
order
<=
kDenormalExponent
)
return
0
;
else
return
order
-
kDenormalExponent
;
}
private
:
static
const
int
kSignificandSize
=
52
;
static
const
int
kExponentBias
=
0x3FF
;
...
...
include/rapidjson/internal/strtod.h
View file @
b4e2d58c
...
...
@@ -145,8 +145,8 @@ inline bool StrtodFast(double d, int p, double* result) {
// 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
exp
,
double
*
result
)
{
uint64_t
significand
=
0
;
size_t
i
=
0
;
// 2^64 - 1 = 18446744073709551615, 1844674407370955161 = 0x19999999
0x
99999999
for
(;
i
<
length
&&
(
significand
<
=
RAPIDJSON_UINT64_C2
(
0x19999999
,
0x99999999
)
||
decimals
[
i
]
<=
'4'
);
i
++
)
size_t
i
=
0
;
// 2^64 - 1 = 18446744073709551615, 1844674407370955161 = 0x1999999999999999
for
(;
i
<
length
&&
(
significand
<
RAPIDJSON_UINT64_C2
(
0x19999999
,
0x99999999
)
||
decimals
[
i
]
<=
'4'
);
i
++
)
significand
=
significand
*
10
+
(
decimals
[
i
]
-
'0'
);
if
(
i
<
length
&&
decimals
[
i
]
>=
'5'
)
// Rounding
...
...
@@ -154,8 +154,7 @@ inline bool StrtodDiyFp(const char* decimals, size_t length, size_t decimalPosit
DiyFp
v
(
significand
,
0
);
size_t
remaining
=
length
-
i
;
const
int
dExp
=
(
int
)
decimalPosition
-
i
+
exp
;
exp
+=
(
int
)
remaining
;
const
int
dExp
=
(
int
)
decimalPosition
-
(
int
)
i
+
exp
+
(
int
)
remaining
;
const
unsigned
kUlpShift
=
3
;
const
unsigned
kUlp
=
1
<<
kUlpShift
;
...
...
@@ -165,8 +164,10 @@ inline bool StrtodDiyFp(const char* decimals, size_t length, size_t decimalPosit
error
<<=
-
v
.
e
;
int
actualExp
;
v
=
v
*
GetCachedPower10
(
exp
,
&
actualExp
);
if
(
actualExp
!=
exp
)
{
double
temp1
=
v
.
ToDouble
();
v
=
v
*
GetCachedPower10
(
dExp
,
&
actualExp
);
double
temp2
=
v
.
ToDouble
();
if
(
actualExp
!=
dExp
)
{
static
const
DiyFp
kPow10
[]
=
{
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xa0000000
,
00000000
),
-
60
),
// 10^1
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xc8000000
,
00000000
),
-
57
),
// 10^2
...
...
@@ -176,16 +177,35 @@ inline bool StrtodDiyFp(const char* decimals, size_t length, size_t decimalPosit
DiyFp
(
RAPIDJSON_UINT64_C2
(
0xf4240000
,
00000000
),
-
44
),
// 10^6
DiyFp
(
RAPIDJSON_UINT64_C2
(
0x98968000
,
00000000
),
-
40
)
// 10^7
};
v
=
v
*
kPow10
[
actualExp
-
e
xp
-
1
];
v
=
v
*
kPow10
[
dExp
-
actualE
xp
-
1
];
}
double
temp3
=
v
.
ToDouble
();
error
+=
kUlp
+
(
error
==
0
?
0
:
1
);
int
oldExp
=
v
.
e
;
const
int
oldExp
=
v
.
e
;
v
=
v
.
Normalize
();
error
<<=
oldExp
-
v
.
e
;
return
true
;
const
unsigned
effectiveSignificandSize
=
Double
::
EffectiveSignificandSize
(
64
+
v
.
e
);
unsigned
precisionSize
=
64
-
effectiveSignificandSize
;
if
(
precisionSize
+
kUlpShift
>=
64
)
{
unsigned
scaleExp
=
(
precisionSize
+
kUlpShift
)
-
63
;
v
.
f
>>=
scaleExp
;
v
.
e
+=
scaleExp
;
error
=
(
error
>>
scaleExp
)
+
1
+
kUlp
;
precisionSize
-=
scaleExp
;
}
const
uint64_t
precisionBits
=
(
v
.
f
&
((
uint64_t
(
1
)
<<
precisionSize
)
-
1
))
*
kUlp
;
const
uint64_t
halfWay
=
(
uint64_t
(
1
)
<<
(
precisionSize
-
1
))
*
kUlp
;
DiyFp
rounded
(
v
.
f
>>
precisionSize
,
v
.
e
+
precisionSize
);
if
(
precisionBits
>=
halfWay
+
error
)
rounded
.
f
++
;
*
result
=
rounded
.
ToDouble
();
return
halfWay
-
error
>=
precisionBits
||
precisionBits
>=
halfWay
+
error
;
}
inline
double
StrtodBigInteger
(
double
approx
,
const
char
*
decimals
,
size_t
length
,
size_t
decimalPosition
,
int
exp
)
{
...
...
test/unittest/readertest.cpp
View file @
b4e2d58c
...
...
@@ -195,6 +195,7 @@ static void TestParseDouble() {
EXPECT_DOUBLE_EQ
(
x
,
h
.
actual_
);
\
}
#if 0
TEST_DOUBLE(fullPrecision, "0.0", 0.0);
TEST_DOUBLE(fullPrecision, "1.0", 1.0);
TEST_DOUBLE(fullPrecision, "-1.0", -1.0);
...
...
@@ -215,6 +216,7 @@ static void TestParseDouble() {
TEST_DOUBLE(fullPrecision, "2.22507e-308", 2.22507e-308);
TEST_DOUBLE(fullPrecision, "-1.79769e+308", -1.79769e+308);
TEST_DOUBLE(fullPrecision, "-2.22507e-308", -2.22507e-308);
#endif
TEST_DOUBLE
(
fullPrecision
,
"4.9406564584124654e-324"
,
4.9406564584124654e-324
);
// minimum denormal
TEST_DOUBLE
(
fullPrecision
,
"2.2250738585072009e-308"
,
2.2250738585072009e-308
);
// Max subnormal double
TEST_DOUBLE
(
fullPrecision
,
"2.2250738585072014e-308"
,
2.2250738585072014e-308
);
// Min normal positive double
...
...
@@ -257,7 +259,7 @@ static void TestParseDouble() {
TEST_DOUBLE
(
fullPrecision
,
n1e308
,
1E308
);
}
#if
1
#if
0
static const unsigned count = 10000000;
// Random test for double
{
...
...
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