124 lines
2.8 KiB
LLVM
124 lines
2.8 KiB
LLVM
; RUN: opt < %s -jump-threading -S | FileCheck %s
|
|
|
|
target datalayout = "e-p:32:32:32-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:32:64-f32:32:32-f64:32:64-v64:64:64-v128:128:128-a0:0:64-f80:128:128"
|
|
target triple = "i386-apple-darwin7"
|
|
|
|
declare void @use(i32 *)
|
|
|
|
; Check that we propagate nonnull to dominated loads, when we find an available
|
|
; loaded value.
|
|
; CHECK-LABEL: @test1(
|
|
; CHECK-LABEL: ret1:
|
|
; CHECK-NEXT: %[[p1:.*]] = load i32*, i32** %ptr
|
|
; CHECK-NOT: !nonnull
|
|
; CHECK-NEXT: store i32 1, i32* %[[p1]]
|
|
; CHECK-NEXT: tail call void @use(i32* null)
|
|
; CHECK-NEXT: ret void
|
|
|
|
; CHECK-LABEL: ret2:
|
|
; CHECK-NEXT: %[[p2:.*]] = load i32*, i32** %ptr, align 4, !nonnull !0
|
|
; CHECK: tail call void @use(i32* %[[p2]])
|
|
; CHECK-NEXT: ret void
|
|
define void @test1(i32** %ptr, i1 %c) {
|
|
br i1 %c, label %d1, label %d2
|
|
|
|
d1:
|
|
%p1 = load i32*, i32** %ptr, !nonnull !0
|
|
br label %d3
|
|
|
|
d2:
|
|
br label %d3
|
|
|
|
d3:
|
|
%pm = phi i32* [ null, %d2 ], [ %p1, %d1 ]
|
|
%p2 = load i32*, i32** %ptr
|
|
store i32 1, i32* %p2
|
|
%c2 = icmp eq i32* %pm, null
|
|
br i1 %c2, label %ret1, label %ret2
|
|
|
|
ret1:
|
|
tail call void @use(i32* %pm) nounwind
|
|
ret void
|
|
|
|
ret2:
|
|
tail call void @use(i32* %pm) nounwind
|
|
ret void
|
|
}
|
|
|
|
; Check that we propagate nonnull to dominated loads, when we find an available
|
|
; loaded value.
|
|
; CHECK-LABEL: @test2(
|
|
; CHECK-LABEL: d3.thread:
|
|
; CHECK-NEXT: %[[p1:.*]] = load i32*, i32** %ptr, align 4, !nonnull !0
|
|
; CHECK-NEXT: store i32 1, i32* %[[p1]]
|
|
; CHECK-NEXT: br label %ret1
|
|
|
|
; CHECK-LABEL: d3:
|
|
; CHECK-NEXT: %[[p_cmp:.*]] = load i32*, i32** %ptr
|
|
; CHECK-NEXT: %[[p2:.*]] = load i32*, i32** %ptr, align 4, !nonnull !0
|
|
; CHECK-NEXT: store i32 1, i32* %[[p2]]
|
|
; CHECK-NEXT: icmp eq i32* %[[p_cmp]], null
|
|
define void @test2(i32** %ptr, i1 %c) {
|
|
br i1 %c, label %d1, label %d2
|
|
|
|
d1:
|
|
%p1 = load i32*, i32** %ptr
|
|
br label %d3
|
|
|
|
d2:
|
|
br label %d3
|
|
|
|
d3:
|
|
%pm = phi i32* [ null, %d2 ], [ %p1, %d1 ]
|
|
%p2 = load i32*, i32** %ptr, !nonnull !0
|
|
store i32 1, i32* %p2
|
|
%c2 = icmp eq i32* %pm, null
|
|
br i1 %c2, label %ret1, label %ret2
|
|
|
|
ret1:
|
|
tail call void @use(i32* %pm) nounwind
|
|
ret void
|
|
|
|
ret2:
|
|
tail call void @use(i32* %pm) nounwind
|
|
ret void
|
|
}
|
|
|
|
; Check that we do not propagate nonnull to loads predecessors that are combined
|
|
; to a PHI node.
|
|
; CHECK-LABEL: @test3(
|
|
; CHECK-LABEL: d1:
|
|
; CHECK-NEXT: %[[p1:.*]] = load i32*, i32** %ptr
|
|
; CHECK-NOT: !nonnull
|
|
|
|
; CHECK-LABEL: d2:
|
|
; CHECK-NEXT: %[[p2:.*]] = load i32*, i32** %ptr
|
|
; CHECK-NOT: !nonnull
|
|
|
|
; CHECK-LABEL: d3:
|
|
; CHECK-NEXT: phi i32* [ %[[p2]], %d2 ], [ %[[p1]], %d1 ]
|
|
define void @test3(i32** %ptr) {
|
|
d1:
|
|
%x = load i32*, i32** %ptr, !nonnull !0
|
|
br label %d3
|
|
|
|
d2:
|
|
br label %d3
|
|
|
|
d3:
|
|
%y = load i32*, i32** %ptr
|
|
store i32 1, i32* %y
|
|
%c2 = icmp eq i32* %y, @p
|
|
br i1 %c2, label %ret1, label %ret2
|
|
|
|
ret1:
|
|
ret void
|
|
|
|
ret2:
|
|
ret void
|
|
}
|
|
|
|
@p = external global i32
|
|
|
|
!0 = !{}
|