1; RUN: opt -mtriple=i686-pc-windows-msvc -S -x86-winehstate < %s | FileCheck %s 2 3target datalayout = "e-m:x-p:32:32-i64:64-f80:32-n8:16:32-a:0:32-S32" 4target triple = "i686-pc-windows-msvc" 5 6%rtti.TypeDescriptor2 = type { i8**, i8*, [3 x i8] } 7%eh.CatchableType = type { i32, i8*, i32, i32, i32, i32, i8* } 8%eh.CatchableTypeArray.1 = type { i32, [1 x %eh.CatchableType*] } 9%eh.ThrowInfo = type { i32, i8*, i8*, i8* } 10 11$"\01??_R0H@8" = comdat any 12 13$"_CT??_R0H@84" = comdat any 14 15$_CTA1H = comdat any 16 17$_TI1H = comdat any 18 19@"\01??_7type_info@@6B@" = external constant i8* 20@"\01??_R0H@8" = linkonce_odr global %rtti.TypeDescriptor2 { i8** @"\01??_7type_info@@6B@", i8* null, [3 x i8] c".H\00" }, comdat 21@"_CT??_R0H@84" = linkonce_odr unnamed_addr constant %eh.CatchableType { i32 1, i8* bitcast (%rtti.TypeDescriptor2* @"\01??_R0H@8" to i8*), i32 0, i32 -1, i32 0, i32 4, i8* null }, section ".xdata", comdat 22@_CTA1H = linkonce_odr unnamed_addr constant %eh.CatchableTypeArray.1 { i32 1, [1 x %eh.CatchableType*] [%eh.CatchableType* @"_CT??_R0H@84"] }, section ".xdata", comdat 23@_TI1H = linkonce_odr unnamed_addr constant %eh.ThrowInfo { i32 0, i8* null, i8* null, i8* bitcast (%eh.CatchableTypeArray.1* @_CTA1H to i8*) }, section ".xdata", comdat 24 25define i32 @main() #0 personality i32 (...)* @__CxxFrameHandler3 { 26entry: 27 %tmp = alloca i32, align 4 28 ; CHECK: entry: 29 ; CHECK: store i32 -1 30 ; CHECK: call void @g(i32 3) 31 ; CHECK-NEXT: call void @g(i32 4) 32 ; CHECK-NEXT: call void @g(i32 5) 33 call void @g(i32 3) 34 call void @g(i32 4) 35 call void @g(i32 5) 36 store i32 0, i32* %tmp, align 4 37 %0 = bitcast i32* %tmp to i8* 38 ; CHECK: store i32 0 39 ; CHECK: invoke void @_CxxThrowException( 40 invoke void @_CxxThrowException(i8* %0, %eh.ThrowInfo* nonnull @_TI1H) #1 41 to label %unreachable.for.entry unwind label %catch.dispatch 42 43catch.dispatch: ; preds = %entry 44 %cs1 = catchswitch within none [label %catch] unwind to caller 45 46catch: ; preds = %catch.dispatch 47 %1 = catchpad within %cs1 [i8* null, i32 u0x40, i8* null] 48 ; CHECK: catch: 49 ; CHECK: store i32 2 50 ; CHECK: invoke void @_CxxThrowException( 51 invoke void @_CxxThrowException(i8* null, %eh.ThrowInfo* null) [ "funclet"(token %1) ] 52 to label %unreachable unwind label %catch.dispatch.1 53 54catch.dispatch.1: ; preds = %catch 55 %cs2 = catchswitch within %1 [label %catch.3] unwind to caller 56catch.3: ; preds = %catch.dispatch.1 57 %2 = catchpad within %cs2 [i8* null, i32 u0x40, i8* null] 58 ; CHECK: catch.3: 59 ; CHECK: store i32 3 60 ; CHECK: call void @g(i32 1) 61 ; CHECK-NEXT: call void @g(i32 2) 62 ; CHECK-NEXT: call void @g(i32 3) 63 call void @g(i32 1) 64 call void @g(i32 2) 65 call void @g(i32 3) 66 catchret from %2 to label %try.cont 67 68try.cont: ; preds = %catch.3 69 ; CHECK: try.cont: 70 ; CHECK: store i32 1 71 ; CHECK: call void @g(i32 2) 72 ; CHECK-NEXT: call void @g(i32 3) 73 ; CHECK-NEXT: call void @g(i32 4) 74 call void @g(i32 2) 75 call void @g(i32 3) 76 call void @g(i32 4) 77 unreachable 78 79unreachable: ; preds = %catch 80 unreachable 81 82unreachable.for.entry: ; preds = %entry 83 unreachable 84} 85 86define i32 @nopads() #0 personality i32 (...)* @__CxxFrameHandler3 { 87 ret i32 0 88} 89 90; CHECK-LABEL: define i32 @nopads() 91; CHECK-NEXT: ret i32 0 92; CHECK-NOT: __ehhandler$nopads 93 94; CHECK-LABEL: define void @PR25926() 95define void @PR25926() personality i32 (...)* @__CxxFrameHandler3 { 96entry: 97 ; CHECK: entry: 98 ; CHECK: store i32 -1 99 ; CHECK: store i32 0 100 ; CHECK: invoke void @_CxxThrowException( 101 invoke void @_CxxThrowException(i8* null, %eh.ThrowInfo* null) 102 to label %unreachable unwind label %catch.dispatch 103 104catch.dispatch: ; preds = %entry 105 %0 = catchswitch within none [label %catch] unwind to caller 106 107catch: ; preds = %catch.dispatch 108 %1 = catchpad within %0 [i8* null, i32 64, i8* null] 109 ; CHECK: catch: 110 ; CHECK: store i32 3 111 ; CHECK: invoke void @_CxxThrowException( 112 invoke void @_CxxThrowException(i8* null, %eh.ThrowInfo* null) [ "funclet"(token %1) ] 113 to label %unreachable1 unwind label %catch.dispatch1 114 115catch.dispatch1: ; preds = %catch 116 %2 = catchswitch within %1 [label %catch2] unwind label %ehcleanup 117 118catch2: ; preds = %catch.dispatch1 119 %3 = catchpad within %2 [i8* null, i32 64, i8* null] 120 catchret from %3 to label %try.cont 121 122try.cont: ; preds = %catch2 123 ; CHECK: try.cont: 124 ; CHECK: store i32 1 125 ; CHECK: call void @dtor() 126 ; CHECK-NEXT: call void @dtor() 127 ; CHECK-NEXT: call void @dtor() 128 call void @dtor() #3 [ "funclet"(token %1) ] 129 call void @dtor() #3 [ "funclet"(token %1) ] 130 call void @dtor() #3 [ "funclet"(token %1) ] 131 catchret from %1 to label %try.cont4 132 133try.cont4: ; preds = %try.cont 134 ret void 135 136ehcleanup: ; preds = %catch.dispatch1 137 %4 = cleanuppad within %1 [] 138 ; CHECK: ehcleanup: 139 ; CHECK: call void @dtor() 140 call void @dtor() #3 [ "funclet"(token %4) ] 141 cleanupret from %4 unwind to caller 142 143unreachable: ; preds = %entry 144 unreachable 145 146unreachable1: ; preds = %catch 147 unreachable 148} 149 150; CHECK-LABEL: define void @required_state_store( 151define void @required_state_store(i1 zeroext %cond) personality i32 (...)* @_except_handler3 { 152entry: 153 %__exception_code = alloca i32, align 4 154 call void (...) @llvm.localescape(i32* nonnull %__exception_code) 155; CHECK: store i32 -1 156; CHECK: call void @g(i32 0) 157 call void @g(i32 0) 158 br i1 %cond, label %if.then, label %if.end 159 160if.then: ; preds = %entry 161; CHECK: store i32 0 162; CHECK-NEXT: invoke void @g(i32 1) 163 invoke void @g(i32 1) 164 to label %if.end unwind label %catch.dispatch 165 166catch.dispatch: ; preds = %if.then 167 %0 = catchswitch within none [label %__except.ret] unwind to caller 168 169__except.ret: ; preds = %catch.dispatch 170 %1 = catchpad within %0 [i8* bitcast (i32 ()* @"\01?filt$0@0@required_state_store@@" to i8*)] 171 catchret from %1 to label %if.end 172 173if.end: ; preds = %if.then, %__except.ret, %entry 174; CHECK: store i32 -1 175; CHECK-NEXT: call void @dtor() 176 call void @dtor() 177 ret void 178} 179 180define internal i32 @"\01?filt$0@0@required_state_store@@"() { 181entry: 182 %0 = tail call i8* @llvm.frameaddress(i32 1) 183 %1 = tail call i8* @llvm.x86.seh.recoverfp(i8* bitcast (void (i1)* @required_state_store to i8*), i8* %0) 184 %2 = tail call i8* @llvm.localrecover(i8* bitcast (void (i1)* @required_state_store to i8*), i8* %1, i32 0) 185 %__exception_code = bitcast i8* %2 to i32* 186 %3 = getelementptr inbounds i8, i8* %0, i32 -20 187 %4 = bitcast i8* %3 to { i32*, i8* }** 188 %5 = load { i32*, i8* }*, { i32*, i8* }** %4, align 4 189 %6 = getelementptr inbounds { i32*, i8* }, { i32*, i8* }* %5, i32 0, i32 0 190 %7 = load i32*, i32** %6, align 4 191 %8 = load i32, i32* %7, align 4 192 store i32 %8, i32* %__exception_code, align 4 193 ret i32 1 194} 195 196declare void @g(i32) #0 197 198declare void @dtor() 199 200declare x86_stdcallcc void @_CxxThrowException(i8*, %eh.ThrowInfo*) 201 202declare i32 @__CxxFrameHandler3(...) 203 204declare i8* @llvm.frameaddress(i32) 205 206declare i8* @llvm.x86.seh.recoverfp(i8*, i8*) 207 208declare i8* @llvm.localrecover(i8*, i8*, i32) 209 210declare void @llvm.localescape(...) 211 212declare i32 @_except_handler3(...) 213 214attributes #0 = { "disable-tail-calls"="false" "less-precise-fpmad"="false" "no-frame-pointer-elim"="false" "no-infs-fp-math"="false" "no-nans-fp-math"="false" "no-realign-stack" "stack-protector-buffer-size"="8" "unsafe-fp-math"="false" "use-soft-float"="false" } 215attributes #1 = { noreturn } 216 217!llvm.ident = !{!0} 218 219!0 = !{!"clang version 3.8.0 (trunk 245153) (llvm/trunk 245238)"} 220