2011-07-28 23:29:16 +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.
|
2011-07-28 23:29:16 +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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
// Find non reentrant functions
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#include "checknonreentrantfunctions.h"
|
2014-10-31 11:40:42 +01:00
|
|
|
#include "symboldatabase.h"
|
2011-07-28 23:29:16 +02:00
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
// Register this check class (by creating a static instance of it)
|
2011-10-13 20:53:06 +02:00
|
|
|
namespace {
|
|
|
|
CheckNonReentrantFunctions instance;
|
2011-07-28 23:29:16 +02:00
|
|
|
}
|
|
|
|
|
2015-06-29 21:17:15 +02:00
|
|
|
namespace {
|
|
|
|
const std::set<std::string> _nonReentrantFunctions = make_container< std::set<std::string> > ()
|
2015-07-01 07:50:13 +02:00
|
|
|
<< "localtime" << "gmtime" << "strtok" << "gethostbyname" << "gethostbyaddr" << "getservbyname"
|
|
|
|
<< "getservbyport" << "crypt" << "ttyname" << "gethostbyname2"
|
|
|
|
<< "getprotobyname" << "getnetbyname" << "getnetbyaddr" << "getrpcbyname" << "getrpcbynumber" << "getrpcent"
|
|
|
|
<< "ctermid" << "readdir" << "getlogin" << "getpwent" << "getpwnam" << "getpwuid" << "getspent"
|
|
|
|
<< "fgetspent" << "getspnam" << "getgrnam" << "getgrgid" << "getnetgrent" << "tempnam" << "fgetpwent"
|
|
|
|
<< "fgetgrent" << "ecvt" << "gcvt" << "getservent" << "gethostent" << "getgrent" << "fcvt" ;
|
2015-06-29 21:17:15 +02:00
|
|
|
}
|
|
|
|
|
2015-07-01 07:50:13 +02:00
|
|
|
std::string CheckNonReentrantFunctions::generateErrorMessage(const std::string& function)
|
|
|
|
{
|
2015-06-29 21:17:15 +02:00
|
|
|
return std::string("Non reentrant function '") + function + "' called. " +
|
2015-07-01 07:50:13 +02:00
|
|
|
"For threadsafe applications it is recommended to use the reentrant replacement function '" + function + "_r'.";
|
2015-06-29 21:17:15 +02:00
|
|
|
}
|
|
|
|
|
2011-07-28 23:29:16 +02:00
|
|
|
void CheckNonReentrantFunctions::nonReentrantFunctions()
|
|
|
|
{
|
2011-10-22 09:45:48 +02:00
|
|
|
if (!_settings->standards.posix || !_settings->isEnabled("portability"))
|
2011-07-28 23:29:16 +02:00
|
|
|
return;
|
|
|
|
|
2014-10-31 11:40:42 +01:00
|
|
|
const SymbolDatabase * const symbolDatabase = _tokenizer->getSymbolDatabase();
|
|
|
|
const std::size_t functions = symbolDatabase->functionScopes.size();
|
|
|
|
for (std::size_t i = 0; i < functions; ++i) {
|
|
|
|
const Scope * scope = symbolDatabase->functionScopes[i];
|
|
|
|
for (const Token *tok = scope->classStart->next(); tok != scope->classEnd; tok = tok->next()) {
|
|
|
|
// Look for function invocations
|
|
|
|
if (tok->varId() != 0 || !tok->isName() || tok->strAt(1) != "(")
|
2011-10-22 11:54:52 +02:00
|
|
|
continue;
|
|
|
|
|
2014-10-31 11:40:42 +01:00
|
|
|
// Check for non-reentrant function name
|
2015-06-29 21:17:15 +02:00
|
|
|
std::set<std::string>::const_iterator it = _nonReentrantFunctions.find(tok->str());
|
2014-10-31 11:40:42 +01:00
|
|
|
if (it == _nonReentrantFunctions.end())
|
2011-10-22 11:54:52 +02:00
|
|
|
continue;
|
|
|
|
|
2014-10-31 11:40:42 +01:00
|
|
|
const Token *prev = tok->previous();
|
|
|
|
if (prev) {
|
|
|
|
// Ignore function definitions, class members or class definitions
|
|
|
|
if (prev->str() == ".")
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Check for "std" or global namespace, ignore other namespaces
|
2015-06-28 17:54:48 +02:00
|
|
|
if (_tokenizer->isCPP() && prev->str() == "::" && prev->previous() && prev->previous()->str() != "std" && prev->previous()->isName())
|
2014-10-31 11:40:42 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only affecting multi threaded code, therefore this is "portability"
|
2015-06-29 21:17:15 +02:00
|
|
|
reportError(tok, Severity::portability, "nonreentrantFunctions" + *it, generateErrorMessage(*it));
|
2014-10-31 11:40:42 +01:00
|
|
|
}
|
2011-07-28 23:29:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
2015-06-29 21:17:15 +02:00
|
|
|
|
2015-07-01 07:50:13 +02:00
|
|
|
void CheckNonReentrantFunctions::getErrorMessages(ErrorLogger *errorLogger, const Settings *settings) const
|
|
|
|
{
|
2015-06-29 21:17:15 +02:00
|
|
|
CheckNonReentrantFunctions c(0, settings, errorLogger);
|
|
|
|
|
|
|
|
std::set<std::string>::const_iterator it(_nonReentrantFunctions.begin()), itend(_nonReentrantFunctions.end());
|
|
|
|
for (; it!=itend; ++it) {
|
|
|
|
c.reportError(0, Severity::portability, "nonreentrantFunctions"+*it, generateErrorMessage(*it));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-01 07:50:13 +02:00
|
|
|
std::string CheckNonReentrantFunctions::classInfo() const
|
|
|
|
{
|
2015-06-29 21:17:15 +02:00
|
|
|
std::string info = "Warn if any of these non reentrant functions are used:\n";
|
|
|
|
std::set<std::string>::const_iterator it(_nonReentrantFunctions.begin()), itend(_nonReentrantFunctions.end());
|
|
|
|
for (; it!=itend; ++it) {
|
|
|
|
info += "- " + *it + "\n";
|
|
|
|
}
|
|
|
|
return info;
|
|
|
|
}
|