llvm/llvm/test/Analysis/BasicAA/phi-and-select.ll

; RUN: opt < %s -aa-pipeline=basic-aa -passes=aa-eval -print-all-alias-modref-info -disable-output 2>&1 | FileCheck %s

; BasicAA should detect NoAliases in PHIs and Selects.

; Two PHIs in the same block.
; CHECK-LABEL: Function: foo
; CHECK: NoAlias: double* %a, double* %b
define void @foo(i1 %m, ptr noalias %x, ptr noalias %y) {
entry:
  br i1 %m, label %true, label %false

true:
  br label %exit

false:
  br label %exit

exit:
  %a = phi ptr [ %x, %true ], [ %y, %false ]
  %b = phi ptr [ %x, %false ], [ %y, %true ]
  store volatile double 0.0, ptr %a
  store volatile double 1.0, ptr %b
  ret void
}

; Two selects with the same condition.
; CHECK-LABEL: Function: bar
; CHECK: NoAlias: double* %a, double* %b
define void @bar(i1 %m, ptr noalias %x, ptr noalias %y) {
entry:
  %a = select i1 %m, ptr %x, ptr %y
  %b = select i1 %m, ptr %y, ptr %x
  store volatile double 0.000000e+00, ptr %a
  store volatile double 1.000000e+00, ptr %b
  ret void
}

; Two PHIs with disjoint sets of inputs.
; CHECK-LABEL: Function: qux
; CHECK: NoAlias: double* %a, double* %b
define void @qux(i1 %m, ptr noalias %x, ptr noalias %y,
                 i1 %n, ptr noalias %v, ptr noalias %w) {
entry:
  br i1 %m, label %true, label %false

true:
  br label %exit

false:
  br label %exit

exit:
  %a = phi ptr [ %x, %true ], [ %y, %false ]
  br i1 %n, label %ntrue, label %nfalse

ntrue:
  br label %nexit

nfalse:
  br label %nexit

nexit:
  %b = phi ptr [ %v, %ntrue ], [ %w, %nfalse ]
  store volatile double 0.0, ptr %a
  store volatile double 1.0, ptr %b
  ret void
}

; Two selects with disjoint sets of arms.
; CHECK-LABEL: Function: fin
; CHECK: NoAlias: double* %a, double* %b
define void @fin(i1 %m, ptr noalias %x, ptr noalias %y,
                 i1 %n, ptr noalias %v, ptr noalias %w) {
entry:
  %a = select i1 %m, ptr %x, ptr %y
  %b = select i1 %n, ptr %v, ptr %w
  store volatile double 0.000000e+00, ptr %a
  store volatile double 1.000000e+00, ptr %b
  ret void
}

; On the first iteration, sel1 = a1, sel2 = a2, phi = a3
; On the second iteration, sel1 = a2, sel1 = a1, phi = a2
; As such, sel1 and phi may alias.
; CHECK-LABEL: Function: select_backedge
; CHECK: NoAlias:	i32* %sel1, i32* %sel2
; CHECK: MayAlias:	i32* %phi, i32* %sel1
; CHECK: MayAlias:	i32* %phi, i32* %sel2
define void @select_backedge() {
entry:
  %a1 = alloca i32
  %a2 = alloca i32
  %a3 = alloca i32
  br label %loop

loop:
  %phi = phi ptr [ %a3, %entry ], [ %sel2, %loop ]
  %c = phi i1 [ true, %entry ], [ false, %loop ]
  %sel1 = select i1 %c, ptr %a1, ptr %a2
  %sel2 = select i1 %c, ptr %a2, ptr %a1
  load i32, ptr %sel1
  load i32, ptr %sel2
  load i32, ptr %phi
  br label %loop
}