Files
clang-p2996/llvm/test/Transforms/InstCombine/call_nonnull_arg.ll
Juneyoung Lee 4479c0c2c0 Allow nonnull/align attribute to accept poison
Currently LLVM is relying on ValueTracking's `isKnownNonZero` to attach `nonnull`, which can return true when the value is poison.
To make the semantics of `nonnull` consistent with the behavior of `isKnownNonZero`, this makes the semantics of `nonnull` to accept poison, and return poison if the input pointer isn't null.
This makes many transformations like below legal:

```
%p = gep inbounds %x, 1 ; % p is non-null pointer or poison
call void @f(%p)        ; instcombine converts this to call void @f(nonnull %p)
```

Instead, this semantics makes propagation of `nonnull` to caller illegal.
The reason is that, passing poison to `nonnull` does not immediately raise UB anymore, so such program is still well defined, if the callee does not use the argument.
Having `noundef` attribute there re-allows this.

```
define void @f(i8* %p) {       ; functionattr cannot mark %p nonnull here anymore
  call void @g(i8* nonnull %p) ; .. because @g never raises UB if it never uses %p.
  ret void
}
```

Another attribute that needs to be updated is `align`. This patch updates the semantics of align to accept poison as well.

Reviewed By: jdoerfert

Differential Revision: https://reviews.llvm.org/D90529
2021-01-20 11:31:23 +09:00

64 lines
1.9 KiB
LLVM

; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt < %s -instcombine -S | FileCheck %s
; InstCombine should mark null-checked argument as nonnull at callsite
declare void @dummy(i32*, i32)
define void @test(i32* %a, i32 %b) {
; CHECK-LABEL: @test(
; CHECK-NEXT: entry:
; CHECK-NEXT: [[COND1:%.*]] = icmp eq i32* [[A:%.*]], null
; CHECK-NEXT: br i1 [[COND1]], label [[DEAD:%.*]], label [[NOT_NULL:%.*]]
; CHECK: not_null:
; CHECK-NEXT: [[COND2:%.*]] = icmp eq i32 [[B:%.*]], 0
; CHECK-NEXT: br i1 [[COND2]], label [[DEAD]], label [[NOT_ZERO:%.*]]
; CHECK: not_zero:
; CHECK-NEXT: call void @dummy(i32* nonnull [[A]], i32 [[B]])
; CHECK-NEXT: ret void
; CHECK: dead:
; CHECK-NEXT: unreachable
;
entry:
%cond1 = icmp eq i32* %a, null
br i1 %cond1, label %dead, label %not_null
not_null:
%cond2 = icmp eq i32 %b, 0
br i1 %cond2, label %dead, label %not_zero
not_zero:
call void @dummy(i32* %a, i32 %b)
ret void
dead:
unreachable
}
; The nonnull attribute in the 'bar' declaration is
; propagated to the parameters of the 'baz' callsite.
declare void @bar(i8*, i8* nonnull noundef)
declare void @bar_without_noundef(i8*, i8* nonnull)
declare void @baz(i8*, i8*)
define void @deduce_nonnull_from_another_call(i8* %a, i8* %b) {
; CHECK-LABEL: @deduce_nonnull_from_another_call(
; CHECK-NEXT: call void @bar(i8* [[A:%.*]], i8* [[B:%.*]])
; CHECK-NEXT: call void @baz(i8* nonnull [[B]], i8* nonnull [[B]])
; CHECK-NEXT: ret void
;
call void @bar(i8* %a, i8* %b)
call void @baz(i8* %b, i8* %b)
ret void
}
define void @deduce_nonnull_from_another_call2(i8* %a, i8* %b) {
; CHECK-LABEL: @deduce_nonnull_from_another_call2(
; CHECK-NEXT: call void @bar_without_noundef(i8* [[A:%.*]], i8* [[B:%.*]])
; CHECK-NEXT: call void @baz(i8* [[B]], i8* [[B]])
; CHECK-NEXT: ret void
;
call void @bar_without_noundef(i8* %a, i8* %b)
call void @baz(i8* %b, i8* %b)
ret void
}