Compiler projects using llvm
; RUN: llc -verify-machineinstrs -mtriple=i686-pc-windows-msvc < %s | FileCheck --check-prefix=X86 %s
; RUN: llc -verify-machineinstrs -mtriple=x86_64-pc-windows-msvc < %s | FileCheck --check-prefix=X64 %s

%struct.Dtor = type { i8 }

define void @simple_cleanup() #0 personality ptr @__CxxFrameHandler3 {
entry:
  %o = alloca %struct.Dtor, align 1
  invoke void @f(i32 1)
          to label %invoke.cont unwind label %ehcleanup

invoke.cont:                                      ; preds = %entry
  call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(ptr %o) #2
  ret void

ehcleanup:                                        ; preds = %entry
  %0 = cleanuppad within none []
  call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(ptr %o) #2 [ "funclet"(token %0) ]
  cleanupret from %0 unwind to caller
}

; CHECK: simple_cleanup:                         # @simple_cleanup
; CHECK:         pushq   %rbp
; CHECK:         subq    $48, %rsp
; CHECK:         leaq    48(%rsp), %rbp
; CHECK:         movq    $-2, -8(%rbp)
; CHECK:         movl    $1, %ecx
; CHECK:         callq   f
; CHECK:         callq   "??1Dtor@@QAE@XZ"
; CHECK:         nop
; CHECK:         addq    $48, %rsp
; CHECK:         popq    %rbp
; CHECK:         retq

; CHECK: "?dtor$2@?0?simple_cleanup@4HA":
; CHECK:         callq   "??1Dtor@@QAE@XZ"
; CHECK:         retq

; CHECK: $cppxdata$simple_cleanup:
; CHECK-NEXT:         .long   429065506
; CHECK-NEXT:         .long   1
; CHECK-NEXT:         .long   ($stateUnwindMap$simple_cleanup)@IMGREL
; CHECK-NEXT:         .long   0
; CHECK-NEXT:         .long   0
; CHECK-NEXT:         .long   3
; CHECK-NEXT:         .long   ($ip2state$simple_cleanup)@IMGREL
; UnwindHelp offset should match the -2 store above
; CHECK-NEXT:         .long   40
; CHECK-NEXT:         .long   0
; CHECK-NEXT:         .long   1

declare void @f(i32) #0

declare i32 @__CxxFrameHandler3(...)

; Function Attrs: nounwind
declare x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(ptr) #1

define void @nested_cleanup() #0 personality ptr @__CxxFrameHandler3 {
entry:
  %o1 = alloca %struct.Dtor, align 1
  %o2 = alloca %struct.Dtor, align 1
  invoke void @f(i32 1)
          to label %invoke.cont unwind label %cleanup.outer

invoke.cont:                                      ; preds = %entry
  invoke void @f(i32 2)
          to label %invoke.cont.1 unwind label %cleanup.inner

invoke.cont.1:                                    ; preds = %invoke.cont
  call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(ptr %o2) #2
  invoke void @f(i32 3)
          to label %invoke.cont.2 unwind label %cleanup.outer

invoke.cont.2:                                    ; preds = %invoke.cont.1
  call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(ptr %o1) #2
  ret void

cleanup.inner:                                        ; preds = %invoke.cont
  %0 = cleanuppad within none []
  call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(ptr %o2) #2 [ "funclet"(token %0) ]
  cleanupret from %0 unwind label %cleanup.outer

cleanup.outer:                                      ; preds = %invoke.cont.1, %cleanup.inner, %entry
  %1 = cleanuppad within none []
  call x86_thiscallcc void @"\01??1Dtor@@QAE@XZ"(ptr %o1) #2 [ "funclet"(token %1) ]
  cleanupret from %1 unwind to caller
}

; X86-LABEL: _nested_cleanup:
; X86: pushl   $1
; X86: calll   _f
; X86: pushl   $2
; X86: calll   _f
; X86: pushl   $3
; X86: calll   _f

; X86: "?dtor$[[cleanup_inner:[0-9]+]]@?0?nested_cleanup@4HA":
; X86: LBB1_[[cleanup_inner]]: # %cleanup.inner{{$}}
; X86: pushl %ebp
; X86: leal    {{.*}}(%ebp), %ecx
; X86: calll   "??1Dtor@@QAE@XZ"
; X86: popl %ebp
; X86: retl

