Skip to content

Commit

Permalink
[WinEH] Fix problem where CodeGenPrepare incorrectly sinks a bitcast …
Browse files Browse the repository at this point in the history
…into an EH pad.

Differential Revision: http://reviews.llvm.org/D14842



git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@253902 91177308-0d34-0410-b5e6-96231b3b80d8
  • Loading branch information
Andrew Kaylor committed Nov 23, 2015
1 parent 5aafb6d commit e03f5e5
Show file tree
Hide file tree
Showing 2 changed files with 65 additions and 0 deletions.
6 changes: 6 additions & 0 deletions lib/CodeGen/CodeGenPrepare.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -730,6 +730,12 @@ static bool SinkCast(CastInst *CI) {
// Preincrement use iterator so we don't invalidate it.
++UI;

// If the block selected to receive the cast is an EH pad that does not
// allow non-PHI instructions before the terminator, we can't sink the
// cast.
if (UserBB->getTerminator()->isEHPad())
continue;

// If this user is in the same block as the cast, don't change the cast.
if (UserBB == DefBB) continue;

Expand Down
59 changes: 59 additions & 0 deletions test/Transforms/CodeGenPrepare/catchpad-phi-cast.ll
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
; RUN: opt -codegenprepare -S < %s | FileCheck %s

; The following target lines are needed for the test to exercise what it should.
; Without these lines, CodeGenPrepare does not try to sink the bitcasts.
target datalayout = "e-m:w-i64:64-f80:128-n8:16:32:64-S128"
target triple = "x86_64-pc-windows-msvc"

declare i32 @__CxxFrameHandler3(...)

declare void @f()

declare void @g(i8*)

; CodeGenPrepare will want to sink these bitcasts, but it selects the catchpad
; blocks as the place to which the bitcast should be sunk. Since catchpads
; do not allow non-phi instructions before the terminator, this isn't possible.

; CHECK-LABEL: @test(
define void @test(i32* %addr) personality i32 (...)* @__CxxFrameHandler3 {
; CHECK: entry:
; CHECK-NEXT: %x = getelementptr i32, i32* %addr, i32 1
; CHECK-NEXT: %p1 = bitcast i32* %x to i8*
entry:
%x = getelementptr i32, i32* %addr, i32 1
%p1 = bitcast i32* %x to i8*
invoke void @f()
to label %invoke.cont unwind label %catch1

; CHECK: invoke.cont:
; CHECK-NEXT: %y = getelementptr i32, i32* %addr, i32 2
; CHECK-NEXT: %p2 = bitcast i32* %y to i8*
invoke.cont:
%y = getelementptr i32, i32* %addr, i32 2
%p2 = bitcast i32* %y to i8*
invoke void @f()
to label %done unwind label %catch2

done:
ret void

catch1:
%cp1 = catchpad [] to label %catch.dispatch unwind label %catchend1

catch2:
%cp2 = catchpad [] to label %catch.dispatch unwind label %catchend2

; CHECK: catch.dispatch:
; CHECK-NEXT: %p = phi i8* [ %p1, %catch1 ], [ %p2, %catch2 ]
catch.dispatch:
%p = phi i8* [ %p1, %catch1 ], [ %p2, %catch2 ]
call void @g(i8* %p)
unreachable

catchend1:
catchendpad unwind to caller

catchend2:
catchendpad unwind to caller
}

0 comments on commit e03f5e5

Please sign in to comment.