// RUN: %check_clang_tidy -std=c++14 %s bugprone-signal-handler %t -- -- -isystem %clang_tidy_headers -isystem %S/Inputs/signal-handler -target x86_64-unknown-unknown
// FIXME: Fix the checker to work in C++17 or later mode.
#include "stdcpp.h"
#include "stdio.h"
// Functions called "signal" that are different from the system version.
typedef void (*callback_t)(int);
void signal(int, callback_t, int);
namespace ns {
void signal(int, callback_t);
}
extern "C" void handler_unsafe(int) {
printf("xxx");
}
extern "C" void handler_unsafe_1(int) {
printf("xxx");
}
namespace test_invalid_handler {
void handler_non_extern_c(int) {
printf("xxx");
}
struct A {
static void handler_member(int) {
printf("xxx");
}
};
void test() {
std::signal(SIGINT, handler_unsafe_1);
// CHECK-MESSAGES: :[[@LINE-17]]:3: warning: standard function 'printf' may not be asynchronous-safe; calling it from a signal handler may be dangerous [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:23: note: function 'handler_unsafe_1' registered here as signal handler
std::signal(SIGINT, handler_non_extern_c);
// CHECK-MESSAGES: :[[@LINE-1]]:23: warning: functions without C linkage are not allowed as signal handler (until C++17) [bugprone-signal-handler]
std::signal(SIGINT, A::handler_member);
// CHECK-MESSAGES: :[[@LINE-1]]:23: warning: functions without C linkage are not allowed as signal handler (until C++17) [bugprone-signal-handler]
std::signal(SIGINT, [](int) { printf("xxx"); });
// CHECK-MESSAGES: :[[@LINE-1]]:23: warning: lambda function is not allowed as signal handler (until C++17) [bugprone-signal-handler]
// This case is (deliberately) not found by the checker.
std::signal(SIGINT, [](int) -> callback_t { return &handler_unsafe; }(1));
}
} // namespace test_invalid_handler
namespace test_non_standard_signal_call {
struct Signal {
static void signal(int, callback_t);
};
void test() {
// No diagnostics here. All these signal calls differ from the standard system one.
signal(SIGINT, handler_unsafe, 1);
ns::signal(SIGINT, handler_unsafe);
Signal::signal(SIGINT, handler_unsafe);
system_other::signal(SIGINT, handler_unsafe);
}
} // namespace test_non_standard_signal_call
namespace test_cpp_construct_in_handler {
struct Struct {
virtual ~Struct() {}
void f1();
int *begin();
int *end();
static void f2();
};
struct Derived : public Struct {
};
struct X {
X(int, float);
};
Struct *S_Global;
const Struct *S_GlobalConst;
void f_non_extern_c() {
}
void f_default_arg(int P1 = 0) {
}
extern "C" void handler_cpp(int) {
using namespace ::test_cpp_construct_in_handler;
// These calls are not found as problems.
// (Called functions are not analyzed if the current function has already
// other problems.)
f_non_extern_c();
Struct::f2();
// 'auto' is not disallowed
auto Auto = 28u;
Struct S;
// CHECK-MESSAGES: :[[@LINE-1]]:10: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:10: remark: internally, the statement is parsed as a 'CXXConstructExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
S_Global->f1();
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:3: remark: internally, the statement is parsed as a 'CXXMemberCallExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
const Struct &SRef = Struct();
// CHECK-MESSAGES: :[[@LINE-1]]:24: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:24: remark: internally, the statement is parsed as a 'CXXBindTemporaryExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
X(3, 4.4);
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:3: remark: internally, the statement is parsed as a 'CXXTemporaryObjectExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
auto L = [](int i) { printf("%d", i); };
// CHECK-MESSAGES: :[[@LINE-1]]:12: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:12: remark: internally, the statement is parsed as a 'CXXConstructExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
L(2);
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:3: remark: internally, the statement is parsed as a 'CXXOperatorCallExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
try {
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:3: remark: internally, the statement is parsed as a 'CXXTryStmt'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
int A;
} catch (int) {
};
// CHECK-MESSAGES: :[[@LINE-2]]:5: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-3]]:5: remark: internally, the statement is parsed as a 'CXXCatchStmt'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
throw(12);
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:3: remark: internally, the statement is parsed as a 'CXXThrowExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
for (int I : S) {
}
// CHECK-MESSAGES: :[[@LINE-2]]:3: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-3]]:3: remark: internally, the statement is parsed as a 'CXXForRangeStmt'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
// CHECK-MESSAGES: :[[@LINE-5]]:14: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-6]]:14: remark: internally, the statement is parsed as a 'CXXMemberCallExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
int Int = *(reinterpret_cast<int *>(&S));
// CHECK-MESSAGES: :[[@LINE-1]]:15: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:15: remark: internally, the statement is parsed as a 'CXXReinterpretCastExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
Int = static_cast<int>(12.34);
// CHECK-MESSAGES: :[[@LINE-1]]:9: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:9: remark: internally, the statement is parsed as a 'CXXStaticCastExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
Derived *Der = dynamic_cast<Derived *>(S_Global);
// CHECK-MESSAGES: :[[@LINE-1]]:18: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:18: remark: internally, the statement is parsed as a 'CXXDynamicCastExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
Struct *SPtr = const_cast<Struct *>(S_GlobalConst);
// CHECK-MESSAGES: :[[@LINE-1]]:18: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:18: remark: internally, the statement is parsed as a 'CXXConstCastExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
Int = int(12.34);
// CHECK-MESSAGES: :[[@LINE-1]]:9: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:9: remark: internally, the statement is parsed as a 'CXXFunctionalCastExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
int *IPtr = new int[10];
// CHECK-MESSAGES: :[[@LINE-1]]:15: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:15: remark: internally, the statement is parsed as a 'CXXNewExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
delete[] IPtr;
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:3: remark: internally, the statement is parsed as a 'CXXDeleteExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
IPtr = nullptr;
// CHECK-MESSAGES: :[[@LINE-1]]:10: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:10: remark: internally, the statement is parsed as a 'CXXNullPtrLiteralExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
bool Bool = true;
// CHECK-MESSAGES: :[[@LINE-1]]:15: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:15: remark: internally, the statement is parsed as a 'CXXBoolLiteralExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
f_default_arg();
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:3: remark: internally, the statement is parsed as a 'CXXDefaultArgExpr'
// CHECK-MESSAGES: :198:23: note: function 'handler_cpp' registered here as signal handler
}
void test() {
std::signal(SIGINT, handler_cpp);
}
} // namespace test_cpp_construct_in_handler
namespace test_cpp_indirect {
void non_extern_c() {
int *P = nullptr;
}
extern "C" void call_cpp_indirect() {
int *P = nullptr;
// CHECK-MESSAGES: :[[@LINE-1]]:12: warning: C++-only construct is not allowed in signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE-2]]:12: remark: internally, the statement is parsed as a 'CXXNullPtrLiteralExpr'
// CHECK-MESSAGES: :[[@LINE+8]]:3: note: function 'call_cpp_indirect' called here from 'handler_cpp_indirect'
// CHECK-MESSAGES: :[[@LINE+11]]:23: note: function 'handler_cpp_indirect' registered here as signal handler
}
extern "C" void handler_cpp_indirect(int) {
non_extern_c();
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: functions without C linkage are not allowed as signal handler (until C++17) [bugprone-signal-handler]
// CHECK-MESSAGES: :[[@LINE+5]]:23: note: function 'handler_cpp_indirect' registered here as signal handler
call_cpp_indirect();
}
void test() {
std::signal(SIGINT, handler_cpp_indirect);
}
} // namespace test_cpp_indirect