; X86: "?dtor$[[cleanup_outer:[0-9]+]]@?0?nested_cleanup@4HA":
; X86: LBB1_[[cleanup_outer]]: # %cleanup.outer{{$}}
; X86: pushl %ebp
; X86: leal    {{.*}}(%ebp), %ecx
; X86: calll   "??1Dtor@@QAE@XZ"
; X86: popl %ebp
; X86: retl

; X86: L__ehtable$nested_cleanup:
; X86:         .long   429065506
; X86:         .long   2
; X86:         .long   ($stateUnwindMap$nested_cleanup)
; X86:         .long   0
; X86:         .long   0
; X86:         .long   0
; X86:         .long   0
; X86:         .long   0
; X86:         .long   1
; X86: $stateUnwindMap$nested_cleanup:
; X86:         .long   -1
; X86:         .long   "?dtor$[[cleanup_outer]]@?0?nested_cleanup@4HA"
; X86:         .long   0
; X86:         .long   "?dtor$[[cleanup_inner]]@?0?nested_cleanup@4HA"

; X64-LABEL: nested_cleanup:
; X64: .Lfunc_begin1:
; X64: .Ltmp2:
; X64: movl    $1, %ecx
; X64: callq   f
; X64: .Ltmp4:
; X64: movl    $2, %ecx
; X64: callq   f
; X64: .Ltmp5:
; X64: callq   "??1Dtor@@QAE@XZ"
; X64: .Ltmp6:
; X64: movl    $3, %ecx
; X64: callq   f
; X64: .Ltmp7:

; X64: "?dtor$[[cleanup_inner:[0-9]+]]@?0?nested_cleanup@4HA":
; X64: LBB1_[[cleanup_inner]]: # %cleanup.inner{{$}}
; X64: pushq %rbp
; X64: leaq    {{.*}}(%rbp), %rcx
; X64: callq   "??1Dtor@@QAE@XZ"
; X64: popq %rbp
; X64: retq

; X64:        .seh_handlerdata
; X64:        .text
; X64:        .seh_endproc

; X64: "?dtor$[[cleanup_outer:[0-9]+]]@?0?nested_cleanup@4HA":
; X64: LBB1_[[cleanup_outer]]: # %cleanup.outer{{$}}
; X64: pushq %rbp
; X64: leaq    {{.*}}(%rbp), %rcx
; X64: callq   "??1Dtor@@QAE@XZ"
; X64: popq %rbp
; X64: retq

; X64:        .section .xdata,"dr"
; X64-NEXT: .p2align  2
; X64: $cppxdata$nested_cleanup:
; X64-NEXT: .long   429065506
; X64-NEXT: .long   2
; X64-NEXT: .long   ($stateUnwindMap$nested_cleanup)@IMGREL
; X64-NEXT: .long   0
; X64-NEXT: .long   0
; X64-NEXT: .long   5
; X64-NEXT: .long   ($ip2state$nested_cleanup)@IMGREL
; X64-NEXT: .long   56
; X64-NEXT: .long   0
; X64-NEXT: .long   1

; X64: $stateUnwindMap$nested_cleanup:
; X64-NEXT: .long   -1
; X64-NEXT: .long   "?dtor$[[cleanup_outer]]@?0?nested_cleanup@4HA"@IMGREL
; X64-NEXT: .long   0
; X64-NEXT: .long   "?dtor$[[cleanup_inner]]@?0?nested_cleanup@4HA"@IMGREL

; X64: $ip2state$nested_cleanup:
; X64-NEXT: .long   .Lfunc_begin1@IMGREL
; X64-NEXT: .long   -1
; X64-NEXT: .long   .Ltmp2@IMGREL
; X64-NEXT: .long   0
; X64-NEXT: .long   .Ltmp4@IMGREL
; X64-NEXT: .long   1
; X64-NEXT: .long   .Ltmp6@IMGREL
; X64-NEXT: .long   0
; X64-NEXT: .long   .Ltmp7@IMGREL+1
; X64-NEXT: .long   -1

attributes #0 = { "disable-tail-calls"="false" "less-precise-fpmad"="false" "frame-pointer"="none" "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" }
attributes #1 = { nounwind "disable-tail-calls"="false" "less-precise-fpmad"="false" "frame-pointer"="none" "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" }
attributes #2 = { nounwind }