forked from OSchip/llvm-project
[Windows SEH] Fix abnormal-exits in _try
Summary: Per Windows SEH Spec, except _leave, all other early exits of a _try (goto/return/continue/break) are considered abnormal exits. In those cases, the first parameter passes to its _finally funclet should be TRUE to indicate an abnormal-termination. One way to implement abnormal exits in _try is to invoke Windows runtime _local_unwind() (MSVC approach) that will invoke _dtor funclet where abnormal-termination flag is always TRUE when calling _finally. Obviously this approach is less optimal and is complicated to implement in Clang. Clang today has a NormalCleanupDestSlot mechanism to dispatch multiple exits at the end of _try. Since _leave (or try-end fall-through) is always Indexed with 0 in that NormalCleanupDestSlot, this fix takes the advantage of that mechanism and just passes NormalCleanupDest ID as 1st Arg to _finally. Reviewers: rnk, eli.friedman, JosephTremoulet, asmith, efriedma Reviewed By: efriedma Subscribers: efriedma, cfe-commits Tags: #clang Differential Revision: https://reviews.llvm.org/D77936
This commit is contained in:
parent
b257d3c8a8
commit
4eabd00612
|
@ -860,6 +860,9 @@ void CodeGenFunction::PopCleanupBlock(bool FallthroughIsBranchThrough) {
|
|||
// TODO: base this on the number of branch-afters and fixups
|
||||
const unsigned SwitchCapacity = 10;
|
||||
|
||||
// pass the abnormal exit flag to Fn (SEH cleanup)
|
||||
cleanupFlags.setHasExitSwitch();
|
||||
|
||||
llvm::LoadInst *Load =
|
||||
createLoadInstBefore(getNormalCleanupDestSlot(), "cleanup.dest",
|
||||
nullptr);
|
||||
|
|
|
@ -1639,6 +1639,19 @@ struct PerformSEHFinally final : EHScopeStack::Cleanup {
|
|||
|
||||
llvm::Value *IsForEH =
|
||||
llvm::ConstantInt::get(CGF.ConvertType(ArgTys[0]), F.isForEHCleanup());
|
||||
|
||||
// Except _leave and fall-through at the end, all other exits in a _try
|
||||
// (return/goto/continue/break) are considered as abnormal terminations
|
||||
// since _leave/fall-through is always Indexed 0,
|
||||
// just use NormalCleanupDestSlot (>= 1 for goto/return/..),
|
||||
// as 1st Arg to indicate abnormal termination
|
||||
if (!F.isForEHCleanup() && F.hasExitSwitch()) {
|
||||
Address Addr = CGF.getNormalCleanupDestSlot();
|
||||
llvm::Value *Load = CGF.Builder.CreateLoad(Addr, "cleanup.dest");
|
||||
llvm::Value *Zero = llvm::Constant::getNullValue(CGM.Int32Ty);
|
||||
IsForEH = CGF.Builder.CreateICmpNE(Load, Zero);
|
||||
}
|
||||
|
||||
Args.add(RValue::get(IsForEH), ArgTys[0]);
|
||||
Args.add(RValue::get(FP), ArgTys[1]);
|
||||
|
||||
|
|
|
@ -158,9 +158,10 @@ public:
|
|||
/// Generation flags.
|
||||
class Flags {
|
||||
enum {
|
||||
F_IsForEH = 0x1,
|
||||
F_IsForEH = 0x1,
|
||||
F_IsNormalCleanupKind = 0x2,
|
||||
F_IsEHCleanupKind = 0x4
|
||||
F_IsEHCleanupKind = 0x4,
|
||||
F_HasExitSwitch = 0x8,
|
||||
};
|
||||
unsigned flags;
|
||||
|
||||
|
@ -179,8 +180,10 @@ public:
|
|||
/// cleanup.
|
||||
bool isEHCleanupKind() const { return flags & F_IsEHCleanupKind; }
|
||||
void setIsEHCleanupKind() { flags |= F_IsEHCleanupKind; }
|
||||
};
|
||||
|
||||
bool hasExitSwitch() const { return flags & F_HasExitSwitch; }
|
||||
void setHasExitSwitch() { flags |= F_HasExitSwitch; }
|
||||
};
|
||||
|
||||
/// Emit the cleanup. For normal cleanups, this is run in the
|
||||
/// same EH context as when the cleanup was pushed, i.e. the
|
||||
|
|
|
@ -0,0 +1,30 @@
|
|||
// RUN: %clang_cc1 -triple x86_64-windows -fms-extensions -Wno-implicit-function-declaration -S -emit-llvm %s -o - | FileCheck %s
|
||||
|
||||
// CHECK: %[[src:[0-9-]+]] = call i8* @llvm.localaddress()
|
||||
// CHECK-NEXT: %cleanup.dest = load i32, i32* %cleanup.dest.slot, align 4
|
||||
// CHECK-NEXT: %[[src2:[0-9-]+]] = icmp ne i32 %cleanup.dest, 0
|
||||
// CHECK-NEXT: %[[src3:[0-9-]+]] = zext i1 %[[src2]] to i8
|
||||
// CHECK-NEXT: call void @"?fin$0@0@seh_abnormal_exits@@"(i8 %[[src3]], i8* %[[src]])
|
||||
|
||||
void seh_abnormal_exits(int *Counter) {
|
||||
for (int i = 0; i < 5; i++) {
|
||||
__try {
|
||||
if (i == 0)
|
||||
continue; // abnormal termination
|
||||
else if (i == 1)
|
||||
goto t10; // abnormal termination
|
||||
else if (i == 2)
|
||||
__leave; // normal execution
|
||||
else if (i == 4)
|
||||
return; // abnormal termination
|
||||
}
|
||||
__finally {
|
||||
if (AbnormalTermination()) {
|
||||
*Counter += 1;
|
||||
}
|
||||
}
|
||||
t10:;
|
||||
}
|
||||
return; // *Counter == 3
|
||||
}
|
||||
|
Loading…
Reference in New Issue