/* * Copyright (c) 2016-present, Facebook, Inc. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. */ #include #include #include #include extern int rand(); extern void __infer_sql_sink(std::string query, int i); namespace execs { // mocking gflags-generated field extern char* FLAGS_cli_string; extern int FLAGS_cli_int; int callAllSinks(const char* stringSource, char** arrSource) { switch (rand()) { case 1: return execl(NULL, stringSource); case 2: return execl(stringSource, NULL); case 3: // just one test for varargs; assume we get it right in the other cases return execl(NULL, NULL, stringSource); case 4: return execlp(NULL, stringSource); case 5: return execlp(stringSource, NULL); case 6: return execle(NULL, stringSource); case 7: return execle(stringSource, NULL); case 8: return execv(stringSource, NULL); case 9: return execvp(stringSource, NULL); case 10: return execv(NULL, arrSource); case 11: return execvp(NULL, arrSource); } return 0; } int callExecBad() { const char* stringSource = std::getenv("ENV_VAR"); char* arrSource[1] = {std::getenv("ENV_VAR")}; switch (rand()) { case 1: return execl(NULL, stringSource); case 2: return execl(stringSource, NULL); case 3: // just one test for varargs; assume we get it right in the other cases return execl(NULL, NULL, stringSource); case 4: return execlp(NULL, stringSource); case 5: return execlp(stringSource, NULL); case 6: return execle(NULL, stringSource); case 7: return execle(stringSource, NULL); case 8: return execv(stringSource, NULL); case 9: return execvp(stringSource, NULL); case 10: return execv(NULL, arrSource); case 11: return execvp(NULL, arrSource); case 12: return execve(stringSource, NULL, NULL); case 13: return execve(NULL, arrSource, NULL); case 14: return system(stringSource); case 15: FILE* f = popen(stringSource, "w"); return pclose(f); } return 0; } extern char* getenv(const char* var); void execConstantStringOk() { callAllSinks("something.sh", NULL); } void customGetEnvOk() { const char* source = execs::getenv("ENV_VAR"); return execl(NULL, source); } void exec_string_flag_bad() { execl(FLAGS_cli_string, NULL); } void exec_int_flag_ok() { char buffer[25]; sprintf(buffer, "foo -i %d", FLAGS_cli_int); execl(buffer, NULL); } char* return_global() { char* local = FLAGS_cli_string; return local; } void exec_string_flag_interproc_bad() { char* flag = return_global(); execl(flag, NULL); } void sql_on_env_var_bad() { std::string source = (std::string)std::getenv("ENV_VAR"); __infer_sql_sink(source, 0); } void tainted_flag_popen_ok() { char* tainted = std::getenv("something"); popen("ls", tainted); // should not warn; } class Obj { void endpoint(int i, char c, std::string s, char* c_ptr, char c_arr[], std::string* s_ptr) { __infer_sql_sink(nullptr, i); // don't report __infer_sql_sink(nullptr, c); // don't report __infer_sql_sink(s, 0); // report __infer_sql_sink(*s_ptr, 0); // report __infer_sql_sink(c_ptr, 0); // report __infer_sql_sink(c_arr, 0); // report } }; } // namespace execs