xref: /aosp_15_r20/external/protobuf/src/google/protobuf/io/strtod.cc (revision 1b3f573f81763fcece89efc2b6a5209149e44ab8)
1*1b3f573fSAndroid Build Coastguard Worker // Protocol Buffers - Google's data interchange format
2*1b3f573fSAndroid Build Coastguard Worker // Copyright 2008 Google Inc.  All rights reserved.
3*1b3f573fSAndroid Build Coastguard Worker // https://developers.google.com/protocol-buffers/
4*1b3f573fSAndroid Build Coastguard Worker //
5*1b3f573fSAndroid Build Coastguard Worker // Redistribution and use in source and binary forms, with or without
6*1b3f573fSAndroid Build Coastguard Worker // modification, are permitted provided that the following conditions are
7*1b3f573fSAndroid Build Coastguard Worker // met:
8*1b3f573fSAndroid Build Coastguard Worker //
9*1b3f573fSAndroid Build Coastguard Worker //     * Redistributions of source code must retain the above copyright
10*1b3f573fSAndroid Build Coastguard Worker // notice, this list of conditions and the following disclaimer.
11*1b3f573fSAndroid Build Coastguard Worker //     * Redistributions in binary form must reproduce the above
12*1b3f573fSAndroid Build Coastguard Worker // copyright notice, this list of conditions and the following disclaimer
13*1b3f573fSAndroid Build Coastguard Worker // in the documentation and/or other materials provided with the
14*1b3f573fSAndroid Build Coastguard Worker // distribution.
15*1b3f573fSAndroid Build Coastguard Worker //     * Neither the name of Google Inc. nor the names of its
16*1b3f573fSAndroid Build Coastguard Worker // contributors may be used to endorse or promote products derived from
17*1b3f573fSAndroid Build Coastguard Worker // this software without specific prior written permission.
18*1b3f573fSAndroid Build Coastguard Worker //
19*1b3f573fSAndroid Build Coastguard Worker // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20*1b3f573fSAndroid Build Coastguard Worker // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21*1b3f573fSAndroid Build Coastguard Worker // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22*1b3f573fSAndroid Build Coastguard Worker // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23*1b3f573fSAndroid Build Coastguard Worker // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24*1b3f573fSAndroid Build Coastguard Worker // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25*1b3f573fSAndroid Build Coastguard Worker // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26*1b3f573fSAndroid Build Coastguard Worker // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27*1b3f573fSAndroid Build Coastguard Worker // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28*1b3f573fSAndroid Build Coastguard Worker // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29*1b3f573fSAndroid Build Coastguard Worker // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30*1b3f573fSAndroid Build Coastguard Worker 
31*1b3f573fSAndroid Build Coastguard Worker #include <google/protobuf/io/strtod.h>
32*1b3f573fSAndroid Build Coastguard Worker 
33*1b3f573fSAndroid Build Coastguard Worker #include <cstdio>
34*1b3f573fSAndroid Build Coastguard Worker #include <cstring>
35*1b3f573fSAndroid Build Coastguard Worker #include <limits>
36*1b3f573fSAndroid Build Coastguard Worker #include <string>
37*1b3f573fSAndroid Build Coastguard Worker 
38*1b3f573fSAndroid Build Coastguard Worker #include <google/protobuf/stubs/logging.h>
39*1b3f573fSAndroid Build Coastguard Worker #include <google/protobuf/stubs/common.h>
40*1b3f573fSAndroid Build Coastguard Worker 
41*1b3f573fSAndroid Build Coastguard Worker #include <google/protobuf/stubs/strutil.h>
42*1b3f573fSAndroid Build Coastguard Worker 
43*1b3f573fSAndroid Build Coastguard Worker namespace google {
44*1b3f573fSAndroid Build Coastguard Worker namespace protobuf {
45*1b3f573fSAndroid Build Coastguard Worker namespace io {
46*1b3f573fSAndroid Build Coastguard Worker 
47*1b3f573fSAndroid Build Coastguard Worker // This approximately 0x1.ffffffp127, but we don't use 0x1.ffffffp127 because
48*1b3f573fSAndroid Build Coastguard Worker // it won't compile in MSVC.
49*1b3f573fSAndroid Build Coastguard Worker const double MAX_FLOAT_AS_DOUBLE_ROUNDED = 3.4028235677973366e+38;
50*1b3f573fSAndroid Build Coastguard Worker 
SafeDoubleToFloat(double value)51*1b3f573fSAndroid Build Coastguard Worker float SafeDoubleToFloat(double value) {
52*1b3f573fSAndroid Build Coastguard Worker   // static_cast<float> on a number larger than float can result in illegal
53*1b3f573fSAndroid Build Coastguard Worker   // instruction error, so we need to manually convert it to infinity or max.
54*1b3f573fSAndroid Build Coastguard Worker   if (value > std::numeric_limits<float>::max()) {
55*1b3f573fSAndroid Build Coastguard Worker     // Max float value is about 3.4028234664E38 when represented as a double.
56*1b3f573fSAndroid Build Coastguard Worker     // However, when printing float as text, it will be rounded as
57*1b3f573fSAndroid Build Coastguard Worker     // 3.4028235e+38. If we parse the value of 3.4028235e+38 from text and
58*1b3f573fSAndroid Build Coastguard Worker     // compare it to 3.4028234664E38, we may think that it is larger, but
59*1b3f573fSAndroid Build Coastguard Worker     // actually, any number between these two numbers could only be represented
60*1b3f573fSAndroid Build Coastguard Worker     // as the same max float number in float, so we should treat them the same
61*1b3f573fSAndroid Build Coastguard Worker     // as max float.
62*1b3f573fSAndroid Build Coastguard Worker     if (value <= MAX_FLOAT_AS_DOUBLE_ROUNDED) {
63*1b3f573fSAndroid Build Coastguard Worker       return std::numeric_limits<float>::max();
64*1b3f573fSAndroid Build Coastguard Worker     }
65*1b3f573fSAndroid Build Coastguard Worker     return std::numeric_limits<float>::infinity();
66*1b3f573fSAndroid Build Coastguard Worker   } else if (value < -std::numeric_limits<float>::max()) {
67*1b3f573fSAndroid Build Coastguard Worker     if (value >= -MAX_FLOAT_AS_DOUBLE_ROUNDED) {
68*1b3f573fSAndroid Build Coastguard Worker       return -std::numeric_limits<float>::max();
69*1b3f573fSAndroid Build Coastguard Worker     }
70*1b3f573fSAndroid Build Coastguard Worker     return -std::numeric_limits<float>::infinity();
71*1b3f573fSAndroid Build Coastguard Worker   } else {
72*1b3f573fSAndroid Build Coastguard Worker     return static_cast<float>(value);
73*1b3f573fSAndroid Build Coastguard Worker   }
74*1b3f573fSAndroid Build Coastguard Worker }
75*1b3f573fSAndroid Build Coastguard Worker 
NoLocaleStrtod(const char * str,char ** endptr)76*1b3f573fSAndroid Build Coastguard Worker double NoLocaleStrtod(const char* str, char** endptr) {
77*1b3f573fSAndroid Build Coastguard Worker   return google::protobuf::internal::NoLocaleStrtod(str, endptr);
78*1b3f573fSAndroid Build Coastguard Worker }
79*1b3f573fSAndroid Build Coastguard Worker 
80*1b3f573fSAndroid Build Coastguard Worker }  // namespace io
81*1b3f573fSAndroid Build Coastguard Worker }  // namespace protobuf
82*1b3f573fSAndroid Build Coastguard Worker }  // namespace google
83