Branch data Line data Source code
1 : : // Copyright (c) 2009-2010 Satoshi Nakamoto
2 : : // Copyright (c) 2009-2022 The Bitcoin Core developers
3 : : // Distributed under the MIT software license, see the accompanying
4 : : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
5 : :
6 : : #include <util/moneystr.h>
7 : :
8 : : #include <consensus/amount.h>
9 : : #include <tinyformat.h>
10 : : #include <util/strencodings.h>
11 : : #include <util/string.h>
12 : :
13 : : #include <cstdint>
14 : : #include <optional>
15 : :
16 : : using util::ContainsNoNUL;
17 : : using util::TrimString;
18 : :
19 : 212754 : std::string FormatMoney(const CAmount n)
20 : : {
21 : : // Note: not using straight sprintf here because we do NOT want
22 : : // localized number formatting.
23 : 212754 : static_assert(COIN > 1);
24 : 212754 : int64_t quotient = n / COIN;
25 : 212754 : int64_t remainder = n % COIN;
26 [ + + ]: 212754 : if (n < 0) {
27 : 1402 : quotient = -quotient;
28 : 1402 : remainder = -remainder;
29 : : }
30 : 212754 : std::string str = strprintf("%d.%08d", quotient, remainder);
31 : :
32 : : // Right-trim excess zeros before the decimal point:
33 : 212754 : int nTrim = 0;
34 [ - + + + : 919763 : for (int i = str.size()-1; (str[i] == '0' && IsDigit(str[i-2])); --i)
+ + ]
35 : 707009 : ++nTrim;
36 [ + + ]: 212754 : if (nTrim)
37 [ + - ]: 176336 : str.erase(str.size()-nTrim, nTrim);
38 : :
39 [ + + ]: 212754 : if (n < 0)
40 [ + - ]: 1402 : str.insert(uint32_t{0}, 1, '-');
41 : 212754 : return str;
42 : 0 : }
43 : :
44 : :
45 : 1234 : std::optional<CAmount> ParseMoney(const std::string& money_string)
46 : : {
47 [ - + + + ]: 2468 : if (!ContainsNoNUL(money_string)) {
48 : 6 : return std::nullopt;
49 : : }
50 : 1228 : const std::string str = TrimString(money_string);
51 [ + + ]: 1228 : if (str.empty()) {
52 : 6 : return std::nullopt;
53 : : }
54 : :
55 : 1222 : std::string strWhole;
56 : 1222 : int64_t nUnits = 0;
57 : 1222 : const char* p = str.c_str();
58 [ + + ]: 2538 : for (; *p; p++)
59 : : {
60 [ + + ]: 2425 : if (*p == '.')
61 : : {
62 : 1095 : p++;
63 : 1095 : int64_t nMult = COIN / 10;
64 [ + + + + ]: 5911 : while (IsDigit(*p) && (nMult > 0))
65 : : {
66 : 4816 : nUnits += nMult * (*p++ - '0');
67 : 4816 : nMult /= 10;
68 : : }
69 : : break;
70 : : }
71 [ + + ]: 1330 : if (IsSpace(*p))
72 : 8 : return std::nullopt;
73 [ + + ]: 1322 : if (!IsDigit(*p))
74 : 6 : return std::nullopt;
75 [ - + + - ]: 1316 : strWhole.insert(strWhole.end(), *p);
76 : : }
77 [ + + ]: 1208 : if (*p) {
78 : 8 : return std::nullopt;
79 : : }
80 [ - + + + ]: 1200 : if (strWhole.size() > 10) // guard against 63 bit overflow
81 : 2 : return std::nullopt;
82 [ - + ]: 1198 : if (nUnits < 0 || nUnits > COIN)
83 : 0 : return std::nullopt;
84 [ + - ]: 1198 : int64_t nWhole = LocaleIndependentAtoi<int64_t>(strWhole);
85 : 1198 : CAmount value = nWhole * COIN + nUnits;
86 : :
87 [ + + ]: 1198 : if (!MoneyRange(value)) {
88 : 2 : return std::nullopt;
89 : : }
90 : :
91 : 1196 : return value;
92 : 2450 : }
|