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/>.
*/
# include "tokenize.h"
# include "checknonreentrantfunctions.h"
# include "testsuite.h"
# include <sstream>
extern std : : ostringstream errout ;
2011-10-13 20:53:06 +02:00
class TestNonReentrantFunctions : public TestFixture {
2011-07-28 23:29:16 +02:00
public :
2014-11-20 14:20:09 +01:00
TestNonReentrantFunctions ( ) : TestFixture ( " TestNonReentrantFunctions " ) {
2013-08-07 16:27:37 +02:00
}
2011-07-28 23:29:16 +02:00
private :
2014-11-20 14:20:09 +01:00
void run ( ) {
2011-07-28 23:29:16 +02:00
TEST_CASE ( test_crypt ) ;
2011-10-22 11:54:52 +02:00
TEST_CASE ( test_namespace_handling ) ;
2011-07-28 23:29:16 +02:00
}
2014-11-20 14:20:09 +01:00
void check ( const char code [ ] ) {
2011-07-28 23:29:16 +02:00
// Clear the error buffer..
errout . str ( " " ) ;
Settings settings ;
2011-10-22 09:45:48 +02:00
settings . standards . posix = true ;
2011-09-03 15:30:30 +02:00
settings . addEnabled ( " portability " ) ;
2011-07-28 23:29:16 +02:00
// Tokenize..
Tokenizer tokenizer ( & settings , this ) ;
std : : istringstream istr ( code ) ;
tokenizer . tokenize ( istr , " test.cpp " ) ;
2013-12-30 17:45:28 +01:00
tokenizer . simplifyTokenList2 ( ) ;
2011-07-28 23:29:16 +02:00
// Check for non reentrant functions..
CheckNonReentrantFunctions checkNonReentrantFunctions ( & tokenizer , & settings , this ) ;
checkNonReentrantFunctions . nonReentrantFunctions ( ) ;
}
2014-11-20 14:20:09 +01:00
void test_crypt ( ) {
2011-07-28 23:29:16 +02:00
check ( " void f(char *pwd) \n "
" { \n "
" char *cpwd; "
" crypt(pwd, cpwd); \n "
2013-03-20 15:36:16 +01:00
" } " ) ;
2012-07-29 11:30:03 +02:00
ASSERT_EQUALS ( " [test.cpp:3]: (portability) Non reentrant function 'crypt' called. For threadsafe applications it is recommended to use the reentrant replacement function 'crypt_r'. \n " , errout . str ( ) ) ;
2011-07-28 23:29:16 +02:00
check ( " void f() \n "
" { \n "
" char *pwd = getpass( \" Password: \" ); "
" char *cpwd; "
" crypt(pwd, cpwd); \n "
2013-03-20 15:36:16 +01:00
" } " ) ;
2012-07-29 11:30:03 +02:00
ASSERT_EQUALS ( " [test.cpp:3]: (portability) Non reentrant function 'crypt' called. For threadsafe applications it is recommended to use the reentrant replacement function 'crypt_r'. \n " , errout . str ( ) ) ;
2011-07-28 23:29:16 +02:00
check ( " int f() \n "
" { \n "
" int crypt = 0; "
" return crypt; \n "
2013-03-20 15:36:16 +01:00
" } " ) ;
2011-07-28 23:29:16 +02:00
ASSERT_EQUALS ( " " , errout . str ( ) ) ;
}
2014-11-20 14:20:09 +01:00
void test_namespace_handling ( ) {
2011-10-22 11:54:52 +02:00
check ( " int f() \n "
" { \n "
" time_t t = 0; "
" std::localtime(&t); \n "
2013-03-20 15:36:16 +01:00
" } " ) ;
2012-07-29 11:30:03 +02:00
ASSERT_EQUALS ( " [test.cpp:3]: (portability) Non reentrant function 'localtime' called. For threadsafe applications it is recommended to use the reentrant replacement function 'localtime_r'. \n " , errout . str ( ) ) ;
2011-07-28 23:29:16 +02:00
2011-10-22 11:54:52 +02:00
// Passed as function argument
check ( " int f() \n "
" { \n "
2012-01-12 22:28:12 +01:00
" printf( \" Magic guess: %d \n \" , getpwent()); \n "
2013-03-20 15:36:16 +01:00
" } " ) ;
2012-07-29 11:30:03 +02:00
ASSERT_EQUALS ( " [test.cpp:3]: (portability) Non reentrant function 'getpwent' called. For threadsafe applications it is recommended to use the reentrant replacement function 'getpwent_r'. \n " , errout . str ( ) ) ;
2011-10-22 11:54:52 +02:00
// Pass return value
check ( " int f() \n "
" { \n "
" time_t t = 0; "
" struct tm *foo = localtime(&t); \n "
2013-03-20 15:36:16 +01:00
" } " ) ;
2012-07-29 11:30:03 +02:00
ASSERT_EQUALS ( " [test.cpp:3]: (portability) Non reentrant function 'localtime' called. For threadsafe applications it is recommended to use the reentrant replacement function 'localtime_r'. \n " , errout . str ( ) ) ;
2011-10-22 11:54:52 +02:00
// Access via global namespace
check ( " int f() \n "
" { \n "
2012-01-12 22:28:12 +01:00
" ::getpwent(); \n "
2013-03-20 15:36:16 +01:00
" } " ) ;
2012-07-29 11:30:03 +02:00
ASSERT_EQUALS ( " [test.cpp:3]: (portability) Non reentrant function 'getpwent' called. For threadsafe applications it is recommended to use the reentrant replacement function 'getpwent_r'. \n " , errout . str ( ) ) ;
2011-07-28 23:29:16 +02:00
2011-10-22 11:54:52 +02:00
// Be quiet on function definitions
2012-01-12 22:28:12 +01:00
check ( " int getpwent() \n "
2011-10-22 11:54:52 +02:00
" { \n "
" return 123; \n "
2013-03-20 15:36:16 +01:00
" } " ) ;
2011-10-22 11:54:52 +02:00
ASSERT_EQUALS ( " " , errout . str ( ) ) ;
// Be quiet on other namespaces
check ( " int f() \n "
" { \n "
2012-01-12 22:28:12 +01:00
" foobar::getpwent(); \n "
2013-03-20 15:36:16 +01:00
" } " ) ;
2011-10-22 11:54:52 +02:00
ASSERT_EQUALS ( " " , errout . str ( ) ) ;
2011-07-28 23:29:16 +02:00
2011-10-22 11:54:52 +02:00
// Be quiet on class member functions
check ( " int f() \n "
" { \n "
2012-01-12 22:28:12 +01:00
" foobar.getpwent(); \n "
2013-03-20 15:36:16 +01:00
" } " ) ;
2011-10-22 11:54:52 +02:00
ASSERT_EQUALS ( " " , errout . str ( ) ) ;
}
} ;
REGISTER_TEST ( TestNonReentrantFunctions )