2011-10-23 13:07:43 +02:00
/*
* Cppcheck - A tool for static C / C + + code analysis
2023-01-28 10:16:34 +01:00
* Copyright ( C ) 2007 - 2023 Cppcheck team .
2011-10-23 13:07:43 +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/>.
*/
2017-05-27 04:33:47 +02:00
2011-10-23 13:07:43 +02:00
# include "checkboost.h"
2022-01-27 19:03:20 +01:00
# include "errortypes.h"
2017-05-27 04:33:47 +02:00
# include "settings.h"
2023-01-27 08:18:32 +01:00
# include "fixture.h"
2017-05-27 04:33:47 +02:00
# include "tokenize.h"
2011-10-23 13:07:43 +02:00
2022-09-16 07:15:49 +02:00
# include <sstream> // IWYU pragma: keep
2011-10-23 13:07:43 +02:00
class TestBoost : public TestFixture {
public :
2021-08-07 20:51:18 +02:00
TestBoost ( ) : TestFixture ( " TestBoost " ) { }
2011-10-23 13:07:43 +02:00
private :
2023-05-02 11:48:24 +02:00
const Settings settings = settingsBuilder ( ) . severity ( Severity : : style ) . severity ( Severity : : performance ) . build ( ) ;
2015-10-07 18:33:57 +02:00
2022-02-10 23:02:24 +01:00
void run ( ) override {
2020-04-21 17:27:51 +02:00
TEST_CASE ( BoostForeachContainerModification ) ;
2011-10-23 13:07:43 +02:00
}
2021-11-29 07:34:39 +01:00
# define check(code) check_(code, __FILE__, __LINE__)
void check_ ( const char code [ ] , const char * file , int line ) {
2011-10-23 13:07:43 +02:00
// Clear the error buffer..
errout . str ( " " ) ;
// Tokenize..
Tokenizer tokenizer ( & settings , this ) ;
2012-07-16 13:17:14 +02:00
std : : istringstream istr ( code ) ;
2021-11-29 07:34:39 +01:00
ASSERT_LOC ( tokenizer . tokenize ( istr , " test.cpp " ) , file , line ) ;
2011-10-23 13:07:43 +02:00
// Check..
2023-01-29 17:23:03 +01:00
runChecks < CheckBoost > ( & tokenizer , & settings , this ) ;
2011-10-23 13:07:43 +02:00
}
2014-11-20 14:20:09 +01:00
void BoostForeachContainerModification ( ) {
2011-10-23 13:07:43 +02:00
check ( " void f() { \n "
" vector<int> data; \n "
" BOOST_FOREACH(int i, data) { \n "
" data.push_back(123); \n "
" } \n "
" } " ) ;
2012-08-02 11:40:08 +02:00
ASSERT_EQUALS ( " [test.cpp:4]: (error) BOOST_FOREACH caches the end() iterator. It's undefined behavior if you modify the container inside. \n " , errout . str ( ) ) ;
2011-10-23 13:07:43 +02:00
check ( " void f() { \n "
" set<int> data; \n "
" BOOST_FOREACH(int i, data) { \n "
" data.insert(123); \n "
" } \n "
" } " ) ;
2012-08-02 11:40:08 +02:00
ASSERT_EQUALS ( " [test.cpp:4]: (error) BOOST_FOREACH caches the end() iterator. It's undefined behavior if you modify the container inside. \n " , errout . str ( ) ) ;
2011-10-23 13:07:43 +02:00
check ( " void f() { \n "
" set<int> data; \n "
" BOOST_FOREACH(const int &i, data) { \n "
" data.erase(123); \n "
" } \n "
" } " ) ;
2012-08-02 11:40:08 +02:00
ASSERT_EQUALS ( " [test.cpp:4]: (error) BOOST_FOREACH caches the end() iterator. It's undefined behavior if you modify the container inside. \n " , errout . str ( ) ) ;
2011-10-23 13:07:43 +02:00
// Check single line usage
check ( " void f() { \n "
" set<int> data; \n "
" BOOST_FOREACH(const int &i, data) \n "
" data.clear(); \n "
" } " ) ;
2012-08-02 11:40:08 +02:00
ASSERT_EQUALS ( " [test.cpp:4]: (error) BOOST_FOREACH caches the end() iterator. It's undefined behavior if you modify the container inside. \n " , errout . str ( ) ) ;
2011-10-23 13:07:43 +02:00
// Container returned as result of a function -> Be quiet
check ( " void f() { \n "
" BOOST_FOREACH(const int &i, get_data()) \n "
" data.insert(i); \n "
" } " ) ;
ASSERT_EQUALS ( " " , errout . str ( ) ) ;
2014-08-31 20:40:52 +02:00
// Break after modification (#4788)
check ( " void f() { \n "
" vector<int> data; \n "
" BOOST_FOREACH(int i, data) { \n "
" data.push_back(123); \n "
" break; \n "
" } \n "
" } " ) ;
ASSERT_EQUALS ( " " , errout . str ( ) ) ;
2011-10-23 13:07:43 +02:00
}
} ;
REGISTER_TEST ( TestBoost )