Fix SEH state numbering algorithm to handle cleanupendpads
[oota-llvm.git] / test / CodeGen / X86 / win-cleanuppad.ll
1 ; RUN: llc -mtriple=i686-pc-windows-msvc < %s | FileCheck --check-prefix=X86 %s
2 ; RUN: llc -mtriple=x86_64-pc-windows-msvc < %s | FileCheck --check-prefix=X64 %s
3
4 %struct.Dtor = type { i8 }
5
6 define void @simple_cleanup() #0 personality i8* bitcast (i32 (...)* @__CxxFrameHandler3 to i8*) {
7 entry:
8   %o = alloca %struct.Dtor, align 1
9   invoke void @f(i32 1)
10           to label %invoke.cont unwind label %ehcleanup
11
12 invoke.cont:                                      ; preds = %entry
13   call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(%struct.Dtor* %o) #2
14   ret void
15
16 ehcleanup:                                        ; preds = %entry
17   %0 = cleanuppad []
18   call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(%struct.Dtor* %o) #2
19   cleanupret %0 unwind to caller
20 }
21
22 declare void @f(i32) #0
23
24 declare i32 @__CxxFrameHandler3(...)
25
26 ; Function Attrs: nounwind
27 declare x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(%struct.Dtor*) #1
28
29 define void @nested_cleanup() #0 personality i8* bitcast (i32 (...)* @__CxxFrameHandler3 to i8*) {
30 entry:
31   %o1 = alloca %struct.Dtor, align 1
32   %o2 = alloca %struct.Dtor, align 1
33   invoke void @f(i32 1)
34           to label %invoke.cont unwind label %cleanup.outer
35
36 invoke.cont:                                      ; preds = %entry
37   invoke void @f(i32 2)
38           to label %invoke.cont.1 unwind label %cleanup.inner
39
40 invoke.cont.1:                                    ; preds = %invoke.cont
41   call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(%struct.Dtor* %o2) #2
42   invoke void @f(i32 3)
43           to label %invoke.cont.2 unwind label %cleanup.outer
44
45 invoke.cont.2:                                    ; preds = %invoke.cont.1
46   call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(%struct.Dtor* %o1) #2
47   ret void
48
49 cleanup.inner:                                        ; preds = %invoke.cont
50   %0 = cleanuppad []
51   call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(%struct.Dtor* %o2) #2
52   cleanupret %0 unwind label %cleanup.outer
53
54 cleanup.outer:                                      ; preds = %invoke.cont.1, %cleanup.inner, %entry
55   %1 = cleanuppad []
56   call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(%struct.Dtor* %o1) #2
57   cleanupret %1 unwind to caller
58 }
59
60 ; X86-LABEL: _nested_cleanup:
61 ; X86: movl    $1, (%esp)
62 ; X86: calll   _f
63 ; X86: movl    $2, (%esp)
64 ; X86: calll   _f
65 ; X86: movl    $3, (%esp)
66 ; X86: calll   _f
67
68 ; X86: LBB1_[[cleanup_inner:[0-9]+]]: # %cleanup.inner
69 ; X86: leal    {{.*}}(%ebp), %ecx
70 ; X86: calll   "??1Dtor@@QAE@XZ"
71 ; X86: retl # CLEANUPRET
72
73 ; X86: LBB1_[[cleanup_outer:[0-9]+]]: # %cleanup.outer
74 ; X86: leal    {{.*}}(%ebp), %ecx
75 ; X86: calll   "??1Dtor@@QAE@XZ"
76 ; X86: retl # CLEANUPRET
77
78 ; X86: L__ehtable$nested_cleanup:
79 ; X86:         .long   429065506
80 ; X86:         .long   2
81 ; X86:         .long   ($stateUnwindMap$nested_cleanup)
82 ; X86:         .long   0
83 ; X86:         .long   0
84 ; X86:         .long   0
85 ; X86:         .long   0
86 ; X86:         .long   0
87 ; X86:         .long   1
88 ; X86: $stateUnwindMap$nested_cleanup:
89 ; X86:         .long   -1
90 ; X86:         .long   LBB1_[[cleanup_outer]]
91 ; X86:         .long   0
92 ; X86:         .long   LBB1_[[cleanup_inner]]
93
94 ; X64-LABEL: nested_cleanup:
95 ; X64: movl    $1, %ecx
96 ; X64: callq   f
97 ; X64: movl    $2, %ecx
98 ; X64: callq   f
99 ; X64: movl    $3, %ecx
100 ; X64: callq   f
101
102 ; X64: .LBB1_[[cleanup_inner:[0-9]+]]: # %cleanup.inner
103 ; X64: leaq    {{.*}}(%rbp), %rcx
104 ; X64: callq   "??1Dtor@@QAE@XZ"
105 ; X64: retq # CLEANUPRET
106
107 ; X64: .LBB1_[[cleanup_outer:[0-9]+]]: # %cleanup.outer
108 ; X64: leaq    {{.*}}(%rbp), %rcx
109 ; X64: callq   "??1Dtor@@QAE@XZ"
110 ; X64: retq # CLEANUPRET
111
112 ; X64:        .seh_handlerdata
113 ; X64:        .long   ($cppxdata$nested_cleanup)@IMGREL
114 ; X64:        .align  4
115 ; X64:$cppxdata$nested_cleanup:
116 ; X64:        .long   429065506
117 ; X64:        .long   2
118 ; X64:        .long   ($stateUnwindMap$nested_cleanup)@IMGREL
119 ; X64:        .long   0
120 ; X64:        .long   0
121 ; X64:        .long   1
122 ; X64:        .long   ($ip2state$nested_cleanup)@IMGREL
123 ; X64:        .long   40
124 ; X64:        .long   0
125 ; X64:        .long   1
126 ; X64:$stateUnwindMap$nested_cleanup:
127 ; X64:        .long   -1
128 ; X64:        .long   .LBB1_[[cleanup_outer]]@IMGREL
129 ; X64:        .long   0
130 ; X64:        .long   .LBB1_[[cleanup_inner]]@IMGREL
131 ; FIXME: The ip2state table is totally wrong.
132 ; X64:$ip2state$nested_cleanup:
133 ; X64:        .long   .Lfunc_begin1@IMGREL
134 ; X64:        .long   -1
135
136 attributes #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" }
137 attributes #1 = { nounwind "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" }
138 attributes #2 = { nounwind }