Blame test/testfunctions.cpp

Packit 2035a7
/*
Packit 2035a7
 * Cppcheck - A tool for static C/C++ code analysis
Packit 2035a7
 * Copyright (C) 2007-2017 Cppcheck team.
Packit 2035a7
 *
Packit 2035a7
 * This program is free software: you can redistribute it and/or modify
Packit 2035a7
 * it under the terms of the GNU General Public License as published by
Packit 2035a7
 * the Free Software Foundation, either version 3 of the License, or
Packit 2035a7
 * (at your option) any later version.
Packit 2035a7
 *
Packit 2035a7
 * This program is distributed in the hope that it will be useful,
Packit 2035a7
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 2035a7
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 2035a7
 * GNU General Public License for more details.
Packit 2035a7
 *
Packit 2035a7
 * You should have received a copy of the GNU General Public License
Packit 2035a7
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Packit 2035a7
 */
Packit 2035a7
Packit 2035a7
#include <tinyxml2.h>
Packit 2035a7
Packit 2035a7
#include "checkfunctions.h"
Packit 2035a7
#include "library.h"
Packit 2035a7
#include "settings.h"
Packit 2035a7
#include "standards.h"
Packit 2035a7
#include "testsuite.h"
Packit 2035a7
#include "tokenize.h"
Packit 2035a7
Packit 2035a7
Packit 2035a7
class TestFunctions : public TestFixture {
Packit 2035a7
public:
Packit 2035a7
    TestFunctions() : TestFixture("TestFunctions") {
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
private:
Packit 2035a7
    Settings settings;
Packit 2035a7
Packit 2035a7
    void run() {
Packit 2035a7
        settings.addEnabled("style");
Packit 2035a7
        settings.addEnabled("warning");
Packit 2035a7
        settings.addEnabled("portability");
Packit 2035a7
        settings.standards.posix = true;
Packit 2035a7
        settings.standards.c = Standards::C11;
Packit 2035a7
        settings.standards.cpp = Standards::CPP11;
Packit 2035a7
        LOAD_LIB_2(settings.library, "std.cfg");
Packit 2035a7
        LOAD_LIB_2(settings.library, "posix.cfg");
Packit 2035a7
Packit 2035a7
        // Prohibited functions
Packit 2035a7
        TEST_CASE(prohibitedFunctions_posix);
Packit 2035a7
        TEST_CASE(prohibitedFunctions_index);
Packit 2035a7
        TEST_CASE(prohibitedFunctions_qt_index); // FP when using the Qt function 'index'?
Packit 2035a7
        TEST_CASE(prohibitedFunctions_rindex);
Packit 2035a7
        TEST_CASE(prohibitedFunctions_var); // no false positives for variables
Packit 2035a7
        TEST_CASE(prohibitedFunctions_gets); // dangerous function
Packit 2035a7
        TEST_CASE(prohibitedFunctions_alloca);
Packit 2035a7
        TEST_CASE(prohibitedFunctions_declaredFunction); // declared function ticket #3121
Packit 2035a7
        TEST_CASE(prohibitedFunctions_std_gets); // test std::gets
Packit 2035a7
        TEST_CASE(prohibitedFunctions_multiple); // multiple use of obsolete functions
Packit 2035a7
        TEST_CASE(prohibitedFunctions_c_declaration); // c declared function
Packit 2035a7
        TEST_CASE(prohibitedFunctions_functionWithBody); // function with body
Packit 2035a7
        TEST_CASE(prohibitedFunctions_crypt); // Non-reentrant function
Packit 2035a7
        TEST_CASE(prohibitedFunctions_namespaceHandling);
Packit 2035a7
Packit 2035a7
        // Invalid function usage
Packit 2035a7
        TEST_CASE(invalidFunctionUsage1);
Packit 2035a7
Packit 2035a7
        // Math function usage
Packit 2035a7
        TEST_CASE(mathfunctionCall_fmod);
Packit 2035a7
        TEST_CASE(mathfunctionCall_sqrt);
Packit 2035a7
        TEST_CASE(mathfunctionCall_log);
Packit 2035a7
        TEST_CASE(mathfunctionCall_acos);
Packit 2035a7
        TEST_CASE(mathfunctionCall_asin);
Packit 2035a7
        TEST_CASE(mathfunctionCall_pow);
Packit 2035a7
        TEST_CASE(mathfunctionCall_atan2);
Packit 2035a7
        TEST_CASE(mathfunctionCall_precision);
Packit 2035a7
Packit 2035a7
        // Ignored return value
Packit 2035a7
        TEST_CASE(checkIgnoredReturnValue);
Packit 2035a7
Packit 2035a7
        // memset..
Packit 2035a7
        TEST_CASE(memsetZeroBytes);
Packit 2035a7
        TEST_CASE(memsetInvalid2ndParam);
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void check(const char code[], const char filename[]="test.cpp", const Settings* settings_=nullptr) {
Packit 2035a7
        // Clear the error buffer..
Packit 2035a7
        errout.str("");
Packit 2035a7
Packit 2035a7
        if (!settings_)
Packit 2035a7
            settings_ = &settings;
Packit 2035a7
Packit 2035a7
        // Tokenize..
Packit 2035a7
        Tokenizer tokenizer(settings_, this);
Packit 2035a7
        std::istringstream istr(code);
Packit 2035a7
        tokenizer.tokenize(istr, filename);
Packit 2035a7
Packit 2035a7
        CheckFunctions checkFunctions(&tokenizer, settings_, this);
Packit 2035a7
        checkFunctions.runChecks(&tokenizer, settings_, this);
Packit 2035a7
Packit 2035a7
        // Simplify...
Packit 2035a7
        tokenizer.simplifyTokenList2();
Packit 2035a7
Packit 2035a7
        // Check...
Packit 2035a7
        checkFunctions.runSimplifiedChecks(&tokenizer, settings_, this);
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_posix() {
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    bsd_signal(SIGABRT, SIG_IGN);\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (style) Obsolescent function 'bsd_signal' called. It is recommended to use 'sigaction' instead.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("int f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    int bsd_signal(0);\n"
Packit 2035a7
              "    return bsd_signal;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    struct hostent *hp;\n"
Packit 2035a7
              "    if(!hp = gethostbyname(\"127.0.0.1\")) {\n"
Packit 2035a7
              "        exit(1);\n"
Packit 2035a7
              "    }\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:4]: (style) Obsolescent function 'gethostbyname' called. It is recommended to use 'getaddrinfo' instead.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    long addr;\n"
Packit 2035a7
              "    addr = inet_addr(\"127.0.0.1\");\n"
Packit 2035a7
              "    if(!hp = gethostbyaddr((char *) &addr, sizeof(addr), AF_INET)) {\n"
Packit 2035a7
              "        exit(1);\n"
Packit 2035a7
              "    }\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:5]: (style) Obsolescent function 'gethostbyaddr' called. It is recommended to use 'getnameinfo' instead.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    usleep( 1000 );\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (style) Obsolescent function 'usleep' called. It is recommended to use 'nanosleep' or 'setitimer' instead.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_index() {
Packit 2035a7
        check("namespace n1 {\n"
Packit 2035a7
              "    int index(){};\n"
Packit 2035a7
              "}\n"
Packit 2035a7
              "int main()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    n1::index();\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("std::size_t f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::size_t index(0);\n"
Packit 2035a7
              "    index++;\n"
Packit 2035a7
              "    return index;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("int f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    return this->index();\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    int index( 0 );\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("const char f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    const char var[6] = \"index\";\n"
Packit 2035a7
              "    const char i = index(var, 0);\n"
Packit 2035a7
              "    return i;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:4]: (style) Obsolescent function 'index' called. It is recommended to use 'strchr' instead.\n",
Packit 2035a7
                      errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_qt_index() {
Packit 2035a7
        check("void TDataModel::forceRowRefresh(int row) {\n"
Packit 2035a7
              "    emit dataChanged(index(row, 0), index(row, columnCount() - 1));\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:2]: (style) Obsolescent function 'index' called. It is recommended to use 'strchr' instead.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_rindex() {
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    int rindex( 0 );\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    const char var[7] = \"rindex\";\n"
Packit 2035a7
              "    print(rindex(var, 0));\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:4]: (style) Obsolescent function 'rindex' called. It is recommended to use 'strrchr' instead.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_var() {
Packit 2035a7
        check("class Fred {\n"
Packit 2035a7
              "public:\n"
Packit 2035a7
              "    Fred() : index(0) { }\n"
Packit 2035a7
              "    int index;\n"
Packit 2035a7
              "};");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_gets() {
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char *x = gets(a);\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Obsolete function 'gets' called. It is recommended to use 'fgets' or 'gets_s' instead.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    foo(x, gets(a));\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Obsolete function 'gets' called. It is recommended to use 'fgets' or 'gets_s' instead.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_alloca() {
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char *x = alloca(10);\n"
Packit 2035a7
              "}", "test.cpp");  // #4382 - there are no VLAs in C++
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Obsolete function 'alloca' called.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char *x = alloca(10);\n"
Packit 2035a7
              "}", "test.c");
Packit 2035a7
        ASSERT_EQUALS("[test.c:3]: (warning) Obsolete function 'alloca' called. In C99 and later it is recommended to use a variable length array instead.\n", errout.str());
Packit 2035a7
Packit 2035a7
        settings.standards.c = Standards::C89;
Packit 2035a7
        settings.standards.cpp = Standards::CPP03;
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char *x = alloca(10);\n"
Packit 2035a7
              "}", "test.cpp");  // #4382 - there are no VLAs in C++
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char *x = alloca(10);\n"
Packit 2035a7
              "}", "test.c"); // #7558 - no alternative to alloca in C89
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char *x = alloca(10);\n"
Packit 2035a7
              "}", "test.c");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
        settings.standards.c = Standards::C11;
Packit 2035a7
        settings.standards.cpp = Standards::CPP11;
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    // ticket #3121
Packit 2035a7
    void prohibitedFunctions_declaredFunction() {
Packit 2035a7
        check("int ftime ( int a )\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    return a;\n"
Packit 2035a7
              "}\n"
Packit 2035a7
              "int main ()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    int b ; b = ftime ( 1 ) ;\n"
Packit 2035a7
              "    return 0 ;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    // test std::gets
Packit 2035a7
    void prohibitedFunctions_std_gets() {
Packit 2035a7
        check("void f(char * str)\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char *x = std::gets(str);\n"
Packit 2035a7
              "    char *y = gets(str);\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Obsolete function 'gets' called. It is recommended to use 'fgets' or 'gets_s' instead.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Obsolete function 'gets' called. It is recommended to use 'fgets' or 'gets_s' instead.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    // multiple use
Packit 2035a7
    void prohibitedFunctions_multiple() {
Packit 2035a7
        check("void f(char * str)\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char *x = std::gets(str);\n"
Packit 2035a7
              "    usleep( 1000 );\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Obsolete function 'gets' called. It is recommended to use 'fgets' or 'gets_s' instead.\n"
Packit 2035a7
                      "[test.cpp:4]: (style) Obsolescent function 'usleep' called. It is recommended to use 'nanosleep' or 'setitimer' instead.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_c_declaration() {
Packit 2035a7
        check("char * gets ( char * c ) ;\n"
Packit 2035a7
              "int main ()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char s [ 10 ] ;\n"
Packit 2035a7
              "    gets ( s ) ;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:5]: (warning) Obsolete function 'gets' called. It is recommended to use 'fgets' or 'gets_s' instead.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("int getcontext(ucontext_t *ucp);\n"
Packit 2035a7
              "int f (ucontext_t *ucp)\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    getcontext ( ucp ) ;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:4]: (portability) Obsolescent function 'getcontext' called. Applications are recommended to be rewritten to use POSIX threads.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_functionWithBody() {
Packit 2035a7
        check("char * gets ( char * c ) { return c; }\n"
Packit 2035a7
              "int main ()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char s [ 10 ] ;\n"
Packit 2035a7
              "    gets ( s ) ;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_crypt() {
Packit 2035a7
        check("void f(char *pwd)\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char *cpwd;"
Packit 2035a7
              "    crypt(pwd, cpwd);\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Return value of function crypt() is not used.\n"
Packit 2035a7
                      "[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());
Packit 2035a7
Packit 2035a7
        check("void f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    char *pwd = getpass(\"Password:\");"
Packit 2035a7
              "    char *cpwd;"
Packit 2035a7
              "    crypt(pwd, cpwd);\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Return value of function crypt() is not used.\n"
Packit 2035a7
                      "[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());
Packit 2035a7
Packit 2035a7
        check("int f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    int crypt = 0;"
Packit 2035a7
              "    return crypt;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void prohibitedFunctions_namespaceHandling() {
Packit 2035a7
        check("int f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    time_t t = 0;"
Packit 2035a7
              "    std::localtime(&t);\n"
Packit 2035a7
              "}");
Packit 2035a7
        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());
Packit 2035a7
Packit 2035a7
        // Passed as function argument
Packit 2035a7
        check("int f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    printf(\"Magic guess: %d\", getpwent());\n"
Packit 2035a7
              "}");
Packit 2035a7
        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());
Packit 2035a7
Packit 2035a7
        // Pass return value
Packit 2035a7
        check("int f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    time_t t = 0;"
Packit 2035a7
              "    struct tm *foo = localtime(&t);\n"
Packit 2035a7
              "}");
Packit 2035a7
        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());
Packit 2035a7
Packit 2035a7
        // Access via global namespace
Packit 2035a7
        check("int f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    ::getpwent();\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Return value of function getpwent() is not used.\n"
Packit 2035a7
                      "[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());
Packit 2035a7
Packit 2035a7
        // Be quiet on function definitions
Packit 2035a7
        check("int getpwent()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    return 123;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // Be quiet on other namespaces
Packit 2035a7
        check("int f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    foobar::getpwent();\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // Be quiet on class member functions
Packit 2035a7
        check("int f()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    foobar.getpwent();\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void invalidFunctionUsage1() {
Packit 2035a7
        check("int f() { memset(a,b,sizeof(a)!=12); }");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:1]: (error) Invalid memset() argument nr 3. A non-boolean value is required.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("int f() { memset(a,b,sizeof(a)!=0); }");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:1]: (error) Invalid memset() argument nr 3. A non-boolean value is required.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("int f() { memset(a,b,!c); }");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:1]: (error) Invalid memset() argument nr 3. A non-boolean value is required.\n", errout.str());
Packit 2035a7
Packit 2035a7
        // Ticket #6990
Packit 2035a7
        check("int f(bool c) { memset(a,b,c); }");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:1]: (error) Invalid memset() argument nr 3. A non-boolean value is required.\n", errout.str());
Packit 2035a7
        check("int f() { memset(a,b,true); }");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:1]: (error) Invalid memset() argument nr 3. A non-boolean value is required.\n", errout.str());
Packit 2035a7
Packit 2035a7
        // Ticket #6588 (c mode)
Packit 2035a7
        check("void record(char* buf, int n) {\n"
Packit 2035a7
              "  memset(buf, 0, n < 255);\n"           /* KO */
Packit 2035a7
              "  memset(buf, 0, n < 255 ? n : 255);\n" /* OK */
Packit 2035a7
              "}", "test.c");
Packit 2035a7
        ASSERT_EQUALS("[test.c:2]: (error) Invalid memset() argument nr 3. A non-boolean value is required.\n", errout.str());
Packit 2035a7
Packit 2035a7
        // Ticket #6588 (c++ mode)
Packit 2035a7
        check("void record(char* buf, int n) {\n"
Packit 2035a7
              "  memset(buf, 0, n < 255);\n"           /* KO */
Packit 2035a7
              "  memset(buf, 0, n < 255 ? n : 255);\n" /* OK */
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:2]: (error) Invalid memset() argument nr 3. A non-boolean value is required.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("int f() { strtol(a,b,sizeof(a)!=12); }");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:1]: (error) Invalid strtol() argument nr 3. The value is 0 or 1 (boolean) but the valid values are '0,2:36'.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("int f() { strtol(a,b,1); }");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:1]: (error) Invalid strtol() argument nr 3. The value is 1 but the valid values are '0,2:36'.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("int f() { strtol(a,b,10); }");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void mathfunctionCall_sqrt() {
Packit 2035a7
        // sqrt, sqrtf, sqrtl
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  sqrt(-1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  sqrtf(-1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  sqrtl(-1) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing value -1 to sqrt() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing value -1 to sqrtf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing value -1 to sqrtl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
Packit 2035a7
        // implementation-defined behaviour for "finite values of x<0" only:
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  sqrt(-0.) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  sqrtf(-0.) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  sqrtl(-0.) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  sqrt(1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  sqrtf(1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  sqrtl(1) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void mathfunctionCall_log() {
Packit 2035a7
        // log,log10,logf,logl,log10f,log10l
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  log(-2) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logf(-2) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logl(-2) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing value -2 to log() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing value -2 to logf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing value -2 to logl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  log(-1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logf(-1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logl(-1) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing value -1 to log() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing value -1 to logf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing value -1 to logl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  log(-1.0) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logf(-1.0) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logl(-1.0) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing value -1.0 to log() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing value -1.0 to logf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing value -1.0 to logl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  log(-0.1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logf(-0.1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logl(-0.1) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing value -0.1 to log() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing value -0.1 to logf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing value -0.1 to logl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  log(0) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logf(0.) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logl(0.0) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing value 0 to log() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing value 0. to logf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing value 0.0 to logl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  log(1E-3)    << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logf(1E-3)   << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logl(1E-3)   << std::endl;\n"
Packit 2035a7
              "    std::cout <<  log(1.0E-3)  << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logf(1.0E-3) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logl(1.0E-3) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  log(1.0E+3)  << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logf(1.0E+3) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logl(1.0E+3) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  log(2.0)     << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logf(2.0)    << std::endl;\n"
Packit 2035a7
              "    std::cout <<  logf(2.0f)   << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::string *log(0);\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #3473 - no warning if "log" is a variable
Packit 2035a7
        check("Fred::Fred() : log(0) { }");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #5748
Packit 2035a7
        check("void f() { foo.log(0); }");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void mathfunctionCall_acos() {
Packit 2035a7
        // acos, acosf, acosl
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              " return acos(-1)      \n"
Packit 2035a7
              "    + acos(0.1)       \n"
Packit 2035a7
              "    + acos(0.0001)    \n"
Packit 2035a7
              "    + acos(0.01)      \n"
Packit 2035a7
              "    + acos(1.0E-1)    \n"
Packit 2035a7
              "    + acos(-1.0E-1)   \n"
Packit 2035a7
              "    + acos(+1.0E-1)   \n"
Packit 2035a7
              "    + acos(0.1E-1)    \n"
Packit 2035a7
              "    + acos(+0.1E-1)   \n"
Packit 2035a7
              "    + acos(-0.1E-1)   \n"
Packit 2035a7
              "    + acosf(-1)       \n"
Packit 2035a7
              "    + acosf(0.1)      \n"
Packit 2035a7
              "    + acosf(0.0001)   \n"
Packit 2035a7
              "    + acosf(0.01)     \n"
Packit 2035a7
              "    + acosf(1.0E-1)   \n"
Packit 2035a7
              "    + acosf(-1.0E-1)  \n"
Packit 2035a7
              "    + acosf(+1.0E-1)  \n"
Packit 2035a7
              "    + acosf(0.1E-1)   \n"
Packit 2035a7
              "    + acosf(+0.1E-1)  \n"
Packit 2035a7
              "    + acosf(-0.1E-1)  \n"
Packit 2035a7
              "    + acosl(-1)       \n"
Packit 2035a7
              "    + acosl(0.1)      \n"
Packit 2035a7
              "    + acosl(0.0001)   \n"
Packit 2035a7
              "    + acosl(0.01)     \n"
Packit 2035a7
              "    + acosl(1.0E-1)   \n"
Packit 2035a7
              "    + acosl(-1.0E-1)  \n"
Packit 2035a7
              "    + acosl(+1.0E-1)  \n"
Packit 2035a7
              "    + acosl(0.1E-1)   \n"
Packit 2035a7
              "    + acosl(+0.1E-1)  \n"
Packit 2035a7
              "    + acosl(-0.1E-1); \n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  acos(1.1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  acosf(1.1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  acosl(1.1) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing value 1.1 to acos() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing value 1.1 to acosf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing value 1.1 to acosl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  acos(-1.1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  acosf(-1.1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  acosl(-1.1) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing value -1.1 to acos() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing value -1.1 to acosf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing value -1.1 to acosl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void mathfunctionCall_asin() {
Packit 2035a7
        // asin, asinf, asinl
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              " return asin(1)       \n"
Packit 2035a7
              "    + asin(-1)        \n"
Packit 2035a7
              "    + asin(0.1)       \n"
Packit 2035a7
              "    + asin(0.0001)    \n"
Packit 2035a7
              "    + asin(0.01)      \n"
Packit 2035a7
              "    + asin(1.0E-1)    \n"
Packit 2035a7
              "    + asin(-1.0E-1)   \n"
Packit 2035a7
              "    + asin(+1.0E-1)   \n"
Packit 2035a7
              "    + asin(0.1E-1)    \n"
Packit 2035a7
              "    + asin(+0.1E-1)   \n"
Packit 2035a7
              "    + asin(-0.1E-1)   \n"
Packit 2035a7
              "    + asinf(1)        \n"
Packit 2035a7
              "    + asinf(-1)       \n"
Packit 2035a7
              "    + asinf(0.1)      \n"
Packit 2035a7
              "    + asinf(0.0001)   \n"
Packit 2035a7
              "    + asinf(0.01)     \n"
Packit 2035a7
              "    + asinf(1.0E-1)   \n"
Packit 2035a7
              "    + asinf(-1.0E-1)  \n"
Packit 2035a7
              "    + asinf(+1.0E-1)  \n"
Packit 2035a7
              "    + asinf(0.1E-1)   \n"
Packit 2035a7
              "    + asinf(+0.1E-1)  \n"
Packit 2035a7
              "    + asinf(-0.1E-1)  \n"
Packit 2035a7
              "    + asinl(1)        \n"
Packit 2035a7
              "    + asinl(-1)       \n"
Packit 2035a7
              "    + asinl(0.1)      \n"
Packit 2035a7
              "    + asinl(0.0001)   \n"
Packit 2035a7
              "    + asinl(0.01)     \n"
Packit 2035a7
              "    + asinl(1.0E-1)   \n"
Packit 2035a7
              "    + asinl(-1.0E-1)  \n"
Packit 2035a7
              "    + asinl(+1.0E-1)  \n"
Packit 2035a7
              "    + asinl(0.1E-1)   \n"
Packit 2035a7
              "    + asinl(+0.1E-1)  \n"
Packit 2035a7
              "    + asinl(-0.1E-1); \n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  asin(1.1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  asinf(1.1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  asinl(1.1) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing value 1.1 to asin() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing value 1.1 to asinf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing value 1.1 to asinl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  asin(-1.1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  asinf(-1.1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  asinl(-1.1) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing value -1.1 to asin() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing value -1.1 to asinf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing value -1.1 to asinl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void mathfunctionCall_pow() {
Packit 2035a7
        // pow, powf, powl
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  pow(0,-10) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  powf(0,-10) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  powl(0,-10) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing values 0 and -10 to pow() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing values 0 and -10 to powf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing values 0 and -10 to powl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  pow(0,10) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  powf(0,10) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  powl(0,10) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void mathfunctionCall_atan2() {
Packit 2035a7
        // atan2
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  atan2(1,1)         ;\n"
Packit 2035a7
              "    std::cout <<  atan2(-1,-1)       ;\n"
Packit 2035a7
              "    std::cout <<  atan2(0.1,1)       ;\n"
Packit 2035a7
              "    std::cout <<  atan2(0.0001,100)  ;\n"
Packit 2035a7
              "    std::cout <<  atan2(0.0,1e-1)    ;\n"
Packit 2035a7
              "    std::cout <<  atan2(1.0E-1,-3)   ;\n"
Packit 2035a7
              "    std::cout <<  atan2(-1.0E-1,+2)  ;\n"
Packit 2035a7
              "    std::cout <<  atan2(+1.0E-1,0)   ;\n"
Packit 2035a7
              "    std::cout <<  atan2(0.1E-1,3)    ;\n"
Packit 2035a7
              "    std::cout <<  atan2(+0.1E-1,1)   ;\n"
Packit 2035a7
              "    std::cout <<  atan2(-0.1E-1,8)   ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(1,1)        ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(-1,-1)      ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(0.1,1)      ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(0.0001,100) ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(0.0,1e-1)   ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(1.0E-1,-3)  ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(-1.0E-1,+2) ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(+1.0E-1,0)  ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(0.1E-1,3)   ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(+0.1E-1,1)  ;\n"
Packit 2035a7
              "    std::cout <<  atan2f(-0.1E-1,8)  ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(1,1)        ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(-1,-1)      ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(0.1,1)      ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(0.0001,100) ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(0.0,1e-1)   ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(1.0E-1,-3)  ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(-1.0E-1,+2) ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(+1.0E-1,0)  ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(0.1E-1,3)   ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(+0.1E-1,1)  ;\n"
Packit 2035a7
              "    std::cout <<  atan2l(-0.1E-1,8)  ;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  atan2(0,0) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  atan2f(0,0) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  atan2l(0,0) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing values 0 and 0 to atan2() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing values 0 and 0 to atan2f() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing values 0 and 0 to atan2l() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void mathfunctionCall_fmod() {
Packit 2035a7
        // fmod, fmodl, fmodf
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  fmod(1.0,0) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  fmodf(1.0,0) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  fmodl(1.0,0) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Passing values 1.0 and 0 to fmod() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:4]: (warning) Passing values 1.0 and 0 to fmodf() leads to implementation-defined result.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) Passing values 1.0 and 0 to fmodl() leads to implementation-defined result.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo()\n"
Packit 2035a7
              "{\n"
Packit 2035a7
              "    std::cout <<  fmod(1.0,1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  fmodf(1.0,1) << std::endl;\n"
Packit 2035a7
              "    std::cout <<  fmodl(1.0,1) << std::endl;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void mathfunctionCall_precision() {
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    print(exp(x) - 1);\n"
Packit 2035a7
              "    print(log(1 + x));\n"
Packit 2035a7
              "    print(1 - erf(x));\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:2]: (style) Expression 'exp(x) - 1' can be replaced by 'expm1(x)' to avoid loss of precision.\n"
Packit 2035a7
                      "[test.cpp:3]: (style) Expression 'log(1 + x)' can be replaced by 'log1p(x)' to avoid loss of precision.\n"
Packit 2035a7
                      "[test.cpp:4]: (style) Expression '1 - erf(x)' can be replaced by 'erfc(x)' to avoid loss of precision.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    print(exp(x) - 1.0);\n"
Packit 2035a7
              "    print(log(1.0 + x));\n"
Packit 2035a7
              "    print(1.0 - erf(x));\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:2]: (style) Expression 'exp(x) - 1' can be replaced by 'expm1(x)' to avoid loss of precision.\n"
Packit 2035a7
                      "[test.cpp:3]: (style) Expression 'log(1 + x)' can be replaced by 'log1p(x)' to avoid loss of precision.\n"
Packit 2035a7
                      "[test.cpp:4]: (style) Expression '1 - erf(x)' can be replaced by 'erfc(x)' to avoid loss of precision.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    print(exp(3 + x*f(a)) - 1);\n"
Packit 2035a7
              "    print(log(x*4 + 1));\n"
Packit 2035a7
              "    print(1 - erf(34*x + f(x) - c));\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:2]: (style) Expression 'exp(x) - 1' can be replaced by 'expm1(x)' to avoid loss of precision.\n"
Packit 2035a7
                      "[test.cpp:3]: (style) Expression 'log(1 + x)' can be replaced by 'log1p(x)' to avoid loss of precision.\n"
Packit 2035a7
                      "[test.cpp:4]: (style) Expression '1 - erf(x)' can be replaced by 'erfc(x)' to avoid loss of precision.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    print(2*exp(x) - 1);\n"
Packit 2035a7
              "    print(1 - erf(x)/2.0);\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void checkIgnoredReturnValue() {
Packit 2035a7
        Settings settings2;
Packit 2035a7
        settings2.addEnabled("warning");
Packit 2035a7
        const char xmldata[] = "\n"
Packit 2035a7
                               "<def version=\"2\">\n"
Packit 2035a7
                               "  <function name=\"mystrcmp,foo::mystrcmp\">\n"
Packit 2035a7
                               "    <use-retval/>\n"
Packit 2035a7
                               "    <arg nr=\"1\"/>\n"
Packit 2035a7
                               "    <arg nr=\"2\"/>\n"
Packit 2035a7
                               "  </function>\n"
Packit 2035a7
                               "</def>";
Packit 2035a7
        tinyxml2::XMLDocument doc;
Packit 2035a7
        doc.Parse(xmldata, sizeof(xmldata));
Packit 2035a7
        settings2.library.load(doc);
Packit 2035a7
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "  mystrcmp(a, b);\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:2]: (warning) Return value of function mystrcmp() is not used.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "  foo::mystrcmp(a, b);\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:2]: (warning) Return value of function foo::mystrcmp() is not used.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "  foo x;\n"
Packit 2035a7
              "  x.mystrcmp(a, b);\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Return value of function x.mystrcmp() is not used.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("bool mystrcmp(char* a, char* b);\n" // cppcheck sees a custom strcmp definition, but it returns a value. Assume it is the one specified in the library.
Packit 2035a7
              "void foo() {\n"
Packit 2035a7
              "    mystrcmp(a, b);\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) Return value of function mystrcmp() is not used.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void mystrcmp(char* a, char* b);\n" // cppcheck sees a custom strcmp definition which returns void!
Packit 2035a7
              "void foo() {\n"
Packit 2035a7
              "    mystrcmp(a, b);\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    class mystrcmp { mystrcmp() {} };\n" // strcmp is a constructor definition here
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    return mystrcmp(a, b);\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    return foo::mystrcmp(a, b);\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    if(mystrcmp(a, b));\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    bool b = mystrcmp(a, b);\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #6194
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    MyStrCmp mystrcmp(x, y);\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #6197
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "    abc::def.mystrcmp(a,b);\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #6233
Packit 2035a7
        check("int main() {\n"
Packit 2035a7
              "    auto lambda = [](double value) {\n"
Packit 2035a7
              "        double rounded = floor(value + 0.5);\n"
Packit 2035a7
              "        printf(\"Rounded value = %f\\n\", rounded);\n"
Packit 2035a7
              "    };\n"
Packit 2035a7
              "    lambda(13.3);\n"
Packit 2035a7
              "    return 0;\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #6669
Packit 2035a7
        check("void foo(size_t size) {\n"
Packit 2035a7
              "   void * res{malloc(size)};\n"
Packit 2035a7
              "}");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #7447
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "   int x{mystrcmp(a,b)};\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #7905
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "   int x({mystrcmp(a,b)});\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #7979 - code is not well configured
Packit 2035a7
        check("void foo() {\n"
Packit 2035a7
              "  DEBUG(x(); mystrcmp(a,b););\n"
Packit 2035a7
              "}", "test.cpp", &settings2);
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void foo() {\n" // don't crash
Packit 2035a7
              "  DEBUG(123)(mystrcmp(a,b))(fd);\n"
Packit 2035a7
              "}", "test.c", &settings2);
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void memsetZeroBytes() {
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "    memset(p, 10, 0x0);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:2]: (warning) memset() called to fill 0 bytes.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "    memset(p, sizeof(p), 0);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:2]: (warning) memset() called to fill 0 bytes.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "    memset(p, sizeof(p), i);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #6269 false positives in case of overloaded standard library functions
Packit 2035a7
        check("class c {\n"
Packit 2035a7
              "  void memset( int i );\n"
Packit 2035a7
              "  void f( void )   {\n"
Packit 2035a7
              "     memset( 0 );\n"
Packit 2035a7
              "  }\n"
Packit 2035a7
              "};");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        // #7285
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "    memset(&tm, sizeof(tm), 0);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:2]: (warning) memset() called to fill 0 bytes.\n", errout.str());
Packit 2035a7
Packit 2035a7
    }
Packit 2035a7
Packit 2035a7
    void memsetInvalid2ndParam() {
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "    int* is = new int[10];\n"
Packit 2035a7
              "    memset(is, 1.0f, 40);\n"
Packit 2035a7
              "    int* is2 = new int[10];\n"
Packit 2035a7
              "    memset(is2, 0.1f, 40);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (portability) The 2nd memset() argument '1.0f' is a float, its representation is implementation defined.\n"
Packit 2035a7
                      "[test.cpp:5]: (portability) The 2nd memset() argument '0.1f' is a float, its representation is implementation defined.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "    int* is = new int[10];\n"
Packit 2035a7
              "    float g = computeG();\n"
Packit 2035a7
              "    memset(is, g, 40);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:4]: (portability) The 2nd memset() argument 'g' is a float, its representation is implementation defined.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "    int* is = new int[10];\n"
Packit 2035a7
              "    memset(is, 0.0f, 40);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f() {\n" // FP
Packit 2035a7
              "    float x = 2.3f;\n"
Packit 2035a7
              "    memset(a, (x?64:0), 40);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "    short ss[] = {1, 2};\n"
Packit 2035a7
              "    memset(ss, 256, 4);\n"
Packit 2035a7
              "    short ss2[2];\n"
Packit 2035a7
              "    memset(ss2, -129, 4);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:3]: (warning) The 2nd memset() argument '256' doesn't fit into an 'unsigned char'.\n"
Packit 2035a7
                      "[test.cpp:5]: (warning) The 2nd memset() argument '-129' doesn't fit into an 'unsigned char'.\n", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "    int is[10];\n"
Packit 2035a7
              "    memset(is, 0xEE, 40);\n"
Packit 2035a7
              "    unsigned char* cs = malloc(256);\n"
Packit 2035a7
              "    memset(cs, -1, 256);\n"
Packit 2035a7
              "    short* ss[30];\n"
Packit 2035a7
              "    memset(ss, -128, 60);\n"
Packit 2035a7
              "    char cs2[30];\n"
Packit 2035a7
              "    memset(cs2, 255, 30);\n"
Packit 2035a7
              "    char cs3[30];\n"
Packit 2035a7
              "    memset(cs3, 0, 30);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("", errout.str());
Packit 2035a7
Packit 2035a7
        check("void f() {\n"
Packit 2035a7
              "    int is[10];\n"
Packit 2035a7
              "    const int i = g();\n"
Packit 2035a7
              "    memset(is, 1.0f + i, 40);\n"
Packit 2035a7
              "}\n");
Packit 2035a7
        ASSERT_EQUALS("[test.cpp:4]: (portability) The 2nd memset() argument '1.0f+i' is a float, its representation is implementation defined.\n", errout.str());
Packit 2035a7
    }
Packit 2035a7
};
Packit 2035a7
Packit 2035a7
REGISTER_TEST(TestFunctions)