Blame lib/mathlib.cpp

Packit 2035a7
/*
Packit 2035a7
 * Cppcheck - A tool for static C/C++ code analysis
Packit 2035a7
 * Copyright (C) 2007-2018 Cppcheck team.
Packit 2035a7
 *
Packit 2035a7
 * This program is free software: you can redistribute it and/or modify
Packit 2035a7
 * it under the terms of the GNU General Public License as published by
Packit 2035a7
 * the Free Software Foundation, either version 3 of the License, or
Packit 2035a7
 * (at your option) any later version.
Packit 2035a7
 *
Packit 2035a7
 * This program is distributed in the hope that it will be useful,
Packit 2035a7
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 2035a7
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 2035a7
 * GNU General Public License for more details.
Packit 2035a7
 *
Packit 2035a7
 * You should have received a copy of the GNU General Public License
Packit 2035a7
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Packit 2035a7
 */
Packit 2035a7
Packit 2035a7
Packit 2035a7
#include "mathlib.h"
Packit 2035a7
#include "errorlogger.h"
Packit 2035a7
#include "utils.h"
Packit 2035a7
Packit 2035a7
#include <cctype>
Packit 2035a7
#include <cmath>
Packit 2035a7
#include <cstdlib>
Packit 2035a7
#include <limits>
Packit 2035a7
#include <locale>
Packit 2035a7
Packit 2035a7
#if defined(_MSC_VER) && _MSC_VER <= 1700  // VS2012 doesn't have std::isinf and std::isnan
Packit 2035a7
#define ISINF(x)      (!_finite(x))
Packit 2035a7
#define ISNAN(x)      (_isnan(x))
Packit 2035a7
#elif defined(__INTEL_COMPILER)
Packit 2035a7
#define ISINF(x)      (isinf(x))
Packit 2035a7
#define ISNAN(x)      (isnan(x))
Packit 2035a7
#else  // Use C++11 functions
Packit 2035a7
#define ISINF(x)      (std::isinf(x))
Packit 2035a7
#define ISNAN(x)      (std::isnan(x))
Packit 2035a7
#endif
Packit 2035a7
Packit 2035a7
const int MathLib::bigint_bits = 64;
Packit 2035a7
Packit 2035a7
MathLib::value::value(const std::string &s) :
Packit 2035a7
    intValue(0), doubleValue(0), isUnsigned(false)
