
Added by https://github.com/llvm/llvm-project/pull/144745. These tests cause Clang -cc1 to generate the option -x86-asm-syntax=intel, which is only available if you have included the x86 target. <<<<<< 1: clang: warning: argument unused during compilation: '-c' [-Wunused-command-line-argument] label:38'0 X~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ error: no match found label:38'1 ? possible intended match 2: clang (LLVM option parsing): Unknown command line argument '-x86-asm-syntax=intel'. Try: 'clang (LLVM option parsing) --help' label:38'0 ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ 3: clang (LLVM option parsing): Did you mean '--asan-stack=intel'? label:38'0 ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ >>>>>>
141 lines
3.6 KiB
C++
141 lines
3.6 KiB
C++
// REQUIRES: x86-registered-target
|
|
// RUN: %clang_cl -c --target=x86_64-windows-msvc -EHs-c- -O2 -GS- \
|
|
// RUN: -Xclang=-import-call-optimization \
|
|
// RUN: -clang:-S -clang:-o- -- %s 2>&1 \
|
|
// RUN: | FileCheck %s
|
|
|
|
#ifdef __clang__
|
|
#define NO_TAIL __attribute((disable_tail_calls))
|
|
#else
|
|
#define NO_TAIL
|
|
#endif
|
|
|
|
void might_throw();
|
|
void other_func(int x);
|
|
|
|
void does_not_throw() noexcept(true);
|
|
|
|
extern "C" void __declspec(dllimport) some_dll_import();
|
|
|
|
class HasDtor {
|
|
int x;
|
|
char foo[40];
|
|
|
|
public:
|
|
explicit HasDtor(int x);
|
|
~HasDtor();
|
|
};
|
|
|
|
void normal_has_regions() {
|
|
{
|
|
HasDtor hd{42};
|
|
|
|
// because state changes, we expect the HasDtor::HasDtor() call to have a NOP
|
|
might_throw();
|
|
}
|
|
|
|
other_func(10);
|
|
}
|
|
// CHECK-LABEL: .def "?normal_has_regions@@YAXXZ"
|
|
// CHECK: .seh_endprologue
|
|
// CHECK: call "??0HasDtor@@QEAA@H@Z"
|
|
// CHECK-NEXT: call "?might_throw@@YAXXZ"
|
|
// CHECK-NEXT: mov
|
|
// CHECK: call "??1HasDtor@@QEAA@XZ"
|
|
// CHECK-NEXT: mov ecx, 10
|
|
// CHECK-NEXT: call "?other_func@@YAXH@Z"
|
|
// CHECK-NEXT: nop
|
|
// CHECK-NEXT: .seh_startepilogue
|
|
// CHECK-NOT: "$ip2state$?normal_has_regions@@YAXXZ"
|
|
|
|
// This tests a tail call to a destructor.
|
|
void case_dtor_arg_empty_body(HasDtor x)
|
|
{
|
|
}
|
|
// CHECK-LABEL: .def "?case_dtor_arg_empty_body@@YAXVHasDtor@@@Z"
|
|
// CHECK: jmp "??1HasDtor@@QEAA@XZ"
|
|
|
|
int case_dtor_arg_empty_with_ret(HasDtor x)
|
|
{
|
|
// The call to HasDtor::~HasDtor() should NOT have a NOP because the
|
|
// following "mov eax, 100" instruction is in the same EH state.
|
|
return 100;
|
|
}
|
|
// CHECK-LABEL: .def "?case_dtor_arg_empty_with_ret@@YAHVHasDtor@@@Z"
|
|
// CHECK: .seh_endprologue
|
|
// CHECK: call "??1HasDtor@@QEAA@XZ"
|
|
// CHECK-NOT: nop
|
|
// CHECK: mov eax, 100
|
|
// CHECK: .seh_startepilogue
|
|
// CHECK: .seh_endepilogue
|
|
// CHECK: .seh_endproc
|
|
|
|
void case_except_simple_call() NO_TAIL
|
|
{
|
|
does_not_throw();
|
|
}
|
|
|
|
// This tests that the destructor is called right before SEH_BeginEpilogue,
|
|
// but in a function that has a return value.
|
|
int case_dtor_arg_calls_no_throw(HasDtor x)
|
|
{
|
|
does_not_throw(); // no NOP expected
|
|
return 100;
|
|
}
|
|
|
|
// Check the behavior of CALLs that are at the end of MBBs. If a CALL is within
|
|
// a non-null EH state (state -1) and is at the end of an MBB, then we expect
|
|
// to find an EH_LABEL after the CALL. This causes us to insert a NOP, which
|
|
// is the desired result.
|
|
void case_dtor_runs_after_join(int x) {
|
|
|
|
// ctor call does not need a NOP, because it has real instructions after it
|
|
HasDtor hd{42};
|
|
|
|
if (x) {
|
|
might_throw();
|
|
} else {
|
|
other_func(10);
|
|
}
|
|
does_not_throw();
|
|
// ~HasDtor() runs
|
|
}
|
|
|
|
// CHECK-LABEL: .def "?case_dtor_runs_after_join@@YAXH@Z"
|
|
// CHECK: .seh_endprologue
|
|
// CHECK: call "??0HasDtor@@QEAA@H@Z"
|
|
// CHECK-NEXT: test
|
|
// CHECK: call "?might_throw@@YAXXZ"
|
|
// CHECK-NEXT: jmp
|
|
// CHECK: call "?other_func@@YAXH@Z"
|
|
// CHECK-NEXT: .LBB
|
|
// CHECK: call "?does_not_throw@@YAXXZ"
|
|
// CHECK-NEXT: lea
|
|
// CHECK-NEXT: call "??1HasDtor@@QEAA@XZ"
|
|
// CHECK-NEXT: nop
|
|
// CHECK-NEXT: .seh_startepilogue
|
|
// CHECK-NOT: "$ip2state$?case_dtor_runs_after_join@@YAXH@Z":
|
|
|
|
|
|
// Check the behavior of NOP padding around tail calls.
|
|
// We do not expect to insert NOPs around tail calls.
|
|
// However, the first call (to other_func()) does get a NOP
|
|
// because it comes before .seh_startepilogue.
|
|
void case_tail_call_no_eh() {
|
|
// ordinary call
|
|
other_func(10);
|
|
|
|
// tail call; no NOP padding after JMP
|
|
does_not_throw();
|
|
}
|
|
|
|
// CHECK-LABEL: .def "?case_tail_call_no_eh@@YAXXZ"
|
|
// CHECK: .seh_endprologue
|
|
// CHECK: call "?other_func@@YAXH@Z"
|
|
// CHECK-NEXT: nop
|
|
// CHECK-NEXT: .seh_startepilogue
|
|
// CHECK: .seh_endepilogue
|
|
// CHECK: jmp "?does_not_throw@@YAXXZ"
|
|
// CHECK-NOT: nop
|
|
// CHECK: .seh_endproc
|