/* * Cppcheck - A tool for static C/C++ code analysis * Copyright (C) 2007-2009 Daniel Marjamäki and Cppcheck team. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see #include #include #include #include #include #include long MathLib::toLongNumber(const std::string &str) { if (strncmp(str.c_str(), "0x", 2) == 0) { return std::strtoul(str.c_str(), '\0', 16); } if (strncmp(str.c_str(), "0", 1) == 0) { return std::strtoul(str.c_str(), '\0', 8); } return std::atol(str.c_str()); } double MathLib::toDoubleNumber(const std::string &str) { if (strncmp(str.c_str(), "0x", 2) == 0) { return std::strtoul(str.c_str(), '\0', 16); } return std::atof(str.c_str()); } template std::string MathLib::toString(T d) { std::ostringstream result; result << d; return result.str(); } bool MathLib::isInt(const std::string & str) { if (str.find(".", 0) != std::string::npos || str.find("e", 0) != std::string::npos || str.find("E", 0) != std::string::npos) { return false; } return true; } std::string MathLib::add(const std::string & first, const std::string & second) { if (MathLib::isInt(first) && MathLib::isInt(second)) { return toString(toLongNumber(first) + toLongNumber(second)); } return toString(toDoubleNumber(first) + toDoubleNumber(second)); } std::string MathLib::subtract(const std::string &first, const std::string &second) { if (MathLib::isInt(first) && MathLib::isInt(second)) { return toString(toLongNumber(first) - toLongNumber(second)); } return toString(toDoubleNumber(first) - toDoubleNumber(second)); } std::string MathLib::divide(const std::string &first, const std::string &second) { if (MathLib::isInt(first) && MathLib::isInt(second)) { return toString(toLongNumber(first) / toLongNumber(second)); } return toString(toDoubleNumber(first) / toDoubleNumber(second)); } std::string MathLib::multiply(const std::string &first, const std::string &second) { if (MathLib::isInt(first) && MathLib::isInt(second)) { return toString(toLongNumber(first) * toLongNumber(second)); } return toString(toDoubleNumber(first) * toDoubleNumber(second)); } std::string MathLib::calculate(const std::string &first, const std::string &second, char action) { std::string result("0"); switch (action) { case '+': result = MathLib::add(first, second); break; case '-': result = MathLib::subtract(first, second); break; case '*': result = MathLib::multiply(first, second); break; case '/': result = MathLib::divide(first, second); break; default: std::cout << "##### If you see this, there is a bug: " << "MathLib::calculate() was called with unknown action '" << action << "' #####" << std::endl; break; } return result; } std::string MathLib::sin(const std::string &tok) { return toString(std::sin(toDoubleNumber(tok))); } std::string MathLib::cos(const std::string &tok) { return toString(std::cos(toDoubleNumber(tok))); } std::string MathLib::tan(const std::string &tok) { return toString(std::tan(toDoubleNumber(tok))); } std::string MathLib::abs(const std::string &tok) { return toString(std::abs(toDoubleNumber(tok))); } bool MathLib::isGreater(const std::string &first, const std::string &second) { return toDoubleNumber(first) > toDoubleNumber(second); }