Packit 2035a7
{
Packit 2035a7
    if (MathLib::isFloat(s)) {
Packit 2035a7
        type = MathLib::value::FLOAT;
Packit 2035a7
        doubleValue = MathLib::toDoubleNumber(s);
Packit 2035a7
        return;
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    if (!MathLib::isInt(s))
Packit 2035a7
        throw InternalError(nullptr, "Invalid value: " + s);
Packit 2035a7
Packit 2035a7
    type = MathLib::value::INT;
Packit 2035a7
    intValue = MathLib::toLongNumber(s);
Packit 2035a7
Packit 2035a7
    if (isIntHex(s) && intValue < 0)
Packit 2035a7
        isUnsigned = true;
Packit 2035a7
Packit 2035a7
    // read suffix
Packit 2035a7
    if (s.size() >= 2U) {
Packit 2035a7
        for (std::size_t i = s.size() - 1U; i > 0U; --i) {
Packit 2035a7
            char c = s[i];
Packit 2035a7
            if (c == 'u' || c == 'U')
Packit 2035a7
                isUnsigned = true;
Packit 2035a7
            else if (c == 'l' || c == 'L') {
Packit 2035a7
                if (type == MathLib::value::INT)
Packit 2035a7
                    type = MathLib::value::LONG;
Packit 2035a7
                else if (type == MathLib::value::LONG)
Packit 2035a7
                    type = MathLib::value::LONGLONG;
Packit 2035a7
            } else if (i > 2U && c == '4' && s[i-1] == '6' && s[i-2] == 'i')
Packit 2035a7
                type = MathLib::value::LONGLONG;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::value::str() const
Packit 2035a7
{
Packit 2035a7
    std::ostringstream ostr;
Packit 2035a7
    if (type == MathLib::value::FLOAT) {
Packit 2035a7
        if (ISNAN(doubleValue))
Packit 2035a7
            return "nan.0";
Packit 2035a7
        if (ISINF(doubleValue))
Packit 2035a7
            return (doubleValue > 0) ? "inf.0" : "-inf.0";
Packit 2035a7
Packit 2035a7
        ostr.precision(9);
Packit 2035a7
        ostr << std::fixed << doubleValue;
Packit 2035a7
Packit 2035a7
        // remove trailing zeros
Packit 2035a7
        std::string ret(ostr.str());
Packit 2035a7
        std::string::size_type pos = ret.size() - 1U;
Packit 2035a7
        while (ret[pos] == '0')
Packit 2035a7
            pos--;
Packit 2035a7
        if (ret[pos] == '.')
Packit 2035a7
            ++pos;
Packit 2035a7
Packit 2035a7
        return ret.substr(0, pos+1);
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    if (isUnsigned)
Packit 2035a7
        ostr << static_cast<biguint>(intValue) << "U";
Packit 2035a7
    else
Packit 2035a7
        ostr << intValue;
Packit 2035a7
    if (type == MathLib::value::LONG)
Packit 2035a7
        ostr << "L";
Packit 2035a7
    else if (type == MathLib::value::LONGLONG)
Packit 2035a7
        ostr << "LL";
Packit 2035a7
    return ostr.str();
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
void MathLib::value::promote(const MathLib::value &v)
Packit 2035a7
{
Packit 2035a7
    if (isInt() && v.isInt()) {
Packit 2035a7
        if (type < v.type) {
Packit 2035a7
            type = v.type;
Packit 2035a7
            isUnsigned = v.isUnsigned;
Packit 2035a7
        } else if (type == v.type) {
Packit 2035a7
            isUnsigned |= v.isUnsigned;
Packit 2035a7
        }
Packit 2035a7
    } else if (!isFloat()) {
Packit 2035a7
        isUnsigned = false;
Packit 2035a7
        doubleValue = intValue;
Packit 2035a7
        type = MathLib::value::FLOAT;
Packit 2035a7
    }
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
Packit 2035a7
MathLib::value MathLib::value::calc(char op, const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    value temp(v1);
Packit 2035a7
    temp.promote(v2);
Packit 2035a7
    if (temp.isFloat()) {
Packit 2035a7
        switch (op) {
Packit 2035a7
        case '+':
Packit 2035a7
            temp.doubleValue += v2.getDoubleValue();
Packit 2035a7
            break;
Packit 2035a7
        case '-':
Packit 2035a7
            temp.doubleValue -= v2.getDoubleValue();
Packit 2035a7
            break;
Packit 2035a7
        case '*':
Packit 2035a7
            temp.doubleValue *= v2.getDoubleValue();
Packit 2035a7
            break;
Packit 2035a7
        case '/':
Packit 2035a7
            temp.doubleValue /= v2.getDoubleValue();
Packit 2035a7
            break;
Packit 2035a7
        case '%':
Packit 2035a7
        case '&':
Packit 2035a7
        case '|':
Packit 2035a7
        case '^':
Packit 2035a7
            throw InternalError(nullptr, "Invalid calculation");
Packit 2035a7
        default:
Packit 2035a7
            throw InternalError(nullptr, "Unhandled calculation");
Packit 2035a7
        }
Packit 2035a7
    } else if (temp.isUnsigned) {
Packit 2035a7
        switch (op) {
Packit 2035a7
        case '+':
Packit 2035a7
            temp.intValue += (unsigned long long)v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '-':
Packit 2035a7
            temp.intValue -= (unsigned long long)v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '*':
Packit 2035a7
            temp.intValue *= (unsigned long long)v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '/':
Packit 2035a7
            if (v2.intValue == 0)
Packit 2035a7
                throw InternalError(nullptr, "Internal Error: Division by zero");
Packit 2035a7
            if (v1.intValue == std::numeric_limits<bigint>::min() && std::abs(v2.intValue)<=1)
Packit 2035a7
                throw InternalError(nullptr, "Internal Error: Division overflow");
Packit 2035a7
            temp.intValue /= (unsigned long long)v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '%':
Packit 2035a7
            if (v2.intValue == 0)
Packit 2035a7
                throw InternalError(nullptr, "Internal Error: Division by zero");
Packit 2035a7
            temp.intValue %= (unsigned long long)v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '&':
Packit 2035a7
            temp.intValue &= (unsigned long long)v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '|':
Packit 2035a7
            temp.intValue |= (unsigned long long)v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '^':
Packit 2035a7
            temp.intValue ^= (unsigned long long)v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        default:
Packit 2035a7
            throw InternalError(nullptr, "Unhandled calculation");
Packit 2035a7
        }
Packit 2035a7
    } else {
Packit 2035a7
        switch (op) {
Packit 2035a7
        case '+':
Packit 2035a7
            temp.intValue += v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '-':
Packit 2035a7
            temp.intValue -= v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '*':
Packit 2035a7
            temp.intValue *= v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '/':
Packit 2035a7
            if (v2.intValue == 0)
Packit 2035a7
                throw InternalError(nullptr, "Internal Error: Division by zero");
Packit 2035a7
            if (v1.intValue == std::numeric_limits<bigint>::min() && std::abs(v2.intValue)<=1)
Packit 2035a7
                throw InternalError(nullptr, "Internal Error: Division overflow");
Packit 2035a7
            temp.intValue /= v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '%':
Packit 2035a7
            if (v2.intValue == 0)
Packit 2035a7
                throw InternalError(nullptr, "Internal Error: Division by zero");
Packit 2035a7
            temp.intValue %= v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '&':
Packit 2035a7
            temp.intValue &= v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '|':
Packit 2035a7
            temp.intValue |= v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        case '^':
Packit 2035a7
            temp.intValue ^= v2.intValue;
Packit 2035a7
            break;
Packit 2035a7
        default:
Packit 2035a7
            throw InternalError(nullptr, "Unhandled calculation");
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
    return temp;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
Packit 2035a7
int MathLib::value::compare(const MathLib::value &v) const
Packit 2035a7
{
Packit 2035a7
    value temp(*this);
Packit 2035a7
    temp.promote(v);
Packit 2035a7
Packit 2035a7
    if (temp.isFloat()) {
Packit 2035a7
        if (temp.doubleValue < v.getDoubleValue())
Packit 2035a7
            return -1;
Packit 2035a7
        if (temp.doubleValue > v.getDoubleValue())
Packit 2035a7
            return 1;
Packit 2035a7
        return 0;
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    if (temp.isUnsigned) {
Packit 2035a7
        if ((unsigned long long)intValue < (unsigned long long)v.intValue)
Packit 2035a7
            return -1;
Packit 2035a7
        if ((unsigned long long)intValue > (unsigned long long)v.intValue)
Packit 2035a7
            return 1;
Packit 2035a7
        return 0;
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    if (intValue < v.intValue)
Packit 2035a7
        return -1;
Packit 2035a7
    if (intValue > v.intValue)
Packit 2035a7
        return 1;
Packit 2035a7
    return 0;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value MathLib::value::add(int v) const
Packit 2035a7
{
Packit 2035a7
    MathLib::value temp(*this);
Packit 2035a7
    if (temp.isInt())
Packit 2035a7
        temp.intValue += v;
Packit 2035a7
    else
Packit 2035a7
        temp.doubleValue += v;
Packit 2035a7
    return temp;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value MathLib::value::shiftLeft(const MathLib::value &v) const
Packit 2035a7
{
Packit 2035a7
    if (!isInt() || !v.isInt())
Packit 2035a7
        throw InternalError(nullptr, "Shift operand is not integer");
Packit 2035a7
    MathLib::value ret(*this);
Packit 2035a7
    if (v.intValue >= MathLib::bigint_bits) {
Packit 2035a7
        return ret;
Packit 2035a7
    }
Packit 2035a7
    ret.intValue <<= v.intValue;
Packit 2035a7
    return ret;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value MathLib::value::shiftRight(const MathLib::value &v) const
Packit 2035a7
{
Packit 2035a7
    if (!isInt() || !v.isInt())
Packit 2035a7
        throw InternalError(nullptr, "Shift operand is not integer");
Packit 2035a7
    MathLib::value ret(*this);
Packit 2035a7
    if (v.intValue >= MathLib::bigint_bits) {
Packit 2035a7
        return ret;
Packit 2035a7
    }
Packit 2035a7
    ret.intValue >>= v.intValue;
Packit 2035a7
    return ret;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
Packit 2035a7
MathLib::biguint MathLib::toULongNumber(const std::string & str)
Packit 2035a7
{
Packit 2035a7
    // hexadecimal numbers:
Packit 2035a7
    if (isIntHex(str)) {
Packit 2035a7
        if (str[0] == '-') {
Packit 2035a7
            biguint ret = 0;
Packit 2035a7
            std::istringstream istr(str);
Packit 2035a7
            istr >> std::hex >> ret;
Packit 2035a7
            return ret;
Packit 2035a7
        } else {
Packit 2035a7
            unsigned long long ret = 0;
Packit 2035a7
            std::istringstream istr(str);
Packit 2035a7
            istr >> std::hex >> ret;
Packit 2035a7
            return (biguint)ret;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    // octal numbers:
Packit 2035a7
    if (isOct(str)) {
Packit 2035a7
        biguint ret = 0;
Packit 2035a7
        std::istringstream istr(str);
Packit 2035a7
        istr >> std::oct >> ret;
Packit 2035a7
        return ret;
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    // binary numbers:
Packit 2035a7
    if (isBin(str)) {
Packit 2035a7
        biguint ret = 0;
Packit 2035a7
        for (std::string::size_type i = str[0] == '0'?2:3; i < str.length(); i++) {
Packit 2035a7
            ret <<= 1;
Packit 2035a7
            if (str[i] == '1')
Packit 2035a7
                ret |= 1;
Packit 2035a7
        }
Packit 2035a7
        /* if (str[0] == '-')
Packit 2035a7
                ret = -ret; */
Packit 2035a7
        return ret;
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    if (isFloat(str)) {
Packit 2035a7
        // Things are going to be less precise now: the value can't b represented in the biguint type.
Packit 2035a7
        // Use min/max values as an approximation. See #5843
Packit 2035a7
        const double doubleval = std::atof(str.c_str());
Packit 2035a7
        if (doubleval > (double)std::numeric_limits<biguint>::max())
Packit 2035a7
            return std::numeric_limits<biguint>::max();
Packit 2035a7
        else
Packit 2035a7
            return static_cast<biguint>(doubleval);
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    biguint ret = 0;
Packit 2035a7
    std::istringstream istr(str);
Packit 2035a7
    istr >> ret;
Packit 2035a7
    return ret;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
static unsigned int encodeMultiChar(const std::string& str)
Packit 2035a7
{
Packit 2035a7
    unsigned int retval = 0;
Packit 2035a7
    for (std::string::const_iterator it=str.begin(); it!=str.end(); ++it) {
Packit 2035a7
        retval = (retval << 8) | *it;
Packit 2035a7
    }
Packit 2035a7
    return retval;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
static bool isoctal(int c)
Packit 2035a7
{
Packit 2035a7
    return c>='0' && c<='7';
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::bigint MathLib::characterLiteralToLongNumber(const std::string& str)
Packit 2035a7
{
Packit 2035a7
    if (str.empty())
Packit 2035a7
        return 0; // <- only possible in unit testing
Packit 2035a7
Packit 2035a7
    // '\xF6'
Packit 2035a7
    if (str.size() == 4 && str.compare(0,2,"\\x")==0 && std::isxdigit(str[2]) && std::isxdigit(str[3])) {
Packit 2035a7
        return std::strtoul(str.substr(2).c_str(), nullptr, 16);
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    // '\123'
Packit 2035a7
    if (str.size() == 4 && str[0] == '\\' && isoctal(str[1]) && isoctal(str[2]) && isoctal(str[3])) {
Packit 2035a7
        return (char)std::strtoul(str.substr(1).c_str(), NULL, 8);
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    // C99 6.4.4.4
Packit 2035a7
    // The value of an integer character constant containing more than one character (e.g., 'ab'),
Packit 2035a7
    // or containing a character or escape sequence that does not map to a single-byte execution character,
Packit 2035a7
    // is implementation-defined.
Packit 2035a7
    // clang and gcc seem to use the following encoding: 'AB' as (('A' << 8) | 'B')
Packit 2035a7
    const std::string& normStr = normalizeCharacterLiteral(str);
Packit 2035a7
    return encodeMultiChar(normStr);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::normalizeCharacterLiteral(const std::string& iLiteral)
Packit 2035a7
{
Packit 2035a7
    std::string normalizedLiteral;
Packit 2035a7
    const std::string::size_type iLiteralLen = iLiteral.size();
Packit 2035a7
    for (std::string::size_type idx = 0; idx < iLiteralLen ; ++idx) {
Packit 2035a7
        if (iLiteral[idx] != '\\') {
Packit 2035a7
            normalizedLiteral.push_back(iLiteral[idx]);
Packit 2035a7
            continue;
Packit 2035a7
        }
Packit 2035a7
        ++idx;
Packit 2035a7
        if (idx == iLiteralLen) {
Packit 2035a7
            throw InternalError(nullptr, "Internal Error. MathLib::normalizeCharacterLiteral: Unhandled char constant '" + iLiteral + "'.");
Packit 2035a7
        }
Packit 2035a7
        switch (iLiteral[idx]) {
Packit 2035a7
        case 'x':
Packit 2035a7
            // Hexa-decimal number: skip \x and interpret the next two characters
Packit 2035a7
        {
Packit 2035a7
            if (++idx == iLiteralLen)
Packit 2035a7
                throw InternalError(nullptr, "Internal Error. MathLib::normalizeCharacterLiteral: Unhandled char constant '" + iLiteral + "'.");
Packit 2035a7
            std::string tempBuf;
Packit 2035a7
            tempBuf.push_back(iLiteral[idx]);
Packit 2035a7
            if (++idx != iLiteralLen)
Packit 2035a7
                tempBuf.push_back(iLiteral[idx]);
Packit 2035a7
            normalizedLiteral.push_back(static_cast<char>(MathLib::toULongNumber("0x" + tempBuf)));
Packit 2035a7
            continue;
Packit 2035a7
        }
Packit 2035a7
        case 'u':
Packit 2035a7
        case 'U':
Packit 2035a7
            // Unicode string; just skip the \u or \U
Packit 2035a7
            if (idx + 1 == iLiteralLen)
Packit 2035a7
                throw InternalError(nullptr, "Internal Error. MathLib::characterLiteralToLongNumber: Unhandled char constant '" + iLiteral + "'.");
Packit 2035a7
            continue;
Packit 2035a7
        }
Packit 2035a7
        // Single digit octal number
Packit 2035a7
        if (1 == iLiteralLen - idx) {
Packit 2035a7
            switch (iLiteral[idx]) {
Packit 2035a7
            case '0':
Packit 2035a7
            case '1':
Packit 2035a7
            case '2':
Packit 2035a7
            case '3':
Packit 2035a7
            case '4':
Packit 2035a7
            case '5':
Packit 2035a7
            case '6':
Packit 2035a7
            case '7':
Packit 2035a7
                normalizedLiteral.push_back(iLiteral[idx]-'0');
Packit 2035a7
                break;
Packit 2035a7
            case 'a':
Packit 2035a7
                normalizedLiteral.push_back('\a');
Packit 2035a7
                break;
Packit 2035a7
            case 'b':
Packit 2035a7
                normalizedLiteral.push_back('\b');
Packit 2035a7
                break;
Packit 2035a7
            case 'e':
Packit 2035a7
                normalizedLiteral.push_back(0x1B); // clang, gcc, tcc interpnormalizedLiteral this as 0x1B - escape character
Packit 2035a7
                break;
Packit 2035a7
            case 'f':
Packit 2035a7
                normalizedLiteral.push_back('\f');
Packit 2035a7
                break;
Packit 2035a7
            case 'n':
Packit 2035a7
                normalizedLiteral.push_back('\n');
Packit 2035a7
                break;
Packit 2035a7
            case 'r':
Packit 2035a7
                normalizedLiteral.push_back('\r');
Packit 2035a7
                break;
Packit 2035a7
            case 't':
Packit 2035a7
                normalizedLiteral.push_back('\t');
Packit 2035a7
                break;
Packit 2035a7
            case 'v':
Packit 2035a7
                normalizedLiteral.push_back('\v');
Packit 2035a7
                break;
Packit 2035a7
            case '\\':
Packit 2035a7
            case '\?':
Packit 2035a7
            case '\'':
Packit 2035a7
            case '\"':
Packit 2035a7
                normalizedLiteral.push_back(iLiteral[idx]);
Packit 2035a7
                break;
Packit 2035a7
            default:
Packit 2035a7
                throw InternalError(nullptr, "Internal Error. MathLib::normalizeCharacterLiteral: Unhandled char constant '" + iLiteral + "'.");
Packit 2035a7
            }
Packit 2035a7
            continue;
Packit 2035a7
        }
Packit 2035a7
        // 2-3 digit octal number
Packit 2035a7
        if (!MathLib::isOctalDigit(iLiteral[idx]))
Packit 2035a7
            throw InternalError(nullptr, "Internal Error. MathLib::normalizeCharacterLiteral: Unhandled char constant '" + iLiteral + "'.");
Packit 2035a7
        std::string tempBuf;
Packit 2035a7
        tempBuf.push_back(iLiteral[idx]);
Packit 2035a7
        ++idx;
Packit 2035a7
        if (MathLib::isOctalDigit(iLiteral[idx])) {
Packit 2035a7
            tempBuf.push_back(iLiteral[idx]);
Packit 2035a7
            ++idx;
Packit 2035a7
            if (MathLib::isOctalDigit(iLiteral[idx])) {
Packit 2035a7
                tempBuf.push_back(iLiteral[idx]);
Packit 2035a7
            }
Packit 2035a7
        }
Packit 2035a7
        normalizedLiteral.push_back(static_cast<char>(MathLib::toLongNumber("0" + tempBuf)));
Packit 2035a7
    }
Packit 2035a7
    return normalizedLiteral;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::bigint MathLib::toLongNumber(const std::string & str)
Packit 2035a7
{
Packit 2035a7
    // hexadecimal numbers:
Packit 2035a7
    if (isIntHex(str)) {
Packit 2035a7
        if (str[0] == '-') {
Packit 2035a7
            bigint ret = 0;
Packit 2035a7
            std::istringstream istr(str);
Packit 2035a7
            istr >> std::hex >> ret;
Packit 2035a7
            return ret;
Packit 2035a7
        } else {
Packit 2035a7
            unsigned long long ret = 0;
Packit 2035a7
            std::istringstream istr(str);
Packit 2035a7
            istr >> std::hex >> ret;
Packit 2035a7
            return (bigint)ret;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    // octal numbers:
Packit 2035a7
    if (isOct(str)) {
Packit 2035a7
        bigint ret = 0;
Packit 2035a7
        std::istringstream istr(str);
Packit 2035a7
        istr >> std::oct >> ret;
Packit 2035a7
        return ret;
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    // binary numbers:
Packit 2035a7
    if (isBin(str)) {
Packit 2035a7
        bigint ret = 0;
Packit 2035a7
        for (std::string::size_type i = str[0] == '0'?2:3; i < str.length(); i++) {
Packit 2035a7
            ret <<= 1;
Packit 2035a7
            if (str[i] == '1')
Packit 2035a7
                ret |= 1;
Packit 2035a7
        }
Packit 2035a7
        if (str[0] == '-')
Packit 2035a7
            ret = -ret;
Packit 2035a7
        return ret;
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    if (isFloat(str)) {
Packit 2035a7
        // Things are going to be less precise now: the value can't be represented in the bigint type.
Packit 2035a7
        // Use min/max values as an approximation. See #5843
Packit 2035a7
        const double doubleval = toDoubleNumber(str);
Packit 2035a7
        if (doubleval > (double)std::numeric_limits<bigint>::max())
Packit 2035a7
            return std::numeric_limits<bigint>::max();
Packit 2035a7
        else if (doubleval < (double)std::numeric_limits<bigint>::min())
Packit 2035a7
            return std::numeric_limits<bigint>::min();
Packit 2035a7
        else
Packit 2035a7
            return static_cast<bigint>(doubleval);
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    if (str[0] == '\'' && str.size() >= 3U && endsWith(str,'\'')) {
Packit 2035a7
        return characterLiteralToLongNumber(str.substr(1,str.size()-2));
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    if (str[0] == '-') {
Packit 2035a7
        bigint ret = 0;
Packit 2035a7
        std::istringstream istr(str);
Packit 2035a7
        istr >> ret;
Packit 2035a7
        return ret;
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    biguint ret = 0;
Packit 2035a7
    std::istringstream istr(str);
Packit 2035a7
    istr >> ret;
Packit 2035a7
    return ret;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
Packit 2035a7
double MathLib::toDoubleNumber(const std::string &str)
Packit 2035a7
{
Packit 2035a7
    if (str[0] == '\'' && str.size() >= 3U && endsWith(str,'\''))
Packit 2035a7
        return characterLiteralToLongNumber(str.substr(1,str.size()-2));
Packit 2035a7
    if (isIntHex(str))
Packit 2035a7
        return static_cast<double>(toLongNumber(str));
Packit 2035a7
    // nullcheck
Packit 2035a7
    if (isNullValue(str))
Packit 2035a7
        return 0.0;
Packit 2035a7
#ifdef __clang__
Packit 2035a7
    if (isFloat(str)) // Workaround libc++ bug at http://llvm.org/bugs/show_bug.cgi?id=17782
Packit 2035a7
        // TODO : handle locale
Packit 2035a7
        return std::strtod(str.c_str(), nullptr);
Packit 2035a7
#endif
Packit 2035a7
    // otherwise, convert to double
Packit 2035a7
    std::istringstream istr(str);
Packit 2035a7
    istr.imbue(std::locale::classic());
Packit 2035a7
    double ret;
Packit 2035a7
    istr >> ret;
Packit 2035a7
    return ret;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
template<> std::string MathLib::toString(double value)
Packit 2035a7
{
Packit 2035a7
    std::ostringstream result;
Packit 2035a7
    result.precision(12);
Packit 2035a7
    result << value;
Packit 2035a7
    if (result.str() == "-0")
Packit 2035a7
        return "0.0";
Packit 2035a7
    if (result.str().find('.') == std::string::npos)
Packit 2035a7
        return result.str() + ".0";
Packit 2035a7
    return result.str();
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isFloat(const std::string &str)
Packit 2035a7
{
Packit 2035a7
    return isDecimalFloat(str) || isFloatHex(str);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isDecimalFloat(const std::string &str)
Packit 2035a7
{
Packit 2035a7
    if (str.empty())
Packit 2035a7
        return false;
Packit 2035a7
    enum State {
Packit 2035a7
        START, BASE_PLUSMINUS, BASE_DIGITS1, LEADING_DECIMAL, TRAILING_DECIMAL, BASE_DIGITS2, E, MANTISSA_PLUSMINUS, MANTISSA_DIGITS, SUFFIX_F, SUFFIX_L
Packit 2035a7
    } state = START;
Packit 2035a7
    for (std::string::const_iterator it = str.begin(); it != str.end(); ++it) {
Packit 2035a7
        switch (state) {
Packit 2035a7
        case START:
Packit 2035a7
            if (*it=='+' || *it=='-')
Packit 2035a7
                state=BASE_PLUSMINUS;
Packit 2035a7
            else if (*it=='.')
Packit 2035a7
                state=LEADING_DECIMAL;
Packit 2035a7
            else if (std::isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state=BASE_DIGITS1;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case BASE_PLUSMINUS:
Packit 2035a7
            if (*it=='.')
Packit 2035a7
                state=LEADING_DECIMAL;
Packit 2035a7
            else if (std::isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state=BASE_DIGITS1;
Packit 2035a7
            else if (*it=='e' || *it=='E')
Packit 2035a7
                state=E;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case LEADING_DECIMAL:
Packit 2035a7
            if (std::isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state=BASE_DIGITS2;
Packit 2035a7
            else if (*it=='e' || *it=='E')
Packit 2035a7
                state=E;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case BASE_DIGITS1:
Packit 2035a7
            if (*it=='e' || *it=='E')
Packit 2035a7
                state=E;
Packit 2035a7
            else if (*it=='.')
Packit 2035a7
                state=TRAILING_DECIMAL;
Packit 2035a7
            else if (!std::isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case TRAILING_DECIMAL:
Packit 2035a7
            if (*it=='e' || *it=='E')
Packit 2035a7
                state=E;
Packit 2035a7
            else if (*it=='f' || *it=='F')
Packit 2035a7
                state=SUFFIX_F;
Packit 2035a7
            else if (*it=='l' || *it=='L')
Packit 2035a7
                state=SUFFIX_L;
Packit 2035a7
            else if (std::isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state=BASE_DIGITS2;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case BASE_DIGITS2:
Packit 2035a7
            if (*it=='e' || *it=='E')
Packit 2035a7
                state=E;
Packit 2035a7
            else if (*it=='f' || *it=='F')
Packit 2035a7
                state=SUFFIX_F;
Packit 2035a7
            else if (*it=='l' || *it=='L')
Packit 2035a7
                state=SUFFIX_L;
Packit 2035a7
            else if (!std::isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case E:
Packit 2035a7
            if (*it=='+' || *it=='-')
Packit 2035a7
                state=MANTISSA_PLUSMINUS;
Packit 2035a7
            else if (std::isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state=MANTISSA_DIGITS;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case MANTISSA_PLUSMINUS:
Packit 2035a7
            if (!std::isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                return false;
Packit 2035a7
            else
Packit 2035a7
                state=MANTISSA_DIGITS;
Packit 2035a7
            break;
Packit 2035a7
        case MANTISSA_DIGITS:
Packit 2035a7
            if (*it=='f' || *it=='F')
Packit 2035a7
                state=SUFFIX_F;
Packit 2035a7
            else if (*it=='l' || *it=='L')
Packit 2035a7
                state=SUFFIX_L;
Packit 2035a7
            else if (!std::isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case SUFFIX_F:
Packit 2035a7
            return false;
Packit 2035a7
        case SUFFIX_L:
Packit 2035a7
            return false;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
    return (state==BASE_DIGITS2 || state==MANTISSA_DIGITS || state==TRAILING_DECIMAL || state==SUFFIX_F || state==SUFFIX_L);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isNegative(const std::string &str)
Packit 2035a7
{
Packit 2035a7
    if (str.empty())
Packit 2035a7
        return false;
Packit 2035a7
    return (str[0] == '-');
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isPositive(const std::string &str)
Packit 2035a7
{
Packit 2035a7
    if (str.empty())
Packit 2035a7
        return false;
Packit 2035a7
    return !MathLib::isNegative(str);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
/*! \brief Does the string represent an octal number?
Packit 2035a7
 * In case leading or trailing white space is provided, the function
Packit 2035a7
 * returns false.
Packit 2035a7
 * Additional information can be found here:
Packit 2035a7
 * http://gcc.gnu.org/onlinedocs/gcc/Binary-constants.html
Packit 2035a7
 *
Packit 2035a7
 * \param str The string to check. In case the string is empty, the function returns false.
Packit 2035a7
 * \return Return true in case a octal number is provided and false otherwise.
Packit 2035a7
 **/
Packit 2035a7
bool MathLib::isOct(const std::string& str)
Packit 2035a7
{
Packit 2035a7
    enum Status {
Packit 2035a7
        START, PLUSMINUS, OCTAL_PREFIX, DIGITS
Packit 2035a7
    } state = START;
Packit 2035a7
    for (std::string::const_iterator it = str.begin(); it != str.end(); ++it) {
Packit 2035a7
        switch (state) {
Packit 2035a7
        case START:
Packit 2035a7
            if (*it == '+' || *it == '-')
Packit 2035a7
                state = PLUSMINUS;
Packit 2035a7
            else if (*it == '0')
Packit 2035a7
                state = OCTAL_PREFIX;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case PLUSMINUS:
Packit 2035a7
            if (*it == '0')
Packit 2035a7
                state = OCTAL_PREFIX;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case OCTAL_PREFIX:
Packit 2035a7
            if (isOctalDigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = DIGITS;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case DIGITS:
Packit 2035a7
            if (isOctalDigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = DIGITS;
Packit 2035a7
            else
Packit 2035a7
                return isValidIntegerSuffix(it,str.end());
Packit 2035a7
            break;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
    return state == DIGITS;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isIntHex(const std::string& str)
Packit 2035a7
{
Packit 2035a7
    enum Status {
Packit 2035a7
        START, PLUSMINUS, HEX_PREFIX, DIGIT, DIGITS
Packit 2035a7
    } state = START;
Packit 2035a7
    for (std::string::const_iterator it = str.begin(); it != str.end(); ++it) {
Packit 2035a7
        switch (state) {
Packit 2035a7
        case START:
Packit 2035a7
            if (*it == '+' || *it == '-')
Packit 2035a7
                state = PLUSMINUS;
Packit 2035a7
            else if (*it == '0')
Packit 2035a7
                state = HEX_PREFIX;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case PLUSMINUS:
Packit 2035a7
            if (*it == '0')
Packit 2035a7
                state = HEX_PREFIX;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case HEX_PREFIX:
Packit 2035a7
            if (*it == 'x' || *it == 'X')
Packit 2035a7
                state = DIGIT;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case DIGIT:
Packit 2035a7
            if (isxdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = DIGITS;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case DIGITS:
Packit 2035a7
            if (isxdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = DIGITS;
Packit 2035a7
            else
Packit 2035a7
                return isValidIntegerSuffix(it,str.end());
Packit 2035a7
            break;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
    return state == DIGITS;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isFloatHex(const std::string& str)
Packit 2035a7
{
Packit 2035a7
    enum Status {
Packit 2035a7
        START, PLUSMINUS, HEX_PREFIX, WHOLE_NUMBER_DIGIT, WHOLE_NUMBER_DIGITS, FRACTION, EXPONENT_DIGIT, EXPONENT_DIGITS
Packit 2035a7
    } state = START;
Packit 2035a7
    for (std::string::const_iterator it = str.begin(); it != str.end(); ++it) {
Packit 2035a7
        switch (state) {
Packit 2035a7
        case START:
Packit 2035a7
            if (*it == '+' || *it == '-')
Packit 2035a7
                state = PLUSMINUS;
Packit 2035a7
            else if (*it == '0')
Packit 2035a7
                state = HEX_PREFIX;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case PLUSMINUS:
Packit 2035a7
            if (*it == '0')
Packit 2035a7
                state = HEX_PREFIX;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case HEX_PREFIX:
Packit 2035a7
            if (*it == 'x' || *it == 'X')
Packit 2035a7
                state = WHOLE_NUMBER_DIGIT;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case WHOLE_NUMBER_DIGIT:
Packit 2035a7
            if (isxdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = WHOLE_NUMBER_DIGITS;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case WHOLE_NUMBER_DIGITS:
Packit 2035a7
            if (isxdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = WHOLE_NUMBER_DIGITS;
Packit 2035a7
            else if (*it=='.')
Packit 2035a7
                state = FRACTION;
Packit 2035a7
            else if (*it=='p' || *it=='P')
Packit 2035a7
                state = EXPONENT_DIGIT;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case FRACTION:
Packit 2035a7
            if (isxdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = FRACTION;
Packit 2035a7
            else if (*it=='p' || *it=='P')
Packit 2035a7
                state = EXPONENT_DIGIT;
Packit 2035a7
            break;
Packit 2035a7
        case EXPONENT_DIGIT:
Packit 2035a7
            if (isxdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = EXPONENT_DIGITS;
Packit 2035a7
            else if (*it=='+' || *it=='-')
Packit 2035a7
                state = EXPONENT_DIGITS;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case EXPONENT_DIGITS:
Packit 2035a7
            if (isxdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = EXPONENT_DIGITS;
Packit 2035a7
            else
Packit 2035a7
                return *it=='f'||*it=='F'||*it=='l'||*it=='L';
Packit 2035a7
            break;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
    return state==EXPONENT_DIGITS;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isValidIntegerSuffix(const std::string& str)
Packit 2035a7
{
Packit 2035a7
    return isValidIntegerSuffix(str.begin(), str.end());
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isValidIntegerSuffix(std::string::const_iterator it, std::string::const_iterator end)
Packit 2035a7
{
Packit 2035a7
    enum {START, SUFFIX_U, SUFFIX_UL, SUFFIX_ULL, SUFFIX_L, SUFFIX_LU, SUFFIX_LL, SUFFIX_LLU, SUFFIX_I, SUFFIX_I6, SUFFIX_I64, SUFFIX_UI, SUFFIX_UI6, SUFFIX_UI64} state = START;
Packit 2035a7
    for (; it != end; ++it) {
Packit 2035a7
        switch (state) {
Packit 2035a7
        case START:
Packit 2035a7
            if (*it == 'u' || *it == 'U')
Packit 2035a7
                state = SUFFIX_U;
Packit 2035a7
            else if (*it == 'l' || *it == 'L')
Packit 2035a7
                state = SUFFIX_L;
Packit 2035a7
            else if (*it == 'i')
Packit 2035a7
                state = SUFFIX_I;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case SUFFIX_U:
Packit 2035a7
            if (*it == 'l' || *it == 'L')
Packit 2035a7
                state = SUFFIX_UL; // UL
Packit 2035a7
            else if (*it == 'i')
Packit 2035a7
                state = SUFFIX_UI;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case SUFFIX_UL:
Packit 2035a7
            if (*it == 'l' || *it == 'L')
Packit 2035a7
                state = SUFFIX_ULL; // ULL
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case SUFFIX_L:
Packit 2035a7
            if (*it == 'u' || *it == 'U')
Packit 2035a7
                state = SUFFIX_LU; // LU
Packit 2035a7
            else if (*it == 'l' || *it == 'L')
Packit 2035a7
                state = SUFFIX_LL; // LL
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case SUFFIX_LU:
Packit 2035a7
            return false;
Packit 2035a7
        case SUFFIX_LL:
Packit 2035a7
            if (*it == 'u' || *it == 'U')
Packit 2035a7
                state = SUFFIX_LLU; // LLU
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case SUFFIX_I:
Packit 2035a7
            if (*it == '6')
Packit 2035a7
                state = SUFFIX_I6;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case SUFFIX_I6:
Packit 2035a7
            if (*it == '4')
Packit 2035a7
                state = SUFFIX_I64;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case SUFFIX_UI:
Packit 2035a7
            if (*it == '6')
Packit 2035a7
                state = SUFFIX_UI6;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case SUFFIX_UI6:
Packit 2035a7
            if (*it == '4')
Packit 2035a7
                state = SUFFIX_UI64;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        default:
Packit 2035a7
            return false;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
    return ((state == SUFFIX_U)   ||
Packit 2035a7
            (state == SUFFIX_L)   ||
Packit 2035a7
            (state == SUFFIX_UL)  ||
Packit 2035a7
            (state == SUFFIX_LU)  ||
Packit 2035a7
            (state == SUFFIX_LL)  ||
Packit 2035a7
            (state == SUFFIX_ULL) ||
Packit 2035a7
            (state == SUFFIX_LLU) ||
Packit 2035a7
            (state == SUFFIX_I64) ||
Packit 2035a7
            (state == SUFFIX_UI64));
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
/*! \brief Does the string represent a binary number?
Packit 2035a7
 * In case leading or trailing white space is provided, the function
Packit 2035a7
 * returns false.
Packit 2035a7
 * Additional information can be found here:
Packit 2035a7
 * http://gcc.gnu.org/onlinedocs/gcc/Binary-constants.html
Packit 2035a7
 *
Packit 2035a7
 * \param str The string to check. In case the string is empty, the function returns false.
Packit 2035a7
 * \return Return true in case a binary number is provided and false otherwise.
Packit 2035a7
 **/
Packit 2035a7
bool MathLib::isBin(const std::string& str)
Packit 2035a7
{
Packit 2035a7
    enum Status {
Packit 2035a7
        START, PLUSMINUS, GNU_BIN_PREFIX, DIGIT, DIGITS
Packit 2035a7
    } state = START;
Packit 2035a7
    for (std::string::const_iterator it = str.begin(); it != str.end(); ++it) {
Packit 2035a7
        switch (state) {
Packit 2035a7
        case START:
Packit 2035a7
            if (*it == '+' || *it == '-')
Packit 2035a7
                state = PLUSMINUS;
Packit 2035a7
            else if (*it == '0')
Packit 2035a7
                state = GNU_BIN_PREFIX;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case PLUSMINUS:
Packit 2035a7
            if (*it == '0')
Packit 2035a7
                state = GNU_BIN_PREFIX;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case GNU_BIN_PREFIX:
Packit 2035a7
            if (*it == 'b' || *it == 'B')
Packit 2035a7
                state = DIGIT;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case DIGIT:
Packit 2035a7
            if (*it == '0' || *it == '1')
Packit 2035a7
                state = DIGITS;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case DIGITS:
Packit 2035a7
            if (*it == '0' || *it == '1')
Packit 2035a7
                state = DIGITS;
Packit 2035a7
            else
Packit 2035a7
                return isValidIntegerSuffix(it,str.end());
Packit 2035a7
            break;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
    return state == DIGITS;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isDec(const std::string & str)
Packit 2035a7
{
Packit 2035a7
    enum Status {
Packit 2035a7
        START, PLUSMINUS, DIGIT
Packit 2035a7
    } state = START;
Packit 2035a7
    for (std::string::const_iterator it = str.begin(); it != str.end(); ++it) {
Packit 2035a7
        switch (state) {
Packit 2035a7
        case START:
Packit 2035a7
            if (*it == '+' || *it == '-')
Packit 2035a7
                state = PLUSMINUS;
Packit 2035a7
            else if (isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = DIGIT;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case PLUSMINUS:
Packit 2035a7
            if (isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = DIGIT;
Packit 2035a7
            else
Packit 2035a7
                return false;
Packit 2035a7
            break;
Packit 2035a7
        case DIGIT:
Packit 2035a7
            if (isdigit(static_cast<unsigned char>(*it)))
Packit 2035a7
                state = DIGIT;
Packit 2035a7
            else
Packit 2035a7
                return isValidIntegerSuffix(it,str.end());
Packit 2035a7
            break;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
    return state == DIGIT;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isInt(const std::string & str)
Packit 2035a7
{
Packit 2035a7
    return isDec(str) || isIntHex(str) || isOct(str) || isBin(str);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::getSuffix(const std::string& value)
Packit 2035a7
{
Packit 2035a7
    if (value.size() > 3 && value[value.size() - 3] == 'i' && value[value.size() - 2] == '6' && value[value.size() - 1] == '4') {
Packit 2035a7
        if (value[value.size() - 4] == 'u')
Packit 2035a7
            return "ULL";
Packit 2035a7
        return "LL";
Packit 2035a7
    }
Packit 2035a7
    bool isUnsigned = false;
Packit 2035a7
    unsigned int longState = 0;
Packit 2035a7
    for (std::size_t i = 1U; i < value.size(); ++i) {
Packit 2035a7
        char c = value[value.size() - i];
Packit 2035a7
        if (c == 'u' || c == 'U')
Packit 2035a7
            isUnsigned = true;
Packit 2035a7
        else if (c == 'L' || c == 'l')
Packit 2035a7
            longState++;
Packit 2035a7
        else break;
Packit 2035a7
    }
Packit 2035a7
    if (longState == 0)
Packit 2035a7
        return isUnsigned ? "U" : "";
Packit 2035a7
    if (longState == 1)
Packit 2035a7
        return isUnsigned ? "UL" : "L";
Packit 2035a7
    if (longState == 2)
Packit 2035a7
        return isUnsigned ? "ULL" : "LL";
Packit 2035a7
    else return "";
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
static std::string intsuffix(const std::string & first, const std::string & second)
Packit 2035a7
{
Packit 2035a7
    const std::string suffix1 = MathLib::getSuffix(first);
Packit 2035a7
    const std::string suffix2 = MathLib::getSuffix(second);
Packit 2035a7
    if (suffix1 == "ULL" || suffix2 == "ULL")
Packit 2035a7
        return "ULL";
Packit 2035a7
    if (suffix1 == "LL" || suffix2 == "LL")
Packit 2035a7
        return "LL";
Packit 2035a7
    if (suffix1 == "UL" || suffix2 == "UL")
Packit 2035a7
        return "UL";
Packit 2035a7
    if (suffix1 == "L" || suffix2 == "L")
Packit 2035a7
        return "L";
Packit 2035a7
    if (suffix1 == "U" || suffix2 == "U")
Packit 2035a7
        return "U";
Packit 2035a7
Packit 2035a7
    return suffix1.empty() ? suffix2 : suffix1;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::add(const std::string & first, const std::string & second)
Packit 2035a7
{
Packit 2035a7
#ifdef TEST_MATHLIB_VALUE
Packit 2035a7
    return (value(first) + value(second)).str();
Packit 2035a7
#else
Packit 2035a7
    if (MathLib::isInt(first) && MathLib::isInt(second)) {
Packit 2035a7
        return toString(toLongNumber(first) + toLongNumber(second)) + intsuffix(first, second);
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    double d1 = toDoubleNumber(first);
Packit 2035a7
    double d2 = toDoubleNumber(second);
Packit 2035a7
Packit 2035a7
    int count = 0;
Packit 2035a7
    while (d1 > 100000.0 * d2 && toString(d1+d2)==first && ++count<5)
Packit 2035a7
        d2 *= 10.0;
Packit 2035a7
    while (d2 > 100000.0 * d1 && toString(d1+d2)==second && ++count<5)
Packit 2035a7
        d1 *= 10.0;
Packit 2035a7
Packit 2035a7
    return toString(d1 + d2);
Packit 2035a7
#endif
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::subtract(const std::string &first, const std::string &second)
Packit 2035a7
{
Packit 2035a7
#ifdef TEST_MATHLIB_VALUE
Packit 2035a7
    return (value(first) - value(second)).str();
Packit 2035a7
#else
Packit 2035a7
    if (MathLib::isInt(first) && MathLib::isInt(second)) {
Packit 2035a7
        return toString(toLongNumber(first) - toLongNumber(second)) + intsuffix(first, second);
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    if (first == second)
Packit 2035a7
        return "0.0" ;
Packit 2035a7
Packit 2035a7
    double d1 = toDoubleNumber(first);
Packit 2035a7
    double d2 = toDoubleNumber(second);
Packit 2035a7
Packit 2035a7
    int count = 0;
Packit 2035a7
    while (d1 > 100000.0 * d2 && toString(d1-d2)==first && ++count<5)
Packit 2035a7
        d2 *= 10.0;
Packit 2035a7
    while (d2 > 100000.0 * d1 && toString(d1-d2)==second && ++count<5)
Packit 2035a7
        d1 *= 10.0;
Packit 2035a7
Packit 2035a7
    return toString(d1 - d2);
Packit 2035a7
#endif
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::incdec(const std::string & var, const std::string & op)
Packit 2035a7
{
Packit 2035a7
#ifdef TEST_MATHLIB_VALUE
Packit 2035a7
    if (op == "++")
Packit 2035a7
        return value(var).add(1).str();
Packit 2035a7
    else if (op == "--")
Packit 2035a7
        return value(var).add(-1).str();
Packit 2035a7
#else
Packit 2035a7
    if (op == "++")
Packit 2035a7
        return MathLib::add(var, "1");
Packit 2035a7
    else if (op == "--")
Packit 2035a7
        return MathLib::subtract(var, "1");
Packit 2035a7
#endif
Packit 2035a7
Packit 2035a7
    throw InternalError(nullptr, std::string("Unexpected operation '") + op + "' in MathLib::incdec(). Please report this to Cppcheck developers.");
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::divide(const std::string &first, const std::string &second)
Packit 2035a7
{
Packit 2035a7
#ifdef TEST_MATHLIB_VALUE
Packit 2035a7
    return (value(first) / value(second)).str();
Packit 2035a7
#else
Packit 2035a7
    if (MathLib::isInt(first) && MathLib::isInt(second)) {
Packit 2035a7
        const bigint a = toLongNumber(first);
Packit 2035a7
        const bigint b = toLongNumber(second);
Packit 2035a7
        if (b == 0)
Packit 2035a7
            throw InternalError(nullptr, "Internal Error: Division by zero");
Packit 2035a7
        if (a == std::numeric_limits<bigint>::min() && std::abs(b)<=1)
Packit 2035a7
            throw InternalError(nullptr, "Internal Error: Division overflow");
Packit 2035a7
        return toString(toLongNumber(first) / b) + intsuffix(first, second);
Packit 2035a7
    } else if (isNullValue(second)) {
Packit 2035a7
        if (isNullValue(first))
Packit 2035a7
            return "nan.0";
Packit 2035a7
        return isPositive(first) ? "inf.0" : "-inf.0";
Packit 2035a7
    }
Packit 2035a7
    return toString(toDoubleNumber(first) / toDoubleNumber(second));
Packit 2035a7
#endif
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::multiply(const std::string &first, const std::string &second)
Packit 2035a7
{
Packit 2035a7
#ifdef TEST_MATHLIB_VALUE
Packit 2035a7
    return (value(first) * value(second)).str();
Packit 2035a7
#else
Packit 2035a7
    if (MathLib::isInt(first) && MathLib::isInt(second)) {
Packit 2035a7
        return toString(toLongNumber(first) * toLongNumber(second)) + intsuffix(first, second);
Packit 2035a7
    }
Packit 2035a7
    return toString(toDoubleNumber(first) * toDoubleNumber(second));
Packit 2035a7
#endif
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::mod(const std::string &first, const std::string &second)
Packit 2035a7
{
Packit 2035a7
#ifdef TEST_MATHLIB_VALUE
Packit 2035a7
    return (value(first) % value(second)).str();
Packit 2035a7
#else
Packit 2035a7
    if (MathLib::isInt(first) && MathLib::isInt(second)) {
Packit 2035a7
        const bigint b = toLongNumber(second);
Packit 2035a7
        if (b == 0)
Packit 2035a7
            throw InternalError(nullptr, "Internal Error: Division by zero");
Packit 2035a7
        return toString(toLongNumber(first) % b) + intsuffix(first, second);
Packit 2035a7
    }
Packit 2035a7
    return toString(std::fmod(toDoubleNumber(first),toDoubleNumber(second)));
Packit 2035a7
#endif
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::calculate(const std::string &first, const std::string &second, char action)
Packit 2035a7
{
Packit 2035a7
    switch (action) {
Packit 2035a7
    case '+':
Packit 2035a7
        return MathLib::add(first, second);
Packit 2035a7
Packit 2035a7
    case '-':
Packit 2035a7
        return MathLib::subtract(first, second);
Packit 2035a7
Packit 2035a7
    case '*':
Packit 2035a7
        return MathLib::multiply(first, second);
Packit 2035a7
Packit 2035a7
    case '/':
Packit 2035a7
        return MathLib::divide(first, second);
Packit 2035a7
Packit 2035a7
    case '%':
Packit 2035a7
        return MathLib::mod(first, second);
Packit 2035a7
Packit 2035a7
    case '&':
Packit 2035a7
        return MathLib::toString(MathLib::toLongNumber(first) & MathLib::toLongNumber(second)) + intsuffix(first,second);
Packit 2035a7
Packit 2035a7
    case '|':
Packit 2035a7
        return MathLib::toString(MathLib::toLongNumber(first) | MathLib::toLongNumber(second)) + intsuffix(first,second);
Packit 2035a7
Packit 2035a7
    case '^':
Packit 2035a7
        return MathLib::toString(MathLib::toLongNumber(first) ^ MathLib::toLongNumber(second)) + intsuffix(first,second);
Packit 2035a7
Packit 2035a7
    default:
Packit 2035a7
        throw InternalError(nullptr, std::string("Unexpected action '") + action + "' in MathLib::calculate(). Please report this to Cppcheck developers.");
Packit 2035a7
    }
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::sin(const std::string &tok)
Packit 2035a7
{
Packit 2035a7
    return toString(std::sin(toDoubleNumber(tok)));
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
Packit 2035a7
std::string MathLib::cos(const std::string &tok)
Packit 2035a7
{
Packit 2035a7
    return toString(std::cos(toDoubleNumber(tok)));
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
std::string MathLib::tan(const std::string &tok)
Packit 2035a7
{
Packit 2035a7
    return toString(std::tan(toDoubleNumber(tok)));
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
Packit 2035a7
std::string MathLib::abs(const std::string &tok)
Packit 2035a7
{
Packit 2035a7
    return toString(std::abs(toDoubleNumber(tok)));
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isEqual(const std::string &first, const std::string &second)
Packit 2035a7
{
Packit 2035a7
    // this conversion is needed for formatting
Packit 2035a7
    // e.g. if first=0.1 and second=1.0E-1, the direct comparison of the strings would fail
Packit 2035a7
    return toString(toDoubleNumber(first)) == toString(toDoubleNumber(second));
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isNotEqual(const std::string &first, const std::string &second)
Packit 2035a7
{
Packit 2035a7
    return !isEqual(first, second);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isGreater(const std::string &first, const std::string &second)
Packit 2035a7
{
Packit 2035a7
    return toDoubleNumber(first) > toDoubleNumber(second);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isGreaterEqual(const std::string &first, const std::string &second)
Packit 2035a7
{
Packit 2035a7
    return toDoubleNumber(first) >= toDoubleNumber(second);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isLess(const std::string &first, const std::string &second)
Packit 2035a7
{
Packit 2035a7
    return toDoubleNumber(first) < toDoubleNumber(second);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isLessEqual(const std::string &first, const std::string &second)
Packit 2035a7
{
Packit 2035a7
    return toDoubleNumber(first) <= toDoubleNumber(second);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
/*! \brief Does the string represent the numerical value of 0?
Packit 2035a7
 * In case leading or trailing white space is provided, the function
Packit 2035a7
 * returns false.
Packit 2035a7
 * Requirement for this function:
Packit 2035a7
 * - This code is allowed to be slow because of simplicity of the code.
Packit 2035a7
 *
Packit 2035a7
 * \param[in] str The string to check. In case the string is empty, the function returns false.
Packit 2035a7
 * \return Return true in case the string represents a numerical null value.
Packit 2035a7
 **/
Packit 2035a7
bool MathLib::isNullValue(const std::string &str)
Packit 2035a7
{
Packit 2035a7
    if (str.empty() || (!std::isdigit(static_cast<unsigned char>(str[0])) && (str.size() < 1 || (str[0] != '.' && str[0] != '-' && str[0] != '+'))))
Packit 2035a7
        return false; // Has to be a number
Packit 2035a7
Packit 2035a7
    for (size_t i = 0; i < str.size(); i++) {
Packit 2035a7
        if (std::isdigit(static_cast<unsigned char>(str[i])) && str[i] != '0') // May not contain digits other than 0
Packit 2035a7
            return false;
Packit 2035a7
        if (str[i] == 'E' || str[i] == 'e')
Packit 2035a7
            return true;
Packit 2035a7
    }
Packit 2035a7
    return true;
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isOctalDigit(char c)
Packit 2035a7
{
Packit 2035a7
    return (c >= '0' && c <= '7');
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
bool MathLib::isDigitSeparator(const std::string& iCode, std::string::size_type iPos)
Packit 2035a7
{
Packit 2035a7
    if (iPos == 0 || iPos >= iCode.size() || iCode[iPos] != '\'')
Packit 2035a7
        return false;
Packit 2035a7
    std::string::size_type i = iPos - 1;
Packit 2035a7
    while (std::isxdigit(iCode[i])) {
Packit 2035a7
        if (i == 0)
Packit 2035a7
            return true; // Only xdigits before '
Packit 2035a7
        --i;
Packit 2035a7
    }
Packit 2035a7
    if (i == iPos - 1) { // No xdigit before '
Packit 2035a7
        return false;
Packit 2035a7
    } else {
Packit 2035a7
        switch (iCode[i]) {
Packit 2035a7
        case ' ':
Packit 2035a7
        case '.':
Packit 2035a7
        case ',':
Packit 2035a7
        case 'x':
Packit 2035a7
        case '(':
Packit 2035a7
        case '{':
Packit 2035a7
        case '+':
Packit 2035a7
        case '-':
Packit 2035a7
        case '*':
Packit 2035a7
        case '%':
Packit 2035a7
        case '/':
Packit 2035a7
        case '&':
Packit 2035a7
        case '|':
Packit 2035a7
        case '^':
Packit 2035a7
        case '~':
Packit 2035a7
        case '=':
Packit 2035a7
            return true;
Packit 2035a7
        case '\'':
Packit 2035a7
            return isDigitSeparator(iCode, i);
Packit 2035a7
        default:
Packit 2035a7
            return false;
Packit 2035a7
        }
Packit 2035a7
    }
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value operator+(const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    return MathLib::value::calc('+',v1,v2);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value operator-(const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    return MathLib::value::calc('-',v1,v2);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value operator*(const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    return MathLib::value::calc('*',v1,v2);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value operator/(const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    return MathLib::value::calc('/',v1,v2);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value operator%(const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    return MathLib::value::calc('%',v1,v2);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value operator&(const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    return MathLib::value::calc('&',v1,v2);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value operator|(const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    return MathLib::value::calc('|',v1,v2);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value operator^(const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    return MathLib::value::calc('^',v1,v2);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value operator<<(const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    return v1.shiftLeft(v2);
Packit 2035a7
}
Packit 2035a7
Packit 2035a7
MathLib::value operator>>(const MathLib::value &v1, const MathLib::value &v2)
Packit 2035a7
{
Packit 2035a7
    return v1.shiftRight(v2);
Packit 2035a7
}