2010-04-13 19:25:08 +02:00
|
|
|
/*
|
|
|
|
* Cppcheck - A tool for static C/C++ code analysis
|
2015-01-03 12:14:58 +01:00
|
|
|
* Copyright (C) 2007-2015 Daniel Marjamäki and Cppcheck team.
|
2010-04-13 19:25:08 +02:00
|
|
|
*
|
|
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
#include "cppcheck.h"
|
|
|
|
|
2011-12-08 17:42:26 +01:00
|
|
|
#include "preprocessor.h" // Preprocessor
|
|
|
|
#include "tokenize.h" // Tokenizer
|
2010-04-13 19:25:08 +02:00
|
|
|
|
|
|
|
#include "check.h"
|
2010-07-19 14:05:44 +02:00
|
|
|
#include "path.h"
|
2010-04-13 19:25:08 +02:00
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include <fstream>
|
2012-04-08 14:18:13 +02:00
|
|
|
#include <sstream>
|
2010-09-06 21:00:56 +02:00
|
|
|
#include <stdexcept>
|
2010-08-31 22:18:07 +02:00
|
|
|
#include "timer.h"
|
2014-01-04 10:41:13 +01:00
|
|
|
#include "version.h"
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2011-02-14 19:37:58 +01:00
|
|
|
#ifdef HAVE_RULES
|
2010-12-12 11:56:22 +01:00
|
|
|
#define PCRE_STATIC
|
|
|
|
#include <pcre.h>
|
2010-12-31 10:24:51 +01:00
|
|
|
#endif
|
2010-12-12 11:56:22 +01:00
|
|
|
|
2014-01-04 10:41:13 +01:00
|
|
|
static const char Version[] = CPPCHECK_VERSION_STRING;
|
2011-08-11 16:34:59 +02:00
|
|
|
static const char ExtraVersion[] = "";
|
|
|
|
|
2010-04-13 19:25:08 +02:00
|
|
|
static TimerResults S_timerResults;
|
|
|
|
|
2011-02-16 02:12:15 +01:00
|
|
|
CppCheck::CppCheck(ErrorLogger &errorLogger, bool useGlobalSuppressions)
|
2013-05-11 11:35:04 +02:00
|
|
|
: _errorLogger(errorLogger), exitcode(0), _useGlobalSuppressions(useGlobalSuppressions), tooManyConfigs(false), _simplify(true)
|
2010-04-13 19:25:08 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
CppCheck::~CppCheck()
|
|
|
|
{
|
2014-11-15 10:43:49 +01:00
|
|
|
while (!fileInfo.empty()) {
|
|
|
|
delete fileInfo.back();
|
|
|
|
fileInfo.pop_back();
|
|
|
|
}
|
2014-01-03 10:24:57 +01:00
|
|
|
S_timerResults.ShowResults(_settings._showtime);
|
2010-04-13 19:25:08 +02:00
|
|
|
}
|
|
|
|
|
2011-04-24 18:10:25 +02:00
|
|
|
const char * CppCheck::version()
|
2010-04-13 19:25:08 +02:00
|
|
|
{
|
2011-08-11 16:34:59 +02:00
|
|
|
return Version;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char * CppCheck::extraVersion()
|
|
|
|
{
|
|
|
|
return ExtraVersion;
|
2010-04-13 19:25:08 +02:00
|
|
|
}
|
|
|
|
|
2011-04-24 18:10:25 +02:00
|
|
|
unsigned int CppCheck::check(const std::string &path)
|
2010-04-13 19:25:08 +02:00
|
|
|
{
|
2014-10-03 10:40:48 +02:00
|
|
|
std::ifstream fin(path.c_str());
|
|
|
|
return processFile(path, fin);
|
2010-04-13 19:25:08 +02:00
|
|
|
}
|
|
|
|
|
2011-04-24 18:10:25 +02:00
|
|
|
unsigned int CppCheck::check(const std::string &path, const std::string &content)
|
2010-04-13 19:25:08 +02:00
|
|
|
{
|
2014-10-03 10:40:48 +02:00
|
|
|
std::istringstream iss(content);
|
|
|
|
return processFile(path, iss);
|
2010-04-13 19:25:08 +02:00
|
|
|
}
|
|
|
|
|
2012-02-24 20:45:56 +01:00
|
|
|
void CppCheck::replaceAll(std::string& code, const std::string &from, const std::string &to)
|
2011-08-16 21:58:27 +02:00
|
|
|
{
|
2012-07-08 23:39:46 +02:00
|
|
|
std::size_t pos = 0;
|
2011-10-13 20:53:06 +02:00
|
|
|
while ((pos = code.find(from, pos)) != std::string::npos) {
|
2011-08-16 21:58:27 +02:00
|
|
|
code.replace(pos, from.length(), to);
|
|
|
|
pos += to.length();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CppCheck::findError(std::string code, const char FileName[])
|
|
|
|
{
|
2014-09-02 18:05:02 +02:00
|
|
|
std::set<unsigned long long> checksums;
|
2011-08-16 21:58:27 +02:00
|
|
|
// First make sure that error occurs with the original code
|
2015-05-25 21:15:55 +02:00
|
|
|
bool internalErrorFound(false);
|
|
|
|
checkFile(code, FileName, checksums, internalErrorFound);
|
2011-10-13 20:53:06 +02:00
|
|
|
if (_errorList.empty()) {
|
2011-08-16 21:58:27 +02:00
|
|
|
// Error does not occur with this code
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-05-25 21:15:55 +02:00
|
|
|
const std::string previousCode = code;
|
2011-08-16 21:58:27 +02:00
|
|
|
std::string error = _errorList.front();
|
2011-10-13 20:53:06 +02:00
|
|
|
for (;;) {
|
2011-08-16 21:58:27 +02:00
|
|
|
|
|
|
|
// Try to remove included files from the source
|
2015-05-25 21:15:55 +02:00
|
|
|
const std::size_t found = previousCode.rfind("\n#endfile");
|
2011-10-13 20:53:06 +02:00
|
|
|
if (found == std::string::npos) {
|
2011-08-16 21:58:27 +02:00
|
|
|
// No modifications can be done to the code
|
2011-10-13 20:53:06 +02:00
|
|
|
} else {
|
2011-08-16 21:58:27 +02:00
|
|
|
// Modify code and re-check it to see if error
|
|
|
|
// is still there.
|
|
|
|
code = previousCode.substr(found+9);
|
|
|
|
_errorList.clear();
|
2014-09-02 18:05:02 +02:00
|
|
|
checksums.clear();
|
2015-05-25 21:15:55 +02:00
|
|
|
checkFile(code, FileName, checksums, internalErrorFound);
|
2011-08-16 21:58:27 +02:00
|
|
|
}
|
|
|
|
|
2011-10-13 20:53:06 +02:00
|
|
|
if (_errorList.empty()) {
|
2011-08-16 21:58:27 +02:00
|
|
|
// Latest code didn't fail anymore. Fall back
|
|
|
|
// to previous code
|
|
|
|
code = previousCode;
|
2011-10-13 20:53:06 +02:00
|
|
|
} else {
|
2011-08-16 21:58:27 +02:00
|
|
|
error = _errorList.front();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add '\n' so that "\n#file" on first line would be found
|
|
|
|
code = "// " + error + "\n" + code;
|
2012-02-24 20:45:56 +01:00
|
|
|
replaceAll(code, "\n#file", "\n// #file");
|
|
|
|
replaceAll(code, "\n#endfile", "\n// #endfile");
|
2011-08-16 21:58:27 +02:00
|
|
|
|
|
|
|
// We have reduced the code as much as we can. Print out
|
|
|
|
// the code and quit.
|
|
|
|
_errorLogger.reportOut(code);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-10-03 10:40:48 +02:00
|
|
|
unsigned int CppCheck::processFile(const std::string& filename, std::istream& fileStream)
|
2010-04-13 19:25:08 +02:00
|
|
|
{
|
|
|
|
exitcode = 0;
|
|
|
|
|
2012-10-03 19:51:09 +02:00
|
|
|
// only show debug warnings for accepted C/C++ source files
|
2013-10-31 19:09:01 +01:00
|
|
|
if (!Path::acceptFile(filename))
|
2012-01-06 16:08:08 +01:00
|
|
|
_settings.debugwarnings = false;
|
|
|
|
|
2011-04-24 18:10:25 +02:00
|
|
|
if (_settings.terminated())
|
|
|
|
return exitcode;
|
|
|
|
|
2011-10-13 20:53:06 +02:00
|
|
|
if (_settings._errorsOnly == false) {
|
2014-04-02 13:56:34 +02:00
|
|
|
std::string fixedpath = Path::simplifyPath(filename);
|
2011-04-24 18:10:25 +02:00
|
|
|
fixedpath = Path::toNativeSeparators(fixedpath);
|
|
|
|
_errorLogger.reportOut(std::string("Checking ") + fixedpath + std::string("..."));
|
|
|
|
}
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2015-05-25 21:15:55 +02:00
|
|
|
bool internalErrorFound(false);
|
2011-10-13 20:53:06 +02:00
|
|
|
try {
|
2011-04-24 18:10:25 +02:00
|
|
|
Preprocessor preprocessor(&_settings, this);
|
|
|
|
std::list<std::string> configurations;
|
2015-05-25 21:15:55 +02:00
|
|
|
std::string filedata;
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2014-10-03 10:40:48 +02:00
|
|
|
{
|
2011-04-24 18:10:25 +02:00
|
|
|
Timer t("Preprocessor::preprocess", _settings._showtime, &S_timerResults);
|
2014-10-03 10:40:48 +02:00
|
|
|
preprocessor.preprocess(fileStream, filedata, configurations, filename, _settings._includePaths);
|
2010-07-19 14:05:44 +02:00
|
|
|
}
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2013-08-31 18:38:52 +02:00
|
|
|
if (_settings.checkConfiguration) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-07-12 16:09:00 +02:00
|
|
|
// Run rules on this code
|
|
|
|
for (std::list<Settings::Rule>::const_iterator it = _settings.rules.begin(); it != _settings.rules.end(); ++it) {
|
|
|
|
if (it->tokenlist == "define") {
|
|
|
|
Tokenizer tokenizer2(&_settings, this);
|
|
|
|
std::istringstream istr2(filedata);
|
|
|
|
tokenizer2.list.createTokens(istr2, filename);
|
|
|
|
|
|
|
|
for (const Token *tok = tokenizer2.list.front(); tok; tok = tok->next()) {
|
|
|
|
if (tok->str() == "#define") {
|
|
|
|
std::string code = std::string(tok->linenr()-1U, '\n');
|
|
|
|
for (const Token *tok2 = tok; tok2 && tok2->linenr() == tok->linenr(); tok2 = tok2->next())
|
|
|
|
code += " " + tok2->str();
|
|
|
|
Tokenizer tokenizer3(&_settings, this);
|
|
|
|
std::istringstream istr3(code);
|
|
|
|
tokenizer3.list.createTokens(istr3, tokenizer2.list.file(tok));
|
|
|
|
executeRules("define", tokenizer3);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-08 16:46:54 +02:00
|
|
|
if (!_settings.userDefines.empty() && _settings._maxConfigs==1U) {
|
2011-04-24 18:10:25 +02:00
|
|
|
configurations.clear();
|
|
|
|
configurations.push_back(_settings.userDefines);
|
|
|
|
}
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2012-12-26 18:35:49 +01:00
|
|
|
if (!_settings._force && configurations.size() > _settings._maxConfigs) {
|
|
|
|
if (_settings.isEnabled("information")) {
|
|
|
|
tooManyConfigsError(Path::toNativeSeparators(filename),configurations.size());
|
|
|
|
} else {
|
|
|
|
tooManyConfigs = true;
|
|
|
|
}
|
2012-05-09 18:54:43 +02:00
|
|
|
}
|
|
|
|
|
2014-09-02 18:05:02 +02:00
|
|
|
std::set<unsigned long long> checksums;
|
2012-05-11 19:38:19 +02:00
|
|
|
unsigned int checkCount = 0;
|
2011-10-13 20:53:06 +02:00
|
|
|
for (std::list<std::string>::const_iterator it = configurations.begin(); it != configurations.end(); ++it) {
|
2011-10-05 07:37:43 +02:00
|
|
|
// Check only a few configurations (default 12), after that bail out, unless --force
|
2011-04-24 18:10:25 +02:00
|
|
|
// was used.
|
2012-12-26 18:35:49 +01:00
|
|
|
if (!_settings._force && ++checkCount > _settings._maxConfigs)
|
2011-04-24 18:10:25 +02:00
|
|
|
break;
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2011-04-24 18:10:25 +02:00
|
|
|
cfg = *it;
|
2010-08-07 13:08:36 +02:00
|
|
|
|
2011-04-24 18:10:25 +02:00
|
|
|
// If only errors are printed, print filename after the check
|
2011-10-13 20:53:06 +02:00
|
|
|
if (_settings._errorsOnly == false && it != configurations.begin()) {
|
2014-04-02 13:56:34 +02:00
|
|
|
std::string fixedpath = Path::simplifyPath(filename);
|
2011-04-24 18:10:25 +02:00
|
|
|
fixedpath = Path::toNativeSeparators(fixedpath);
|
2014-10-03 10:02:46 +02:00
|
|
|
_errorLogger.reportOut("Checking " + fixedpath + ": " + cfg + "...");
|
2010-05-16 07:38:29 +02:00
|
|
|
}
|
|
|
|
|
2013-06-08 16:46:54 +02:00
|
|
|
if (!_settings.userDefines.empty()) {
|
|
|
|
if (!cfg.empty())
|
|
|
|
cfg = ";" + cfg;
|
|
|
|
cfg = _settings.userDefines + cfg;
|
|
|
|
}
|
|
|
|
|
2012-07-10 20:29:04 +02:00
|
|
|
Timer t("Preprocessor::getcode", _settings._showtime, &S_timerResults);
|
2014-10-02 19:40:41 +02:00
|
|
|
std::string codeWithoutCfg = preprocessor.getcode(filedata, cfg, filename);
|
2012-07-10 20:29:04 +02:00
|
|
|
t.Stop();
|
|
|
|
|
2014-10-02 19:40:41 +02:00
|
|
|
codeWithoutCfg += _settings.append();
|
2010-04-15 22:45:38 +02:00
|
|
|
|
2011-10-13 20:53:06 +02:00
|
|
|
if (_settings.debugFalsePositive) {
|
2014-10-02 19:40:41 +02:00
|
|
|
if (findError(codeWithoutCfg, filename.c_str())) {
|
2011-08-16 21:58:27 +02:00
|
|
|
return exitcode;
|
|
|
|
}
|
2011-10-13 20:53:06 +02:00
|
|
|
} else {
|
2015-05-25 21:15:55 +02:00
|
|
|
if (!checkFile(codeWithoutCfg, filename.c_str(), checksums, internalErrorFound)) {
|
2014-09-02 18:05:02 +02:00
|
|
|
if (_settings.isEnabled("information") && (_settings.debug || _settings._verbose))
|
|
|
|
purgedConfigurationMessage(filename, cfg);
|
|
|
|
}
|
2011-08-16 21:58:27 +02:00
|
|
|
}
|
2010-04-13 19:25:08 +02:00
|
|
|
}
|
2012-04-16 16:25:04 +02:00
|
|
|
} catch (const std::runtime_error &e) {
|
2013-05-09 18:50:24 +02:00
|
|
|
internalError(filename, e.what());
|
|
|
|
} catch (const InternalError &e) {
|
|
|
|
internalError(filename, e.errorMessage);
|
2015-05-25 21:15:55 +02:00
|
|
|
exitcode=1; // e.g. reflect a syntax error
|
2011-04-24 18:10:25 +02:00
|
|
|
}
|
|
|
|
|
2015-01-07 19:26:16 +01:00
|
|
|
// In jointSuppressionReport mode, unmatched suppressions are
|
|
|
|
// collected after all files are processed
|
|
|
|
if (!_settings.jointSuppressionReport && (_settings.isEnabled("information") || _settings.checkConfiguration)) {
|
|
|
|
reportUnmatchedSuppressions(_settings.nomsg.getUnmatchedLocalSuppressions(filename, unusedFunctionCheckIsEnabled()));
|
|
|
|
}
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2011-07-25 13:25:09 +02:00
|
|
|
_errorList.clear();
|
2015-05-25 21:15:55 +02:00
|
|
|
if (internalErrorFound && (exitcode==0)) {
|
|
|
|
exitcode=1;
|
|
|
|
}
|
2011-07-25 13:25:09 +02:00
|
|
|
return exitcode;
|
|
|
|
}
|
|
|
|
|
2013-05-09 18:50:24 +02:00
|
|
|
void CppCheck::internalError(const std::string &filename, const std::string &msg)
|
|
|
|
{
|
|
|
|
const std::string fixedpath = Path::toNativeSeparators(filename);
|
2015-01-12 23:09:17 +01:00
|
|
|
const std::string fullmsg("Bailing out from checking " + fixedpath + " since there was an internal error: " + msg);
|
2013-05-09 18:50:24 +02:00
|
|
|
|
|
|
|
if (_settings.isEnabled("information")) {
|
|
|
|
const ErrorLogger::ErrorMessage::FileLocation loc1(filename, 0);
|
|
|
|
std::list<ErrorLogger::ErrorMessage::FileLocation> callstack;
|
|
|
|
callstack.push_back(loc1);
|
|
|
|
|
|
|
|
ErrorLogger::ErrorMessage errmsg(callstack,
|
|
|
|
Severity::information,
|
|
|
|
fullmsg,
|
|
|
|
"internalError",
|
|
|
|
false);
|
|
|
|
|
|
|
|
_errorLogger.reportErr(errmsg);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// Report on stdout
|
|
|
|
_errorLogger.reportOut(fullmsg);
|
|
|
|
}
|
|
|
|
}
|
2011-08-16 21:58:27 +02:00
|
|
|
|
2010-05-21 21:06:11 +02:00
|
|
|
void CppCheck::analyseFile(std::istream &fin, const std::string &filename)
|
|
|
|
{
|
|
|
|
// Preprocess file..
|
|
|
|
Preprocessor preprocessor(&_settings, this);
|
|
|
|
std::list<std::string> configurations;
|
2015-05-25 21:15:55 +02:00
|
|
|
std::string filedata;
|
2010-05-21 21:06:11 +02:00
|
|
|
preprocessor.preprocess(fin, filedata, configurations, filename, _settings._includePaths);
|
2012-01-06 08:01:50 +01:00
|
|
|
const std::string code = preprocessor.getcode(filedata, "", filename);
|
2010-05-21 21:06:11 +02:00
|
|
|
|
2011-10-13 20:53:06 +02:00
|
|
|
if (_settings.checkConfiguration) {
|
2011-05-02 14:58:16 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-05-21 21:06:11 +02:00
|
|
|
// Tokenize..
|
|
|
|
Tokenizer tokenizer(&_settings, this);
|
|
|
|
std::istringstream istr(code);
|
2014-10-03 10:02:46 +02:00
|
|
|
tokenizer.tokenize(istr, filename.c_str());
|
2013-12-30 17:45:28 +01:00
|
|
|
tokenizer.simplifyTokenList2();
|
2010-05-21 21:06:11 +02:00
|
|
|
}
|
2010-04-13 19:25:08 +02:00
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
// CppCheck - A function that checks a specified file
|
|
|
|
//---------------------------------------------------------------------------
|
2015-05-25 21:15:55 +02:00
|
|
|
bool CppCheck::checkFile(const std::string &code, const char FileName[], std::set<unsigned long long>& checksums, bool& internalErrorFound)
|
2010-04-13 19:25:08 +02:00
|
|
|
{
|
2015-05-25 21:15:55 +02:00
|
|
|
internalErrorFound=false;
|
2011-05-02 21:28:33 +02:00
|
|
|
if (_settings.terminated() || _settings.checkConfiguration)
|
2014-09-02 18:05:02 +02:00
|
|
|
return true;
|
2010-04-13 19:25:08 +02:00
|
|
|
|
|
|
|
Tokenizer _tokenizer(&_settings, this);
|
2012-04-10 13:45:34 +02:00
|
|
|
if (_settings._showtime != SHOWTIME_NONE)
|
|
|
|
_tokenizer.setTimerResults(&S_timerResults);
|
2011-11-20 23:41:26 +01:00
|
|
|
try {
|
2013-06-09 15:50:43 +02:00
|
|
|
// Execute rules for "raw" code
|
|
|
|
for (std::list<Settings::Rule>::const_iterator it = _settings.rules.begin(); it != _settings.rules.end(); ++it) {
|
|
|
|
if (it->tokenlist == "raw") {
|
|
|
|
Tokenizer tokenizer2(&_settings, this);
|
|
|
|
std::istringstream istr(code);
|
|
|
|
tokenizer2.list.createTokens(istr, FileName);
|
|
|
|
executeRules("raw", tokenizer2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-20 23:41:26 +01:00
|
|
|
// Tokenize the file
|
|
|
|
std::istringstream istr(code);
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2011-11-20 23:41:26 +01:00
|
|
|
Timer timer("Tokenizer::tokenize", _settings._showtime, &S_timerResults);
|
2014-10-03 10:02:46 +02:00
|
|
|
bool result = _tokenizer.tokenize(istr, FileName, cfg);
|
2011-11-20 23:41:26 +01:00
|
|
|
timer.Stop();
|
2014-09-02 18:05:02 +02:00
|
|
|
|
2014-09-05 11:05:21 +02:00
|
|
|
if (_settings._force || _settings._maxConfigs > 1) {
|
2015-05-25 21:15:55 +02:00
|
|
|
const unsigned long long checksum = _tokenizer.list.calculateChecksum();
|
2014-09-05 11:05:21 +02:00
|
|
|
if (checksums.find(checksum) != checksums.end())
|
|
|
|
return false;
|
|
|
|
checksums.insert(checksum);
|
|
|
|
}
|
2014-09-02 18:05:02 +02:00
|
|
|
|
2011-11-20 23:41:26 +01:00
|
|
|
if (!result) {
|
|
|
|
// File had syntax errors, abort
|
2014-09-02 18:05:02 +02:00
|
|
|
return true;
|
2011-11-20 23:41:26 +01:00
|
|
|
}
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2014-07-14 15:51:45 +02:00
|
|
|
// dump
|
|
|
|
if (_settings.dump) {
|
|
|
|
std::string dumpfile = std::string(FileName) + ".dump";
|
|
|
|
std::ofstream fdump(dumpfile.c_str());
|
|
|
|
if (fdump.is_open()) {
|
|
|
|
fdump << "<?xml version=\"1.0\"?>" << std::endl;
|
|
|
|
fdump << "<dump cfg=\"" << cfg << "\">" << std::endl;
|
|
|
|
_tokenizer.dump(fdump);
|
|
|
|
fdump << "</dump>" << std::endl;
|
|
|
|
}
|
2014-09-02 18:05:02 +02:00
|
|
|
return true;
|
2014-07-14 15:51:45 +02:00
|
|
|
}
|
|
|
|
|
2011-11-20 23:41:26 +01:00
|
|
|
// call all "runChecks" in all registered Check classes
|
2014-04-13 11:51:29 +02:00
|
|
|
for (std::list<Check *>::const_iterator it = Check::instances().begin(); it != Check::instances().end(); ++it) {
|
2011-11-20 23:41:26 +01:00
|
|
|
if (_settings.terminated())
|
2014-09-02 18:05:02 +02:00
|
|
|
return true;
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2011-11-20 23:41:26 +01:00
|
|
|
Timer timerRunChecks((*it)->name() + "::runChecks", _settings._showtime, &S_timerResults);
|
|
|
|
(*it)->runChecks(&_tokenizer, &_settings, this);
|
|
|
|
}
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2014-12-02 06:41:18 +01:00
|
|
|
// Analyse the tokens..
|
|
|
|
for (std::list<Check *>::const_iterator it = Check::instances().begin(); it != Check::instances().end(); ++it) {
|
|
|
|
Check::FileInfo *fi = (*it)->getFileInfo(&_tokenizer, &_settings);
|
|
|
|
if (fi != nullptr)
|
|
|
|
fileInfo.push_back(fi);
|
|
|
|
}
|
2014-11-24 06:37:08 +01:00
|
|
|
|
2013-06-09 14:58:56 +02:00
|
|
|
executeRules("normal", _tokenizer);
|
|
|
|
|
2013-05-11 11:35:04 +02:00
|
|
|
if (!_simplify)
|
2014-09-02 18:05:02 +02:00
|
|
|
return true;
|
2013-05-11 11:35:04 +02:00
|
|
|
|
2014-06-04 17:54:14 +02:00
|
|
|
Timer timer3("Tokenizer::simplifyTokenList2", _settings._showtime, &S_timerResults);
|
2013-12-30 17:45:28 +01:00
|
|
|
result = _tokenizer.simplifyTokenList2();
|
2011-11-20 23:41:26 +01:00
|
|
|
timer3.Stop();
|
|
|
|
if (!result)
|
2014-09-02 18:05:02 +02:00
|
|
|
return true;
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2011-11-20 23:41:26 +01:00
|
|
|
// call all "runSimplifiedChecks" in all registered Check classes
|
2014-04-13 11:51:29 +02:00
|
|
|
for (std::list<Check *>::const_iterator it = Check::instances().begin(); it != Check::instances().end(); ++it) {
|
2011-11-20 23:41:26 +01:00
|
|
|
if (_settings.terminated())
|
2014-09-02 18:05:02 +02:00
|
|
|
return true;
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2011-11-20 23:41:26 +01:00
|
|
|
Timer timerSimpleChecks((*it)->name() + "::runSimplifiedChecks", _settings._showtime, &S_timerResults);
|
|
|
|
(*it)->runSimplifiedChecks(&_tokenizer, &_settings, this);
|
|
|
|
}
|
2010-12-12 11:56:22 +01:00
|
|
|
|
2013-07-24 13:06:59 +02:00
|
|
|
if (_settings.terminated())
|
2014-09-02 18:05:02 +02:00
|
|
|
return true;
|
2013-07-24 13:06:59 +02:00
|
|
|
|
2013-06-09 14:58:56 +02:00
|
|
|
executeRules("simple", _tokenizer);
|
2013-07-24 13:06:59 +02:00
|
|
|
|
|
|
|
if (_settings.terminated())
|
2014-09-02 18:05:02 +02:00
|
|
|
return true;
|
2012-01-09 17:30:36 +01:00
|
|
|
} catch (const InternalError &e) {
|
2015-05-25 21:15:55 +02:00
|
|
|
internalErrorFound=true;
|
2011-11-20 23:41:26 +01:00
|
|
|
std::list<ErrorLogger::ErrorMessage::FileLocation> locationList;
|
|
|
|
ErrorLogger::ErrorMessage::FileLocation loc;
|
2012-01-08 21:19:44 +01:00
|
|
|
if (e.token) {
|
|
|
|
loc.line = e.token->linenr();
|
2012-05-05 18:33:26 +02:00
|
|
|
const std::string fixedpath = Path::toNativeSeparators(_tokenizer.list.file(e.token));
|
2012-01-08 21:19:44 +01:00
|
|
|
loc.setfile(fixedpath);
|
|
|
|
} else {
|
2014-03-27 13:15:21 +01:00
|
|
|
ErrorLogger::ErrorMessage::FileLocation loc2;
|
|
|
|
loc2.setfile(Path::toNativeSeparators(FileName));
|
|
|
|
locationList.push_back(loc2);
|
2014-06-04 18:00:22 +02:00
|
|
|
loc.setfile(_tokenizer.list.getSourceFilePath());
|
2012-01-08 21:19:44 +01:00
|
|
|
}
|
2011-11-20 23:41:26 +01:00
|
|
|
locationList.push_back(loc);
|
|
|
|
const ErrorLogger::ErrorMessage errmsg(locationList,
|
|
|
|
Severity::error,
|
2012-01-08 21:19:44 +01:00
|
|
|
e.errorMessage,
|
2014-03-27 13:15:21 +01:00
|
|
|
e.id,
|
2011-11-20 23:41:26 +01:00
|
|
|
false);
|
|
|
|
|
|
|
|
_errorLogger.reportErr(errmsg);
|
|
|
|
}
|
2014-09-02 18:05:02 +02:00
|
|
|
return true;
|
2010-04-13 19:25:08 +02:00
|
|
|
}
|
|
|
|
|
2013-06-09 14:58:56 +02:00
|
|
|
void CppCheck::executeRules(const std::string &tokenlist, const Tokenizer &tokenizer)
|
|
|
|
{
|
2013-06-12 06:43:52 +02:00
|
|
|
(void)tokenlist;
|
|
|
|
(void)tokenizer;
|
2013-06-09 20:17:26 +02:00
|
|
|
|
|
|
|
#ifdef HAVE_RULES
|
2013-06-09 14:58:56 +02:00
|
|
|
// Are there rules to execute?
|
|
|
|
bool isrule = false;
|
|
|
|
for (std::list<Settings::Rule>::const_iterator it = _settings.rules.begin(); it != _settings.rules.end(); ++it) {
|
|
|
|
if (it->tokenlist == tokenlist)
|
|
|
|
isrule = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// There is no rule to execute
|
|
|
|
if (isrule == false)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Write all tokens in a string that can be parsed by pcre
|
|
|
|
std::ostringstream ostr;
|
|
|
|
for (const Token *tok = tokenizer.tokens(); tok; tok = tok->next())
|
|
|
|
ostr << " " << tok->str();
|
|
|
|
const std::string str(ostr.str());
|
|
|
|
|
|
|
|
for (std::list<Settings::Rule>::const_iterator it = _settings.rules.begin(); it != _settings.rules.end(); ++it) {
|
|
|
|
const Settings::Rule &rule = *it;
|
|
|
|
if (rule.pattern.empty() || rule.id.empty() || rule.severity.empty() || rule.tokenlist != tokenlist)
|
|
|
|
continue;
|
|
|
|
|
2014-02-16 11:47:52 +01:00
|
|
|
const char *error = nullptr;
|
2013-06-09 14:58:56 +02:00
|
|
|
int erroffset = 0;
|
2014-02-15 08:46:28 +01:00
|
|
|
pcre *re = pcre_compile(rule.pattern.c_str(),0,&error,&erroffset,nullptr);
|
2013-06-09 14:58:56 +02:00
|
|
|
if (!re) {
|
|
|
|
if (error) {
|
|
|
|
ErrorLogger::ErrorMessage errmsg(std::list<ErrorLogger::ErrorMessage::FileLocation>(),
|
|
|
|
Severity::error,
|
|
|
|
error,
|
|
|
|
"pcre_compile",
|
|
|
|
false);
|
|
|
|
|
|
|
|
reportErr(errmsg);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
int pos = 0;
|
2015-05-25 21:15:55 +02:00
|
|
|
int ovector[30]= {0};
|
2014-02-15 08:46:28 +01:00
|
|
|
while (pos < (int)str.size() && 0 <= pcre_exec(re, nullptr, str.c_str(), (int)str.size(), pos, 0, ovector, 30)) {
|
2015-05-25 21:15:55 +02:00
|
|
|
const unsigned int pos1 = (unsigned int)ovector[0];
|
|
|
|
const unsigned int pos2 = (unsigned int)ovector[1];
|
2013-06-09 14:58:56 +02:00
|
|
|
|
|
|
|
// jump to the end of the match for the next pcre_exec
|
|
|
|
pos = (int)pos2;
|
|
|
|
|
|
|
|
// determine location..
|
|
|
|
ErrorLogger::ErrorMessage::FileLocation loc;
|
2014-06-04 19:18:27 +02:00
|
|
|
loc.setfile(tokenizer.list.getSourceFilePath());
|
2013-06-09 14:58:56 +02:00
|
|
|
loc.line = 0;
|
|
|
|
|
|
|
|
std::size_t len = 0;
|
|
|
|
for (const Token *tok = tokenizer.tokens(); tok; tok = tok->next()) {
|
|
|
|
len = len + 1U + tok->str().size();
|
|
|
|
if (len > pos1) {
|
|
|
|
loc.setfile(tokenizer.list.getFiles().at(tok->fileIndex()));
|
|
|
|
loc.line = tok->linenr();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::list<ErrorLogger::ErrorMessage::FileLocation> callStack(1, loc);
|
|
|
|
|
|
|
|
// Create error message
|
|
|
|
std::string summary;
|
|
|
|
if (rule.summary.empty())
|
|
|
|
summary = "found '" + str.substr(pos1, pos2 - pos1) + "'";
|
|
|
|
else
|
|
|
|
summary = rule.summary;
|
|
|
|
const ErrorLogger::ErrorMessage errmsg(callStack, Severity::fromString(rule.severity), summary, rule.id, false);
|
|
|
|
|
|
|
|
// Report error
|
|
|
|
reportErr(errmsg);
|
|
|
|
}
|
|
|
|
|
|
|
|
pcre_free(re);
|
|
|
|
}
|
2013-06-09 20:17:26 +02:00
|
|
|
#endif
|
2013-06-09 14:58:56 +02:00
|
|
|
}
|
|
|
|
|
2011-02-16 02:12:15 +01:00
|
|
|
Settings &CppCheck::settings()
|
2010-04-13 19:25:08 +02:00
|
|
|
{
|
|
|
|
return _settings;
|
|
|
|
}
|
|
|
|
|
2012-12-26 18:35:49 +01:00
|
|
|
void CppCheck::tooManyConfigsError(const std::string &file, const std::size_t numberOfConfigurations)
|
|
|
|
{
|
|
|
|
if (!_settings.isEnabled("information") && !tooManyConfigs)
|
|
|
|
return;
|
|
|
|
|
|
|
|
tooManyConfigs = false;
|
|
|
|
|
|
|
|
if (_settings.isEnabled("information") && file.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
std::list<ErrorLogger::ErrorMessage::FileLocation> loclist;
|
|
|
|
if (!file.empty()) {
|
|
|
|
ErrorLogger::ErrorMessage::FileLocation location;
|
|
|
|
location.setfile(file);
|
|
|
|
loclist.push_back(location);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostringstream msg;
|
|
|
|
msg << "Too many #ifdef configurations - cppcheck only checks " << _settings._maxConfigs;
|
|
|
|
if (numberOfConfigurations > _settings._maxConfigs)
|
|
|
|
msg << " of " << numberOfConfigurations << " configurations. Use --force to check all configurations.\n";
|
|
|
|
if (file.empty())
|
|
|
|
msg << " configurations. Use --force to check all configurations. For more details, use --enable=information.\n";
|
|
|
|
msg << "The checking of the file will be interrupted because there are too many "
|
|
|
|
"#ifdef configurations. Checking of all #ifdef configurations can be forced "
|
|
|
|
"by --force command line option or from GUI preferences. However that may "
|
|
|
|
"increase the checking time.";
|
|
|
|
if (file.empty())
|
|
|
|
msg << " For more details, use --enable=information.";
|
|
|
|
|
|
|
|
|
|
|
|
ErrorLogger::ErrorMessage errmsg(loclist,
|
|
|
|
Severity::information,
|
|
|
|
msg.str(),
|
|
|
|
"toomanyconfigs",
|
|
|
|
false);
|
|
|
|
|
|
|
|
reportErr(errmsg);
|
|
|
|
}
|
|
|
|
|
2014-09-02 18:05:02 +02:00
|
|
|
void CppCheck::purgedConfigurationMessage(const std::string &file, const std::string& configuration)
|
|
|
|
{
|
|
|
|
|
|
|
|
tooManyConfigs = false;
|
|
|
|
|
|
|
|
if (_settings.isEnabled("information") && file.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
std::list<ErrorLogger::ErrorMessage::FileLocation> loclist;
|
|
|
|
if (!file.empty()) {
|
|
|
|
ErrorLogger::ErrorMessage::FileLocation location;
|
|
|
|
location.setfile(file);
|
|
|
|
loclist.push_back(location);
|
|
|
|
}
|
|
|
|
|
|
|
|
ErrorLogger::ErrorMessage errmsg(loclist,
|
|
|
|
Severity::information,
|
|
|
|
"The configuration '" + configuration + "' was not checked because its code equals another one.",
|
2014-09-10 20:19:04 +02:00
|
|
|
"purgedConfiguration",
|
2014-09-02 18:05:02 +02:00
|
|
|
false);
|
|
|
|
|
|
|
|
reportErr(errmsg);
|
|
|
|
}
|
|
|
|
|
2010-04-13 19:25:08 +02:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void CppCheck::reportErr(const ErrorLogger::ErrorMessage &msg)
|
|
|
|
{
|
2013-10-20 14:09:10 +02:00
|
|
|
if (!_settings.library.reportErrors(msg.file0))
|
|
|
|
return;
|
|
|
|
|
2015-05-25 21:15:55 +02:00
|
|
|
const std::string errmsg = msg.toString(_settings._verbose);
|
2011-04-10 21:51:27 +02:00
|
|
|
if (errmsg.empty())
|
|
|
|
return;
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2012-04-06 18:16:59 +02:00
|
|
|
// Alert only about unique errors
|
|
|
|
if (std::find(_errorList.begin(), _errorList.end(), errmsg) != _errorList.end())
|
|
|
|
return;
|
|
|
|
|
2011-10-13 20:53:06 +02:00
|
|
|
if (_settings.debugFalsePositive) {
|
2011-08-16 21:58:27 +02:00
|
|
|
// Don't print out error
|
|
|
|
_errorList.push_back(errmsg);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-04-13 19:25:08 +02:00
|
|
|
std::string file;
|
|
|
|
unsigned int line(0);
|
2011-10-13 20:53:06 +02:00
|
|
|
if (!msg._callStack.empty()) {
|
2010-12-15 21:39:46 +01:00
|
|
|
file = msg._callStack.back().getfile(false);
|
2010-04-13 19:25:08 +02:00
|
|
|
line = msg._callStack.back().line;
|
|
|
|
}
|
|
|
|
|
2011-10-13 20:53:06 +02:00
|
|
|
if (_useGlobalSuppressions) {
|
2011-02-16 02:12:15 +01:00
|
|
|
if (_settings.nomsg.isSuppressed(msg._id, file, line))
|
|
|
|
return;
|
2011-10-13 20:53:06 +02:00
|
|
|
} else {
|
2011-02-16 02:12:15 +01:00
|
|
|
if (_settings.nomsg.isSuppressedLocal(msg._id, file, line))
|
|
|
|
return;
|
|
|
|
}
|
2010-04-13 19:25:08 +02:00
|
|
|
|
|
|
|
if (!_settings.nofail.isSuppressed(msg._id, file, line))
|
|
|
|
exitcode = 1;
|
|
|
|
|
|
|
|
_errorList.push_back(errmsg);
|
2012-04-06 18:16:59 +02:00
|
|
|
|
2010-04-13 19:25:08 +02:00
|
|
|
_errorLogger.reportErr(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CppCheck::reportOut(const std::string &outmsg)
|
|
|
|
{
|
|
|
|
_errorLogger.reportOut(outmsg);
|
|
|
|
}
|
|
|
|
|
2012-07-08 23:39:46 +02:00
|
|
|
void CppCheck::reportProgress(const std::string &filename, const char stage[], const std::size_t value)
|
2010-04-13 19:25:08 +02:00
|
|
|
{
|
2010-08-08 08:45:37 +02:00
|
|
|
_errorLogger.reportProgress(filename, stage, value);
|
2010-04-13 19:25:08 +02:00
|
|
|
}
|
|
|
|
|
2012-06-18 23:15:48 +02:00
|
|
|
void CppCheck::reportInfo(const ErrorLogger::ErrorMessage &msg)
|
|
|
|
{
|
2012-07-10 20:29:04 +02:00
|
|
|
// Suppressing info message?
|
|
|
|
std::string file;
|
|
|
|
unsigned int line(0);
|
|
|
|
if (!msg._callStack.empty()) {
|
|
|
|
file = msg._callStack.back().getfile(false);
|
|
|
|
line = msg._callStack.back().line;
|
|
|
|
}
|
|
|
|
if (_useGlobalSuppressions) {
|
|
|
|
if (_settings.nomsg.isSuppressed(msg._id, file, line))
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
if (_settings.nomsg.isSuppressedLocal(msg._id, file, line))
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-06-18 23:15:48 +02:00
|
|
|
_errorLogger.reportInfo(msg);
|
|
|
|
}
|
|
|
|
|
2012-07-08 23:39:46 +02:00
|
|
|
void CppCheck::reportStatus(unsigned int /*fileindex*/, unsigned int /*filecount*/, std::size_t /*sizedone*/, std::size_t /*sizetotal*/)
|
2010-08-03 16:36:21 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-04-13 19:25:08 +02:00
|
|
|
void CppCheck::getErrorMessages()
|
|
|
|
{
|
2015-01-06 15:08:25 +01:00
|
|
|
Settings s(_settings);
|
|
|
|
s.addEnabled("warning");
|
|
|
|
s.addEnabled("style");
|
|
|
|
s.addEnabled("portability");
|
|
|
|
s.addEnabled("performance");
|
|
|
|
s.addEnabled("information");
|
|
|
|
|
2012-12-26 18:35:49 +01:00
|
|
|
tooManyConfigs = true;
|
|
|
|
tooManyConfigsError("",0U);
|
|
|
|
|
2010-04-13 19:25:08 +02:00
|
|
|
// call all "getErrorMessages" in all registered Check classes
|
2014-04-13 11:51:29 +02:00
|
|
|
for (std::list<Check *>::const_iterator it = Check::instances().begin(); it != Check::instances().end(); ++it)
|
2015-01-06 15:08:25 +01:00
|
|
|
(*it)->getErrorMessages(this, &s);
|
2010-04-13 19:25:08 +02:00
|
|
|
|
2015-01-06 15:08:25 +01:00
|
|
|
Tokenizer::getErrorMessages(this, &s);
|
|
|
|
Preprocessor::getErrorMessages(this, &s);
|
2010-04-13 19:25:08 +02:00
|
|
|
}
|
2014-11-15 10:43:49 +01:00
|
|
|
|
|
|
|
void CppCheck::analyseWholeProgram()
|
|
|
|
{
|
2015-01-07 19:26:16 +01:00
|
|
|
// Analyse the tokens
|
2014-11-15 10:43:49 +01:00
|
|
|
for (std::list<Check *>::const_iterator it = Check::instances().begin(); it != Check::instances().end(); ++it)
|
2015-06-28 17:54:48 +02:00
|
|
|
(*it)->analyseWholeProgram(fileInfo, _settings, *this);
|
2014-11-15 10:43:49 +01:00
|
|
|
}
|
|
|
|
|
2015-01-07 19:26:16 +01:00
|
|
|
bool CppCheck::unusedFunctionCheckIsEnabled() const
|
|
|
|
{
|
|
|
|
return (_settings._jobs == 1 && _settings.isEnabled("unusedFunction"));
|
|
|
|
}
|