|              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                 :      139425 : std::string FormatMoney(const CAmount n)
      20                 :             : {
      21                 :             :     // Note: not using straight sprintf here because we do NOT want
      22                 :             :     // localized number formatting.
      23                 :      139425 :     static_assert(COIN > 1);
      24                 :      139425 :     int64_t quotient = n / COIN;
      25                 :      139425 :     int64_t remainder = n % COIN;
      26         [ +  + ]:      139425 :     if (n < 0) {
      27                 :          16 :         quotient = -quotient;
      28                 :          16 :         remainder = -remainder;
      29                 :             :     }
      30                 :      139425 :     std::string str = strprintf("%d.%08d", quotient, remainder);
      31                 :             : 
      32                 :             :     // Right-trim excess zeros before the decimal point:
      33                 :      139425 :     int nTrim = 0;
      34   [ -  +  +  +  :      432919 :     for (int i = str.size()-1; (str[i] == '0' && IsDigit(str[i-2])); --i)
                   +  + ]
      35                 :      293494 :         ++nTrim;
      36         [ +  + ]:      139425 :     if (nTrim)
      37         [ +  - ]:      103140 :         str.erase(str.size()-nTrim, nTrim);
      38                 :             : 
      39         [ +  + ]:      139425 :     if (n < 0)
      40         [ +  - ]:          16 :         str.insert(uint32_t{0}, 1, '-');
      41                 :      139425 :     return str;
      42                 :           0 : }
      43                 :             : 
      44                 :             : 
      45                 :          98 : std::optional<CAmount> ParseMoney(const std::string& money_string)
      46                 :             : {
      47   [ -  +  +  + ]:         196 :     if (!ContainsNoNUL(money_string)) {
      48                 :           6 :         return std::nullopt;
      49                 :             :     }
      50                 :          92 :     const std::string str = TrimString(money_string);
      51         [ +  + ]:          92 :     if (str.empty()) {
      52                 :           6 :         return std::nullopt;
      53                 :             :     }
      54                 :             : 
      55                 :          86 :     std::string strWhole;
      56                 :          86 :     int64_t nUnits = 0;
      57                 :          86 :     const char* p = str.c_str();
      58         [ +  + ]:         266 :     for (; *p; p++)
      59                 :             :     {
      60         [ +  + ]:         258 :         if (*p == '.')
      61                 :             :         {
      62                 :          64 :             p++;
      63                 :          64 :             int64_t nMult = COIN / 10;
      64   [ +  +  +  + ]:         274 :             while (IsDigit(*p) && (nMult > 0))
      65                 :             :             {
      66                 :         210 :                 nUnits += nMult * (*p++ - '0');
      67                 :         210 :                 nMult /= 10;
      68                 :             :             }
      69                 :             :             break;
      70                 :             :         }
      71         [ +  + ]:         194 :         if (IsSpace(*p))
      72                 :           8 :             return std::nullopt;
      73         [ +  + ]:         186 :         if (!IsDigit(*p))
      74                 :           6 :             return std::nullopt;
      75   [ -  +  +  - ]:         180 :         strWhole.insert(strWhole.end(), *p);
      76                 :             :     }
      77         [ +  + ]:          72 :     if (*p) {
      78                 :           8 :         return std::nullopt;
      79                 :             :     }
      80   [ -  +  +  + ]:          64 :     if (strWhole.size() > 10) // guard against 63 bit overflow
      81                 :           2 :         return std::nullopt;
      82         [ -  + ]:          62 :     if (nUnits < 0 || nUnits > COIN)
      83                 :           0 :         return std::nullopt;
      84         [ +  - ]:          62 :     int64_t nWhole = LocaleIndependentAtoi<int64_t>(strWhole);
      85                 :          62 :     CAmount value = nWhole * COIN + nUnits;
      86                 :             : 
      87         [ +  + ]:          62 :     if (!MoneyRange(value)) {
      88                 :           2 :         return std::nullopt;
      89                 :             :     }
      90                 :             : 
      91                 :          60 :     return value;
      92                 :         178 : }
         |