mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-23 03:02:36 +01:00
9f8be8e69b
This also changes -lint from an analysis to a pass. It's similar to -verify, and that is a normal pass, and lives in llvm/IR. Reviewed By: ychen Differential Revision: https://reviews.llvm.org/D87057
204 lines
6.5 KiB
LLVM
204 lines
6.5 KiB
LLVM
; RUN: opt -basic-aa -lint -disable-output < %s 2>&1 | FileCheck %s
|
|
; RUN: opt -aa-pipeline=basic-aa -passes=lint -disable-output < %s 2>&1 | FileCheck %s
|
|
target datalayout = "e-p:64:64:64"
|
|
|
|
declare fastcc void @bar()
|
|
declare void @llvm.stackrestore(i8*)
|
|
declare void @llvm.memcpy.p0i8.p0i8.i64(i8* nocapture, i8* nocapture, i64, i1) nounwind
|
|
declare void @llvm.memcpy.inline.p0i8.p0i8.i64(i8* nocapture, i8* nocapture, i64, i1) nounwind
|
|
declare void @has_sret(i8* sret %p)
|
|
declare void @has_noaliases(i32* noalias %p, i32* %q)
|
|
declare void @one_arg(i32)
|
|
|
|
@CG = constant i32 7
|
|
@CG2 = constant i32 7
|
|
@E = external global i8
|
|
|
|
define i32 @foo() noreturn {
|
|
%buf = alloca i8
|
|
%buf2 = alloca {i8, i8}, align 2
|
|
; CHECK: Caller and callee calling convention differ
|
|
call void @bar()
|
|
; CHECK: Null pointer dereference
|
|
store i32 0, i32* null
|
|
; CHECK: Null pointer dereference
|
|
%t = load i32, i32* null
|
|
; CHECK: Undef pointer dereference
|
|
store i32 0, i32* undef
|
|
; CHECK: Undef pointer dereference
|
|
%u = load i32, i32* undef
|
|
; CHECK: All-ones pointer dereference
|
|
store i32 0, i32* inttoptr (i64 -1 to i32*)
|
|
; CHECK: Address one pointer dereference
|
|
store i32 0, i32* inttoptr (i64 1 to i32*)
|
|
; CHECK: Memory reference address is misaligned
|
|
store i8 0, i8* %buf, align 2
|
|
; CHECK: Memory reference address is misaligned
|
|
%gep = getelementptr {i8, i8}, {i8, i8}* %buf2, i32 0, i32 1
|
|
store i8 0, i8* %gep, align 2
|
|
; CHECK: Division by zero
|
|
%sd = sdiv i32 2, 0
|
|
; CHECK: Division by zero
|
|
%ud = udiv i32 2, 0
|
|
; CHECK: Division by zero
|
|
%sr = srem i32 2, 0
|
|
; CHECK: Division by zero
|
|
%ur = urem i32 2, 0
|
|
; CHECK: extractelement index out of range
|
|
%ee = extractelement <4 x i32> zeroinitializer, i32 4
|
|
; CHECK: insertelement index out of range
|
|
%ie = insertelement <4 x i32> zeroinitializer, i32 0, i32 4
|
|
; CHECK: Shift count out of range
|
|
%r = lshr i32 0, 32
|
|
; CHECK: Shift count out of range
|
|
%q = ashr i32 0, 32
|
|
; CHECK: Shift count out of range
|
|
%l = shl i32 0, 32
|
|
; CHECK: xor(undef, undef)
|
|
%xx = xor i32 undef, undef
|
|
; CHECK: sub(undef, undef)
|
|
%xs = sub i32 undef, undef
|
|
|
|
; CHECK: Write to read-only memory
|
|
store i32 8, i32* @CG
|
|
; CHECK: Write to text section
|
|
store i32 8, i32* bitcast (i32()* @foo to i32*)
|
|
; CHECK: Load from block address
|
|
%lb = load i32, i32* bitcast (i8* blockaddress(@foo, %next) to i32*)
|
|
; CHECK: Call to block address
|
|
call void() bitcast (i8* blockaddress(@foo, %next) to void()*)()
|
|
; CHECK: Undefined behavior: Null pointer dereference
|
|
call void @llvm.stackrestore(i8* null)
|
|
; CHECK: Undefined behavior: Null pointer dereference
|
|
call void @has_sret(i8* null)
|
|
; CHECK: Unusual: noalias argument aliases another argument
|
|
call void @has_noaliases(i32* @CG, i32* @CG)
|
|
; CHECK: Call argument count mismatches callee argument count
|
|
call void (i32, i32) bitcast (void (i32)* @one_arg to void (i32, i32)*)(i32 0, i32 0)
|
|
; CHECK: Call argument count mismatches callee argument count
|
|
call void () bitcast (void (i32)* @one_arg to void ()*)()
|
|
; CHECK: Call argument type mismatches callee parameter type
|
|
call void (float) bitcast (void (i32)* @one_arg to void (float)*)(float 0.0)
|
|
|
|
; CHECK: Write to read-only memory
|
|
call void @llvm.memcpy.p0i8.p0i8.i64(i8* bitcast (i32* @CG to i8*), i8* bitcast (i32* @CG2 to i8*), i64 1, i1 0)
|
|
; CHECK: Write to read-only memory
|
|
call void @llvm.memcpy.inline.p0i8.p0i8.i64(i8* bitcast (i32* @CG to i8*), i8* bitcast (i32* @CG2 to i8*), i64 1, i1 0)
|
|
; CHECK: Unusual: noalias argument aliases another argument
|
|
call void @llvm.memcpy.p0i8.p0i8.i64(i8* bitcast (i32* @CG to i8*), i8* bitcast (i32* @CG to i8*), i64 1, i1 0)
|
|
|
|
; CHECK: Undefined behavior: Buffer overflow
|
|
%wider = bitcast i8* %buf to i16*
|
|
store i16 0, i16* %wider
|
|
; CHECK: Undefined behavior: Buffer overflow
|
|
%inner = getelementptr {i8, i8}, {i8, i8}* %buf2, i32 0, i32 1
|
|
%wider2 = bitcast i8* %inner to i16*
|
|
store i16 0, i16* %wider2
|
|
; CHECK: Undefined behavior: Buffer overflow
|
|
%before = getelementptr i8, i8* %buf, i32 -1
|
|
%wider3 = bitcast i8* %before to i16*
|
|
store i16 0, i16* %wider3
|
|
|
|
br label %next
|
|
|
|
next:
|
|
; CHECK: Static alloca outside of entry block
|
|
%a = alloca i32
|
|
; CHECK: Return statement in function with noreturn attribute
|
|
ret i32 0
|
|
|
|
foo:
|
|
; CHECK-NOT: Undefined behavior: Buffer overflow
|
|
; CHECK-NOT: Memory reference address is misaligned
|
|
%e = bitcast i8* @E to i64*
|
|
store i64 0, i64* %e
|
|
%z = add i32 0, 0
|
|
; CHECK: unreachable immediately preceded by instruction without side effects
|
|
unreachable
|
|
}
|
|
|
|
; CHECK: Unnamed function with non-local linkage
|
|
define void @0() nounwind {
|
|
ret void
|
|
}
|
|
|
|
; CHECK: va_start called in a non-varargs function
|
|
declare void @llvm.va_start(i8*)
|
|
define void @not_vararg(i8* %p) nounwind {
|
|
call void @llvm.va_start(i8* %p)
|
|
ret void
|
|
}
|
|
|
|
; CHECK: Undefined behavior: Branch to non-blockaddress
|
|
define void @use_indbr() {
|
|
indirectbr i8* bitcast (i32()* @foo to i8*), [label %block]
|
|
block:
|
|
unreachable
|
|
}
|
|
|
|
; CHECK: Undefined behavior: Call with "tail" keyword references alloca
|
|
declare void @tailcallee(i8*)
|
|
define void @use_tail(i8* %valist) {
|
|
%t = alloca i8
|
|
tail call void @tailcallee(i8* %t)
|
|
ret void
|
|
}
|
|
|
|
; CHECK: Unusual: Returning alloca value
|
|
define i8* @return_local(i32 %n, i32 %m) {
|
|
%t = alloca i8, i32 %n
|
|
%s = getelementptr i8, i8* %t, i32 %m
|
|
ret i8* %s
|
|
}
|
|
|
|
; CHECK: Unusual: Returning alloca value
|
|
define i32* @return_obscured_local() {
|
|
entry:
|
|
%retval = alloca i32*
|
|
%x = alloca i32
|
|
store i32* %x, i32** %retval
|
|
br label %next
|
|
next:
|
|
%t0 = load i32*, i32** %retval
|
|
%t1 = insertvalue { i32, i32, i32* } zeroinitializer, i32* %t0, 2
|
|
%t2 = extractvalue { i32, i32, i32* } %t1, 2
|
|
br label %exit
|
|
exit:
|
|
%t3 = phi i32* [ %t2, %next ]
|
|
%t4 = bitcast i32* %t3 to i32*
|
|
%t5 = ptrtoint i32* %t4 to i64
|
|
%t6 = add i64 %t5, 0
|
|
%t7 = inttoptr i64 %t6 to i32*
|
|
ret i32* %t7
|
|
}
|
|
|
|
; CHECK: Undefined behavior: Undef pointer dereference
|
|
define i32* @self_reference() {
|
|
entry:
|
|
unreachable
|
|
exit:
|
|
%t3 = phi i32* [ %t4, %exit ]
|
|
%t4 = bitcast i32* %t3 to i32*
|
|
%x = load volatile i32, i32* %t3
|
|
br label %exit
|
|
}
|
|
|
|
; CHECK: Call return type mismatches callee return type
|
|
%struct = type { double, double }
|
|
declare i32 @nonstruct_callee() nounwind
|
|
define void @struct_caller() nounwind {
|
|
entry:
|
|
call %struct bitcast (i32 ()* @foo to %struct ()*)()
|
|
|
|
; CHECK: Undefined behavior: indirectbr with no destinations
|
|
indirectbr i8* null, []
|
|
}
|
|
|
|
define i32 @memcpy_inline_same_address() noreturn {
|
|
%buf = alloca i64, align 1
|
|
%ptr = bitcast i64* %buf to i8*
|
|
; CHECK: Unusual: noalias argument aliases another argument
|
|
call void @llvm.memcpy.inline.p0i8.p0i8.i64(i8* %ptr, i8* %ptr, i64 1, i1 false)
|
|
unreachable
|
|
}
|