• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // © 2018 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 //
4 // From the double-conversion library. Original license:
5 //
6 // Copyright 2010 the V8 project authors. All rights reserved.
7 // Redistribution and use in source and binary forms, with or without
8 // modification, are permitted provided that the following conditions are
9 // met:
10 //
11 //     * Redistributions of source code must retain the above copyright
12 //       notice, this list of conditions and the following disclaimer.
13 //     * Redistributions in binary form must reproduce the above
14 //       copyright notice, this list of conditions and the following
15 //       disclaimer in the documentation and/or other materials provided
16 //       with the distribution.
17 //     * Neither the name of Google Inc. nor the names of its
18 //       contributors may be used to endorse or promote products derived
19 //       from this software without specific prior written permission.
20 //
21 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
24 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
25 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
26 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
27 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
28 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
29 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
30 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
31 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 
33 // ICU PATCH: ifdef around UCONFIG_NO_FORMATTING
34 #include "unicode/utypes.h"
35 #if !UCONFIG_NO_FORMATTING
36 
37 #include <algorithm>
38 #include <climits>
39 #include <cmath>
40 
41 // ICU PATCH: Customize header file paths for ICU.
42 // The file fixed-dtoa.h is not needed.
43 
44 #include "double-conversion-double-to-string.h"
45 
46 #include "double-conversion-bignum-dtoa.h"
47 #include "double-conversion-fast-dtoa.h"
48 #include "double-conversion-ieee.h"
49 #include "double-conversion-utils.h"
50 
51 // ICU PATCH: Wrap in ICU namespace
52 U_NAMESPACE_BEGIN
53 
54 namespace double_conversion {
55 
56 #if 0  // not needed for ICU
57 const DoubleToStringConverter& DoubleToStringConverter::EcmaScriptConverter() {
58   int flags = UNIQUE_ZERO | EMIT_POSITIVE_EXPONENT_SIGN;
59   static DoubleToStringConverter converter(flags,
60                                            "Infinity",
61                                            "NaN",
62                                            'e',
63                                            -6, 21,
64                                            6, 0);
65   return converter;
66 }
67 
68 
69 bool DoubleToStringConverter::HandleSpecialValues(
70     double value,
71     StringBuilder* result_builder) const {
72   Double double_inspect(value);
73   if (double_inspect.IsInfinite()) {
74     if (infinity_symbol_ == NULL) return false;
75     if (value < 0) {
76       result_builder->AddCharacter('-');
77     }
78     result_builder->AddString(infinity_symbol_);
79     return true;
80   }
81   if (double_inspect.IsNan()) {
82     if (nan_symbol_ == NULL) return false;
83     result_builder->AddString(nan_symbol_);
84     return true;
85   }
86   return false;
87 }
88 
89 
90 void DoubleToStringConverter::CreateExponentialRepresentation(
91     const char* decimal_digits,
92     int length,
93     int exponent,
94     StringBuilder* result_builder) const {
95   DOUBLE_CONVERSION_ASSERT(length != 0);
96   result_builder->AddCharacter(decimal_digits[0]);
97   if (length != 1) {
98     result_builder->AddCharacter('.');
99     result_builder->AddSubstring(&decimal_digits[1], length-1);
100   }
101   result_builder->AddCharacter(exponent_character_);
102   if (exponent < 0) {
103     result_builder->AddCharacter('-');
104     exponent = -exponent;
105   } else {
106     if ((flags_ & EMIT_POSITIVE_EXPONENT_SIGN) != 0) {
107       result_builder->AddCharacter('+');
108     }
109   }
110   DOUBLE_CONVERSION_ASSERT(exponent < 1e4);
111   // Changing this constant requires updating the comment of DoubleToStringConverter constructor
112   const int kMaxExponentLength = 5;
113   char buffer[kMaxExponentLength + 1];
114   buffer[kMaxExponentLength] = '\0';
115   int first_char_pos = kMaxExponentLength;
116   if (exponent == 0) {
117     buffer[--first_char_pos] = '0';
118   } else {
119     while (exponent > 0) {
120       buffer[--first_char_pos] = '0' + (exponent % 10);
121       exponent /= 10;
122     }
123   }
124   // Add prefix '0' to make exponent width >= min(min_exponent_with_, kMaxExponentLength)
125   // For example: convert 1e+9 -> 1e+09, if min_exponent_with_ is set to 2
126   while(kMaxExponentLength - first_char_pos < std::min(min_exponent_width_, kMaxExponentLength)) {
127     buffer[--first_char_pos] = '0';
128   }
129   result_builder->AddSubstring(&buffer[first_char_pos],
130                                kMaxExponentLength - first_char_pos);
131 }
132 
133 
134 void DoubleToStringConverter::CreateDecimalRepresentation(
135     const char* decimal_digits,
136     int length,
137     int decimal_point,
138     int digits_after_point,
139     StringBuilder* result_builder) const {
140   // Create a representation that is padded with zeros if needed.
141   if (decimal_point <= 0) {
142       // "0.00000decimal_rep" or "0.000decimal_rep00".
143     result_builder->AddCharacter('0');
144     if (digits_after_point > 0) {
145       result_builder->AddCharacter('.');
146       result_builder->AddPadding('0', -decimal_point);
147       DOUBLE_CONVERSION_ASSERT(length <= digits_after_point - (-decimal_point));
148       result_builder->AddSubstring(decimal_digits, length);
149       int remaining_digits = digits_after_point - (-decimal_point) - length;
150       result_builder->AddPadding('0', remaining_digits);
151     }
152   } else if (decimal_point >= length) {
153     // "decimal_rep0000.00000" or "decimal_rep.0000".
154     result_builder->AddSubstring(decimal_digits, length);
155     result_builder->AddPadding('0', decimal_point - length);
156     if (digits_after_point > 0) {
157       result_builder->AddCharacter('.');
158       result_builder->AddPadding('0', digits_after_point);
159     }
160   } else {
161     // "decima.l_rep000".
162     DOUBLE_CONVERSION_ASSERT(digits_after_point > 0);
163     result_builder->AddSubstring(decimal_digits, decimal_point);
164     result_builder->AddCharacter('.');
165     DOUBLE_CONVERSION_ASSERT(length - decimal_point <= digits_after_point);
166     result_builder->AddSubstring(&decimal_digits[decimal_point],
167                                  length - decimal_point);
168     int remaining_digits = digits_after_point - (length - decimal_point);
169     result_builder->AddPadding('0', remaining_digits);
170   }
171   if (digits_after_point == 0) {
172     if ((flags_ & EMIT_TRAILING_DECIMAL_POINT) != 0) {
173       result_builder->AddCharacter('.');
174     }
175     if ((flags_ & EMIT_TRAILING_ZERO_AFTER_POINT) != 0) {
176       result_builder->AddCharacter('0');
177     }
178   }
179 }
180 
181 
182 bool DoubleToStringConverter::ToShortestIeeeNumber(
183     double value,
184     StringBuilder* result_builder,
185     DoubleToStringConverter::DtoaMode mode) const {
186   DOUBLE_CONVERSION_ASSERT(mode == SHORTEST || mode == SHORTEST_SINGLE);
187   if (Double(value).IsSpecial()) {
188     return HandleSpecialValues(value, result_builder);
189   }
190 
191   int decimal_point;
192   bool sign;
193   const int kDecimalRepCapacity = kBase10MaximalLength + 1;
194   char decimal_rep[kDecimalRepCapacity];
195   int decimal_rep_length;
196 
197   DoubleToAscii(value, mode, 0, decimal_rep, kDecimalRepCapacity,
198                 &sign, &decimal_rep_length, &decimal_point);
199 
200   bool unique_zero = (flags_ & UNIQUE_ZERO) != 0;
201   if (sign && (value != 0.0 || !unique_zero)) {
202     result_builder->AddCharacter('-');
203   }
204 
205   int exponent = decimal_point - 1;
206   if ((decimal_in_shortest_low_ <= exponent) &&
207       (exponent < decimal_in_shortest_high_)) {
208     CreateDecimalRepresentation(decimal_rep, decimal_rep_length,
209                                 decimal_point,
210                                 (std::max)(0, decimal_rep_length - decimal_point),
211                                 result_builder);
212   } else {
213     CreateExponentialRepresentation(decimal_rep, decimal_rep_length, exponent,
214                                     result_builder);
215   }
216   return true;
217 }
218 
219 
220 bool DoubleToStringConverter::ToFixed(double value,
221                                       int requested_digits,
222                                       StringBuilder* result_builder) const {
223   DOUBLE_CONVERSION_ASSERT(kMaxFixedDigitsBeforePoint == 60);
224   const double kFirstNonFixed = 1e60;
225 
226   if (Double(value).IsSpecial()) {
227     return HandleSpecialValues(value, result_builder);
228   }
229 
230   if (requested_digits > kMaxFixedDigitsAfterPoint) return false;
231   if (value >= kFirstNonFixed || value <= -kFirstNonFixed) return false;
232 
233   // Find a sufficiently precise decimal representation of n.
234   int decimal_point;
235   bool sign;
236   // Add space for the '\0' byte.
237   const int kDecimalRepCapacity =
238       kMaxFixedDigitsBeforePoint + kMaxFixedDigitsAfterPoint + 1;
239   char decimal_rep[kDecimalRepCapacity];
240   int decimal_rep_length;
241   DoubleToAscii(value, FIXED, requested_digits,
242                 decimal_rep, kDecimalRepCapacity,
243                 &sign, &decimal_rep_length, &decimal_point);
244 
245   bool unique_zero = ((flags_ & UNIQUE_ZERO) != 0);
246   if (sign && (value != 0.0 || !unique_zero)) {
247     result_builder->AddCharacter('-');
248   }
249 
250   CreateDecimalRepresentation(decimal_rep, decimal_rep_length, decimal_point,
251                               requested_digits, result_builder);
252   return true;
253 }
254 
255 
256 bool DoubleToStringConverter::ToExponential(
257     double value,
258     int requested_digits,
259     StringBuilder* result_builder) const {
260   if (Double(value).IsSpecial()) {
261     return HandleSpecialValues(value, result_builder);
262   }
263 
264   if (requested_digits < -1) return false;
265   if (requested_digits > kMaxExponentialDigits) return false;
266 
267   int decimal_point;
268   bool sign;
269   // Add space for digit before the decimal point and the '\0' character.
270   const int kDecimalRepCapacity = kMaxExponentialDigits + 2;
271   DOUBLE_CONVERSION_ASSERT(kDecimalRepCapacity > kBase10MaximalLength);
272   char decimal_rep[kDecimalRepCapacity];
273 #ifndef NDEBUG
274   // Problem: there is an assert in StringBuilder::AddSubstring() that
275   // will pass this buffer to strlen(), and this buffer is not generally
276   // null-terminated.
277   memset(decimal_rep, 0, sizeof(decimal_rep));
278 #endif
279   int decimal_rep_length;
280 
281   if (requested_digits == -1) {
282     DoubleToAscii(value, SHORTEST, 0,
283                   decimal_rep, kDecimalRepCapacity,
284                   &sign, &decimal_rep_length, &decimal_point);
285   } else {
286     DoubleToAscii(value, PRECISION, requested_digits + 1,
287                   decimal_rep, kDecimalRepCapacity,
288                   &sign, &decimal_rep_length, &decimal_point);
289     DOUBLE_CONVERSION_ASSERT(decimal_rep_length <= requested_digits + 1);
290 
291     for (int i = decimal_rep_length; i < requested_digits + 1; ++i) {
292       decimal_rep[i] = '0';
293     }
294     decimal_rep_length = requested_digits + 1;
295   }
296 
297   bool unique_zero = ((flags_ & UNIQUE_ZERO) != 0);
298   if (sign && (value != 0.0 || !unique_zero)) {
299     result_builder->AddCharacter('-');
300   }
301 
302   int exponent = decimal_point - 1;
303   CreateExponentialRepresentation(decimal_rep,
304                                   decimal_rep_length,
305                                   exponent,
306                                   result_builder);
307   return true;
308 }
309 
310 
311 bool DoubleToStringConverter::ToPrecision(double value,
312                                           int precision,
313                                           StringBuilder* result_builder) const {
314   if (Double(value).IsSpecial()) {
315     return HandleSpecialValues(value, result_builder);
316   }
317 
318   if (precision < kMinPrecisionDigits || precision > kMaxPrecisionDigits) {
319     return false;
320   }
321 
322   // Find a sufficiently precise decimal representation of n.
323   int decimal_point;
324   bool sign;
325   // Add one for the terminating null character.
326   const int kDecimalRepCapacity = kMaxPrecisionDigits + 1;
327   char decimal_rep[kDecimalRepCapacity];
328   int decimal_rep_length;
329 
330   DoubleToAscii(value, PRECISION, precision,
331                 decimal_rep, kDecimalRepCapacity,
332                 &sign, &decimal_rep_length, &decimal_point);
333   DOUBLE_CONVERSION_ASSERT(decimal_rep_length <= precision);
334 
335   bool unique_zero = ((flags_ & UNIQUE_ZERO) != 0);
336   if (sign && (value != 0.0 || !unique_zero)) {
337     result_builder->AddCharacter('-');
338   }
339 
340   // The exponent if we print the number as x.xxeyyy. That is with the
341   // decimal point after the first digit.
342   int exponent = decimal_point - 1;
343 
344   int extra_zero = ((flags_ & EMIT_TRAILING_ZERO_AFTER_POINT) != 0) ? 1 : 0;
345   bool as_exponential =
346       (-decimal_point + 1 > max_leading_padding_zeroes_in_precision_mode_) ||
347       (decimal_point - precision + extra_zero >
348        max_trailing_padding_zeroes_in_precision_mode_);
349   if ((flags_ & NO_TRAILING_ZERO) != 0) {
350     // Truncate trailing zeros that occur after the decimal point (if exponential,
351     // that is everything after the first digit).
352     int stop = as_exponential ? 1 : std::max(1, decimal_point);
353     while (decimal_rep_length > stop && decimal_rep[decimal_rep_length - 1] == '0') {
354       --decimal_rep_length;
355     }
356     // Clamp precision to avoid the code below re-adding the zeros.
357     precision = std::min(precision, decimal_rep_length);
358   }
359   if (as_exponential) {
360     // Fill buffer to contain 'precision' digits.
361     // Usually the buffer is already at the correct length, but 'DoubleToAscii'
362     // is allowed to return less characters.
363     for (int i = decimal_rep_length; i < precision; ++i) {
364       decimal_rep[i] = '0';
365     }
366 
367     CreateExponentialRepresentation(decimal_rep,
368                                     precision,
369                                     exponent,
370                                     result_builder);
371   } else {
372     CreateDecimalRepresentation(decimal_rep, decimal_rep_length, decimal_point,
373                                 (std::max)(0, precision - decimal_point),
374                                 result_builder);
375   }
376   return true;
377 }
378 #endif // not needed for ICU
379 
380 
DtoaToBignumDtoaMode(DoubleToStringConverter::DtoaMode dtoa_mode)381 static BignumDtoaMode DtoaToBignumDtoaMode(
382     DoubleToStringConverter::DtoaMode dtoa_mode) {
383   switch (dtoa_mode) {
384     case DoubleToStringConverter::SHORTEST:  return BIGNUM_DTOA_SHORTEST;
385     case DoubleToStringConverter::SHORTEST_SINGLE:
386         return BIGNUM_DTOA_SHORTEST_SINGLE;
387     case DoubleToStringConverter::FIXED:     return BIGNUM_DTOA_FIXED;
388     case DoubleToStringConverter::PRECISION: return BIGNUM_DTOA_PRECISION;
389     default:
390       DOUBLE_CONVERSION_UNREACHABLE();
391   }
392 }
393 
394 
DoubleToAscii(double v,DtoaMode mode,int requested_digits,char * buffer,int buffer_length,bool * sign,int * length,int * point)395 void DoubleToStringConverter::DoubleToAscii(double v,
396                                             DtoaMode mode,
397                                             int requested_digits,
398                                             char* buffer,
399                                             int buffer_length,
400                                             bool* sign,
401                                             int* length,
402                                             int* point) {
403   Vector<char> vector(buffer, buffer_length);
404   DOUBLE_CONVERSION_ASSERT(!Double(v).IsSpecial());
405   DOUBLE_CONVERSION_ASSERT(mode == SHORTEST || mode == SHORTEST_SINGLE || requested_digits >= 0);
406 
407   if (Double(v).Sign() < 0) {
408     *sign = true;
409     v = -v;
410   } else {
411     *sign = false;
412   }
413 
414   if (mode == PRECISION && requested_digits == 0) {
415     vector[0] = '\0';
416     *length = 0;
417     return;
418   }
419 
420   if (v == 0) {
421     vector[0] = '0';
422     vector[1] = '\0';
423     *length = 1;
424     *point = 1;
425     return;
426   }
427 
428   bool fast_worked;
429   switch (mode) {
430     case SHORTEST:
431       fast_worked = FastDtoa(v, FAST_DTOA_SHORTEST, 0, vector, length, point);
432       break;
433 #if 0 // not needed for ICU
434     case SHORTEST_SINGLE:
435       fast_worked = FastDtoa(v, FAST_DTOA_SHORTEST_SINGLE, 0,
436                              vector, length, point);
437       break;
438     case FIXED:
439       fast_worked = FastFixedDtoa(v, requested_digits, vector, length, point);
440       break;
441     case PRECISION:
442       fast_worked = FastDtoa(v, FAST_DTOA_PRECISION, requested_digits,
443                              vector, length, point);
444       break;
445 #endif // not needed for ICU
446     default:
447       fast_worked = false;
448       DOUBLE_CONVERSION_UNREACHABLE();
449   }
450   if (fast_worked) return;
451 
452   // If the fast dtoa didn't succeed use the slower bignum version.
453   BignumDtoaMode bignum_mode = DtoaToBignumDtoaMode(mode);
454   BignumDtoa(v, bignum_mode, requested_digits, vector, length, point);
455   vector[*length] = '\0';
456 }
457 
458 }  // namespace double_conversion
459 
460 // ICU PATCH: Close ICU namespace
461 U_NAMESPACE_END
462 #endif // ICU PATCH: close #if !UCONFIG_NO_FORMATTING
463