revert my previous patches that introduced an additional parameter to the objectsize intrinsic.

After a lot of discussion, we realized it's not the best option for run-time bounds checking

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@157255 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Nuno Lopes
2012-05-22 15:25:31 +00:00
parent 76ff741836
commit 23e75da7e0
15 changed files with 105 additions and 276 deletions

View File

@@ -8467,8 +8467,8 @@ LLVM</a>.</p>
<h5>Syntax:</h5> <h5>Syntax:</h5>
<pre> <pre>
declare i32 @llvm.objectsize.i32(i8* &lt;object&gt;, i1 &lt;min&gt;, i32 &lt;runtime&gt;) declare i32 @llvm.objectsize.i32(i8* &lt;object&gt;, i1 &lt;min&gt;)
declare i64 @llvm.objectsize.i64(i8* &lt;object&gt;, i1 &lt;min&gt;, i32 &lt;runtime&gt;) declare i64 @llvm.objectsize.i64(i8* &lt;object&gt;, i1 &lt;min&gt;)
</pre> </pre>
<h5>Overview:</h5> <h5>Overview:</h5>
@@ -8479,21 +8479,17 @@ LLVM</a>.</p>
an allocation of a specific class, structure, array, or other object.</p> an allocation of a specific class, structure, array, or other object.</p>
<h5>Arguments:</h5> <h5>Arguments:</h5>
<p>The <tt>llvm.objectsize</tt> intrinsic takes three arguments. The first <p>The <tt>llvm.objectsize</tt> intrinsic takes two arguments. The first
argument is a pointer to or into the <tt>object</tt>. The second argument argument is a pointer to or into the <tt>object</tt>. The second argument
is a boolean and determines whether <tt>llvm.objectsize</tt> returns 0 (if true) is a boolean and determines whether <tt>llvm.objectsize</tt> returns 0 (if
or -1 (if false) when the object size is unknown. true) or -1 (if false) when the object size is unknown.
The third argument, <tt>runtime</tt>, indicates whether the compiler is allowed The second argument only accepts constants.</p>
to return a non-constant value. The higher the value, the higher the potential
run-time performance impact.
The second and third arguments only accepts constants.</p>
<h5>Semantics:</h5> <h5>Semantics:</h5>
<p>The <tt>llvm.objectsize</tt> intrinsic is lowered to a constant representing <p>The <tt>llvm.objectsize</tt> intrinsic is lowered to a constant representing
the size of the object concerned. If the size cannot be determined at compile the size of the object concerned. If the size cannot be determined at compile
time, <tt>llvm.objectsize</tt> either returns <tt>i32/i64 -1 or 0</tt> time, <tt>llvm.objectsize</tt> returns <tt>i32/i64 -1 or 0</tt>
(depending on the <tt>min</tt> argument) if <tt>runtime</tt> is 0, or a run-time (depending on the <tt>min</tt> argument).</p>
value (if <tt>runtime</tt> &gt; 0 and an expression could be generated).</p>
</div> </div>
<!-- _______________________________________________________________________ --> <!-- _______________________________________________________________________ -->

View File

@@ -272,8 +272,7 @@ def int_sigsetjmp : Intrinsic<[llvm_i32_ty] , [llvm_ptr_ty, llvm_i32_ty]>;
def int_siglongjmp : Intrinsic<[], [llvm_ptr_ty, llvm_i32_ty]>; def int_siglongjmp : Intrinsic<[], [llvm_ptr_ty, llvm_i32_ty]>;
// Internal interface for object size checking // Internal interface for object size checking
def int_objectsize : Intrinsic<[llvm_anyint_ty], def int_objectsize : Intrinsic<[llvm_anyint_ty], [llvm_ptr_ty, llvm_i1_ty],
[llvm_ptr_ty, llvm_i1_ty, llvm_i32_ty],
[IntrNoMem]>, [IntrNoMem]>,
GCCBuiltin<"__builtin_object_size">; GCCBuiltin<"__builtin_object_size">;

View File

@@ -165,73 +165,6 @@ Instruction *InstCombiner::SimplifyMemSet(MemSetInst *MI) {
return 0; return 0;
} }
/// computeAllocSize - compute the object size allocated by an allocation
/// site. Returns 0 if the size is not constant (in SizeValue), 1 if the size
/// is constant (in Size), and 2 if the size could not be determined within the
/// given maximum Penalty that the computation would incurr at run-time.
static int computeAllocSize(Value *Alloc, uint64_t &Size, Value* &SizeValue,
uint64_t Penalty, TargetData *TD,
InstCombiner::BuilderTy *Builder) {
if (GlobalVariable *GV = dyn_cast<GlobalVariable>(Alloc)) {
if (GV->hasDefinitiveInitializer()) {
Constant *C = GV->getInitializer();
Size = TD->getTypeAllocSize(C->getType());
return 1;
}
// Can't determine size of the GV.
return 2;
} else if (AllocaInst *AI = dyn_cast<AllocaInst>(Alloc)) {
if (!AI->getAllocatedType()->isSized())
return 2;
Size = TD->getTypeAllocSize(AI->getAllocatedType());
if (!AI->isArrayAllocation())
return 1; // we are done
Value *ArraySize = AI->getArraySize();
if (const ConstantInt *C = dyn_cast<ConstantInt>(ArraySize)) {
Size *= C->getZExtValue();
return 1;
}
if (Penalty < 2)
return 2;
SizeValue = ConstantInt::get(ArraySize->getType(), Size);
SizeValue = Builder->CreateMul(SizeValue, ArraySize);
return 0;
} else if (CallInst *MI = extractMallocCall(Alloc)) {
SizeValue = MI->getArgOperand(0);
if (ConstantInt *CI = dyn_cast<ConstantInt>(SizeValue)) {
Size = CI->getZExtValue();
return 1;
}
return Penalty >= 2 ? 0 : 2;
} else if (CallInst *MI = extractCallocCall(Alloc)) {
Value *Arg1 = MI->getArgOperand(0);
Value *Arg2 = MI->getArgOperand(1);
if (ConstantInt *CI1 = dyn_cast<ConstantInt>(Arg1)) {
if (ConstantInt *CI2 = dyn_cast<ConstantInt>(Arg2)) {
Size = (CI1->getValue() * CI2->getValue()).getZExtValue();
return 1;
}
}
if (Penalty < 2)
return 2;
SizeValue = Builder->CreateMul(Arg1, Arg2);
return 0;
}
DEBUG(errs() << "computeAllocSize failed:\n");
DEBUG(Alloc->dump());
return 2;
}
/// visitCallInst - CallInst simplification. This mostly only handles folding /// visitCallInst - CallInst simplification. This mostly only handles folding
/// of intrinsic instructions. For normal calls, it allows visitCallSite to do /// of intrinsic instructions. For normal calls, it allows visitCallSite to do
/// the heavy lifting. /// the heavy lifting.
@@ -317,59 +250,80 @@ Instruction *InstCombiner::visitCallInst(CallInst &CI) {
if (!TD) return 0; if (!TD) return 0;
Type *ReturnTy = CI.getType(); Type *ReturnTy = CI.getType();
uint64_t Penalty = cast<ConstantInt>(II->getArgOperand(2))->getZExtValue(); uint64_t DontKnow = II->getArgOperand(1) == Builder->getTrue() ? 0 : -1ULL;
// Get to the real allocated thing and offset as fast as possible. // Get to the real allocated thing and offset as fast as possible.
Value *Op1 = II->getArgOperand(0)->stripPointerCasts(); Value *Op1 = II->getArgOperand(0)->stripPointerCasts();
GEPOperator *GEP;
if ((GEP = dyn_cast<GEPOperator>(Op1))) { uint64_t Offset = 0;
// check if we will be able to get the offset uint64_t Size = -1ULL;
if (!GEP->hasAllConstantIndices() && Penalty < 2)
// Try to look through constant GEPs.
if (GEPOperator *GEP = dyn_cast<GEPOperator>(Op1)) {
if (!GEP->hasAllConstantIndices()) return 0;
// Get the current byte offset into the thing. Use the original
// operand in case we're looking through a bitcast.
SmallVector<Value*, 8> Ops(GEP->idx_begin(), GEP->idx_end());
if (!GEP->getPointerOperandType()->isPointerTy())
return 0; return 0;
Offset = TD->getIndexedOffset(GEP->getPointerOperandType(), Ops);
Op1 = GEP->getPointerOperand()->stripPointerCasts(); Op1 = GEP->getPointerOperand()->stripPointerCasts();
// Make sure we're not a constant offset from an external
// global.
if (GlobalVariable *GV = dyn_cast<GlobalVariable>(Op1))
if (!GV->hasDefinitiveInitializer()) return 0;
} }
uint64_t Size; // If we've stripped down to a single global variable that we
Value *SizeValue; // can know the size of then just return that.
int ConstAlloc = computeAllocSize(Op1, Size, SizeValue, Penalty, TD, if (GlobalVariable *GV = dyn_cast<GlobalVariable>(Op1)) {
Builder); if (GV->hasDefinitiveInitializer()) {
Constant *C = GV->getInitializer();
Size = TD->getTypeAllocSize(C->getType());
} else {
// Can't determine size of the GV.
Constant *RetVal = ConstantInt::get(ReturnTy, DontKnow);
return ReplaceInstUsesWith(CI, RetVal);
}
} else if (AllocaInst *AI = dyn_cast<AllocaInst>(Op1)) {
// Get alloca size.
if (AI->getAllocatedType()->isSized()) {
Size = TD->getTypeAllocSize(AI->getAllocatedType());
if (AI->isArrayAllocation()) {
const ConstantInt *C = dyn_cast<ConstantInt>(AI->getArraySize());
if (!C) return 0;
Size *= C->getZExtValue();
}
}
} else if (CallInst *MI = extractMallocCall(Op1)) {
// Get allocation size.
Value *Arg = MI->getArgOperand(0);
if (ConstantInt *CI = dyn_cast<ConstantInt>(Arg))
Size = CI->getZExtValue();
} else if (CallInst *MI = extractCallocCall(Op1)) {
// Get allocation size.
Value *Arg1 = MI->getArgOperand(0);
Value *Arg2 = MI->getArgOperand(1);
if (ConstantInt *CI1 = dyn_cast<ConstantInt>(Arg1))
if (ConstantInt *CI2 = dyn_cast<ConstantInt>(Arg2))
Size = (CI1->getValue() * CI2->getValue()).getZExtValue();
}
// Do not return "I don't know" here. Later optimization passes could // Do not return "I don't know" here. Later optimization passes could
// make it possible to evaluate objectsize to a constant. // make it possible to evaluate objectsize to a constant.
if (ConstAlloc == 2) if (Size == -1ULL)
return 0; return 0;
uint64_t Offset = 0; if (Size < Offset) {
Value *OffsetValue = 0; // Out of bound reference? Negative index normalized to large
// index? Just return "I don't know".
if (GEP) { return ReplaceInstUsesWith(CI, ConstantInt::get(ReturnTy, DontKnow));
if (GEP->hasAllConstantIndices()) {
SmallVector<Value*, 8> Ops(GEP->idx_begin(), GEP->idx_end());
assert(GEP->getPointerOperandType()->isPointerTy());
Offset = TD->getIndexedOffset(GEP->getPointerOperandType(), Ops);
} else
OffsetValue = EmitGEPOffset(GEP, true /*NoNUW*/);
} }
return ReplaceInstUsesWith(CI, ConstantInt::get(ReturnTy, Size-Offset));
if (!OffsetValue && ConstAlloc) {
if (Size < Offset) {
// Out of bounds
return ReplaceInstUsesWith(CI, ConstantInt::get(ReturnTy, 0));
}
return ReplaceInstUsesWith(CI, ConstantInt::get(ReturnTy, Size-Offset));
}
if (!OffsetValue)
OffsetValue = ConstantInt::get(ReturnTy, Offset);
if (ConstAlloc)
SizeValue = ConstantInt::get(ReturnTy, Size);
Value *Val = Builder->CreateSub(SizeValue, OffsetValue);
// return 0 if there's an overflow
Value *Cmp = Builder->CreateICmpULT(SizeValue, OffsetValue);
Val = Builder->CreateSelect(Cmp, ConstantInt::get(ReturnTy, 0), Val);
return ReplaceInstUsesWith(CI, Val);
} }
case Intrinsic::bswap: case Intrinsic::bswap:
// bswap(bswap(x)) -> x // bswap(bswap(x)) -> x

View File

@@ -52,20 +52,6 @@ static bool UpgradeIntrinsicFunction1(Function *F, Function *&NewFn) {
} }
break; break;
} }
case 'o': {
// FIXME: remove in LLVM 3.3
if (Name.startswith("objectsize.") && F->arg_size() == 2) {
Type *Tys[] = {F->getReturnType(),
F->arg_begin()->getType(),
Type::getInt1Ty(F->getContext()),
Type::getInt32Ty(F->getContext())};
NewFn = Intrinsic::getDeclaration(F->getParent(), Intrinsic::objectsize,
Tys);
NewFn->takeName(F);
return true;
}
break;
}
case 'x': { case 'x': {
if (Name.startswith("x86.sse2.pcmpeq.") || if (Name.startswith("x86.sse2.pcmpeq.") ||
Name.startswith("x86.sse2.pcmpgt.") || Name.startswith("x86.sse2.pcmpgt.") ||
@@ -209,7 +195,7 @@ void llvm::UpgradeIntrinsicCall(CallInst *CI, Function *NewFn) {
llvm_unreachable("Unknown function for CallInst upgrade."); llvm_unreachable("Unknown function for CallInst upgrade.");
case Intrinsic::ctlz: case Intrinsic::ctlz:
case Intrinsic::cttz: { case Intrinsic::cttz:
assert(CI->getNumArgOperands() == 1 && assert(CI->getNumArgOperands() == 1 &&
"Mismatch between function args and call args"); "Mismatch between function args and call args");
StringRef Name = CI->getName(); StringRef Name = CI->getName();
@@ -219,16 +205,6 @@ void llvm::UpgradeIntrinsicCall(CallInst *CI, Function *NewFn) {
CI->eraseFromParent(); CI->eraseFromParent();
return; return;
} }
case Intrinsic::objectsize: {
StringRef Name = CI->getName();
CI->setName(Name + ".old");
CI->replaceAllUsesWith(Builder.CreateCall3(NewFn, CI->getArgOperand(0),
CI->getArgOperand(1),
Builder.getInt32(0), Name));
CI->eraseFromParent();
return;
}
}
} }
// This tests each Function to determine if it needs upgrading. When we find // This tests each Function to determine if it needs upgrading. When we find

View File

@@ -43,7 +43,7 @@ bb:
%3 = load i32* @tabsize, align 4 %3 = load i32* @tabsize, align 4
%4 = srem i32 %cols, %3 %4 = srem i32 %cols, %3
%5 = sdiv i32 %cols, %3 %5 = sdiv i32 %cols, %3
%6 = tail call i32 @llvm.objectsize.i32(i8* null, i1 false, i32 0) %6 = tail call i32 @llvm.objectsize.i32(i8* null, i1 false)
%7 = tail call i8* @__memset_chk(i8* null, i32 9, i32 %5, i32 %6) nounwind %7 = tail call i8* @__memset_chk(i8* null, i32 9, i32 %5, i32 %6) nounwind
br label %bb1 br label %bb1
@@ -54,5 +54,5 @@ bb1:
ret void ret void
} }
declare i32 @llvm.objectsize.i32(i8*, i1, i32) nounwind readnone declare i32 @llvm.objectsize.i32(i8*, i1) nounwind readnone
declare i8* @__memset_chk(i8*, i32, i32, i32) nounwind declare i8* @__memset_chk(i8*, i32, i32, i32) nounwind

View File

@@ -23,7 +23,7 @@ bb32: ; preds = %bb6
%3 = load double* %1, align 4 %3 = load double* %1, align 4
%4 = load double* %0, align 4 %4 = load double* %0, align 4
call void @Parse_Vector(double* %0) nounwind call void @Parse_Vector(double* %0) nounwind
%5 = call i32 @llvm.objectsize.i32(i8* undef, i1 false, i32 0) %5 = call i32 @llvm.objectsize.i32(i8* undef, i1 false)
%6 = icmp eq i32 %5, -1 %6 = icmp eq i32 %5, -1
br i1 %6, label %bb34, label %bb33 br i1 %6, label %bb34, label %bb33
@@ -36,7 +36,7 @@ unreachable
} }
declare void @Parse_Vector(double*) declare void @Parse_Vector(double*)
declare i32 @llvm.objectsize.i32(i8*, i1, i32) declare i32 @llvm.objectsize.i32(i8*, i1)
; PR9578 ; PR9578

View File

@@ -1,16 +0,0 @@
; RUN: opt < %s -verify -S | FileCheck %s
; check automatic upgrade of objectsize. To be removed in LLVM 3.3.
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"
define i32 @foo() nounwind {
; CHECK: @foo
%1 = alloca i8, align 4
%2 = getelementptr inbounds i8* %1, i32 0
; CHECK: llvm.objectsize.i32(i8* %2, i1 false, i32 0)
%3 = call i32 @llvm.objectsize.i32(i8* %2, i1 0)
ret i32 %3
}
; CHECK: @llvm.objectsize.i32(i8*, i1, i32)
declare i32 @llvm.objectsize.i32(i8*, i1) nounwind readonly

View File

@@ -8,7 +8,7 @@ target triple = "x86_64-apple-macosx10.6.0"
@aux_temp = external global %struct.dfa, align 8 @aux_temp = external global %struct.dfa, align 8
declare i64 @llvm.objectsize.i64(i8*, i1, i32) nounwind readnone declare i64 @llvm.objectsize.i64(i8*, i1) nounwind readnone
declare void @__memset_chk() nounwind declare void @__memset_chk() nounwind
@@ -21,12 +21,12 @@ if.end.i: ; preds = %entry
br i1 undef, label %land.end.thread.i, label %land.end.i br i1 undef, label %land.end.thread.i, label %land.end.i
land.end.thread.i: ; preds = %if.end.i land.end.thread.i: ; preds = %if.end.i
%0 = call i64 @llvm.objectsize.i64(i8* undef, i1 false, i32 0) nounwind %0 = call i64 @llvm.objectsize.i64(i8* undef, i1 false) nounwind
%cmp1710.i = icmp eq i64 %0, -1 %cmp1710.i = icmp eq i64 %0, -1
br i1 %cmp1710.i, label %cond.false156.i, label %cond.true138.i br i1 %cmp1710.i, label %cond.false156.i, label %cond.true138.i
land.end.i: ; preds = %if.end.i land.end.i: ; preds = %if.end.i
%1 = call i64 @llvm.objectsize.i64(i8* undef, i1 false, i32 0) nounwind %1 = call i64 @llvm.objectsize.i64(i8* undef, i1 false) nounwind
%cmp17.i = icmp eq i64 %1, -1 %cmp17.i = icmp eq i64 %1, -1
br i1 %cmp17.i, label %cond.false156.i, label %cond.true138.i br i1 %cmp17.i, label %cond.false156.i, label %cond.true138.i

View File

@@ -203,7 +203,7 @@ entry:
; <rdar://problem/9187792> ; <rdar://problem/9187792>
define fastcc void @func_61() nounwind sspreq { define fastcc void @func_61() nounwind sspreq {
entry: entry:
%t1 = tail call i64 @llvm.objectsize.i64(i8* undef, i1 false, i32 0) %t1 = tail call i64 @llvm.objectsize.i64(i8* undef, i1 false)
%t2 = icmp eq i64 %t1, -1 %t2 = icmp eq i64 %t1, -1
br i1 %t2, label %bb2, label %bb1 br i1 %t2, label %bb2, label %bb1
@@ -214,7 +214,7 @@ bb2:
ret void ret void
} }
declare i64 @llvm.objectsize.i64(i8*, i1, i32) nounwind readnone declare i64 @llvm.objectsize.i64(i8*, i1) nounwind readnone
; PR10277 ; PR10277
; This test has dead code elimination caused by remat during spilling. ; This test has dead code elimination caused by remat during spilling.

View File

@@ -10,7 +10,7 @@ target triple = "x86_64-apple-darwin10.0"
define void @bar() nounwind ssp { define void @bar() nounwind ssp {
entry: entry:
%tmp = load i8** @p ; <i8*> [#uses=1] %tmp = load i8** @p ; <i8*> [#uses=1]
%0 = call i64 @llvm.objectsize.i64(i8* %tmp, i1 0, i32 0) ; <i64> [#uses=1] %0 = call i64 @llvm.objectsize.i64(i8* %tmp, i1 0) ; <i64> [#uses=1]
%cmp = icmp ne i64 %0, -1 ; <i1> [#uses=1] %cmp = icmp ne i64 %0, -1 ; <i1> [#uses=1]
; X64: movabsq $-1, [[RAX:%r..]] ; X64: movabsq $-1, [[RAX:%r..]]
; X64: cmpq $-1, [[RAX]] ; X64: cmpq $-1, [[RAX]]
@@ -19,7 +19,7 @@ entry:
cond.true: ; preds = %entry cond.true: ; preds = %entry
%tmp1 = load i8** @p ; <i8*> [#uses=1] %tmp1 = load i8** @p ; <i8*> [#uses=1]
%tmp2 = load i8** @p ; <i8*> [#uses=1] %tmp2 = load i8** @p ; <i8*> [#uses=1]
%1 = call i64 @llvm.objectsize.i64(i8* %tmp2, i1 1, i32 0) ; <i64> [#uses=1] %1 = call i64 @llvm.objectsize.i64(i8* %tmp2, i1 1) ; <i64> [#uses=1]
%call = call i8* @__strcpy_chk(i8* %tmp1, i8* getelementptr inbounds ([3 x i8]* @.str, i32 0, i32 0), i64 %1) ssp ; <i8*> [#uses=1] %call = call i8* @__strcpy_chk(i8* %tmp1, i8* getelementptr inbounds ([3 x i8]* @.str, i32 0, i32 0), i64 %1) ssp ; <i8*> [#uses=1]
br label %cond.end br label %cond.end
@@ -33,7 +33,7 @@ cond.end: ; preds = %cond.false, %cond.t
ret void ret void
} }
declare i64 @llvm.objectsize.i64(i8*, i1, i32) nounwind readonly declare i64 @llvm.objectsize.i64(i8*, i1) nounwind readonly
declare i8* @__strcpy_chk(i8*, i8*, i64) ssp declare i8* @__strcpy_chk(i8*, i8*, i64) ssp
@@ -47,7 +47,7 @@ entry:
%tmp = load i8** %__dest.addr ; <i8*> [#uses=1] %tmp = load i8** %__dest.addr ; <i8*> [#uses=1]
%tmp1 = load i8** %__src.addr ; <i8*> [#uses=1] %tmp1 = load i8** %__src.addr ; <i8*> [#uses=1]
%tmp2 = load i8** %__dest.addr ; <i8*> [#uses=1] %tmp2 = load i8** %__dest.addr ; <i8*> [#uses=1]
%0 = call i64 @llvm.objectsize.i64(i8* %tmp2, i1 1, i32 0) ; <i64> [#uses=1] %0 = call i64 @llvm.objectsize.i64(i8* %tmp2, i1 1) ; <i64> [#uses=1]
%call = call i8* @__strcpy_chk(i8* %tmp, i8* %tmp1, i64 %0) ssp ; <i8*> [#uses=1] %call = call i8* @__strcpy_chk(i8* %tmp, i8* %tmp1, i64 %0) ssp ; <i8*> [#uses=1]
store i8* %call, i8** %retval store i8* %call, i8** %retval
%1 = load i8** %retval ; <i8*> [#uses=1] %1 = load i8** %retval ; <i8*> [#uses=1]

View File

@@ -2,13 +2,13 @@
target datalayout = "e-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64-s0:64:64-f80:128:128-n8:16:32:64" target datalayout = "e-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64-s0:64:64-f80:128:128-n8:16:32:64"
target triple = "x86_64-apple-darwin10.0.0" target triple = "x86_64-apple-darwin10.0.0"
declare i64 @llvm.objectsize.i64(i8*, i1, i32) nounwind readnone declare i64 @llvm.objectsize.i64(i8*, i1) nounwind readnone
define void @test5() nounwind optsize noinline ssp { define void @test5() nounwind optsize noinline ssp {
entry: entry:
; CHECK: movq ___stack_chk_guard@GOTPCREL(%rip) ; CHECK: movq ___stack_chk_guard@GOTPCREL(%rip)
%buf = alloca [64 x i8], align 16 %buf = alloca [64 x i8], align 16
%0 = call i64 @llvm.objectsize.i64(i8* undef, i1 false, i32 0) %0 = call i64 @llvm.objectsize.i64(i8* undef, i1 false)
br i1 false, label %if.end, label %if.then br i1 false, label %if.end, label %if.then
if.then: ; preds = %entry if.then: ; preds = %entry

View File

@@ -2,7 +2,7 @@
declare void @llvm.dbg.declare(metadata, metadata) nounwind readnone declare void @llvm.dbg.declare(metadata, metadata) nounwind readnone
declare i64 @llvm.objectsize.i64(i8*, i1, i32) nounwind readnone declare i64 @llvm.objectsize.i64(i8*, i1) nounwind readnone
declare i8* @foo(i8*, i32, i64, i64) nounwind declare i8* @foo(i8*, i32, i64, i64) nounwind
@@ -23,7 +23,7 @@ entry:
%tmp1 = load i32* %__val.addr, align 4, !dbg !21, !tbaa !17 %tmp1 = load i32* %__val.addr, align 4, !dbg !21, !tbaa !17
%tmp2 = load i64* %__len.addr, align 8, !dbg !21, !tbaa !19 %tmp2 = load i64* %__len.addr, align 8, !dbg !21, !tbaa !19
%tmp3 = load i8** %__dest.addr, align 8, !dbg !21, !tbaa !13 %tmp3 = load i8** %__dest.addr, align 8, !dbg !21, !tbaa !13
%0 = call i64 @llvm.objectsize.i64(i8* %tmp3, i1 false, i32 0), !dbg !21 %0 = call i64 @llvm.objectsize.i64(i8* %tmp3, i1 false), !dbg !21
%call = call i8* @foo(i8* %tmp, i32 %tmp1, i64 %tmp2, i64 %0), !dbg !21 %call = call i8* @foo(i8* %tmp, i32 %tmp1, i64 %tmp2, i64 %0), !dbg !21
ret i8* %call, !dbg !21 ret i8* %call, !dbg !21
} }

View File

@@ -9,7 +9,7 @@ target datalayout = "e-p:32:32:32-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:32:64-f3
define i32 @foo() nounwind { define i32 @foo() nounwind {
; CHECK: @foo ; CHECK: @foo
; CHECK-NEXT: ret i32 60 ; CHECK-NEXT: ret i32 60
%1 = call i32 @llvm.objectsize.i32(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i1 false, i32 0) %1 = call i32 @llvm.objectsize.i32(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i1 false)
ret i32 %1 ret i32 %1
} }
@@ -17,7 +17,7 @@ define i8* @bar() nounwind {
; CHECK: @bar ; CHECK: @bar
entry: entry:
%retval = alloca i8* %retval = alloca i8*
%0 = call i32 @llvm.objectsize.i32(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i1 false, i32 0) %0 = call i32 @llvm.objectsize.i32(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i1 false)
%cmp = icmp ne i32 %0, -1 %cmp = icmp ne i32 %0, -1
; CHECK: br i1 true ; CHECK: br i1 true
br i1 %cmp, label %cond.true, label %cond.false br i1 %cmp, label %cond.true, label %cond.false
@@ -34,7 +34,7 @@ cond.false:
define i32 @f() nounwind { define i32 @f() nounwind {
; CHECK: @f ; CHECK: @f
; CHECK-NEXT: ret i32 0 ; CHECK-NEXT: ret i32 0
%1 = call i32 @llvm.objectsize.i32(i8* getelementptr ([60 x i8]* @a, i32 1, i32 0), i1 false, i32 0) %1 = call i32 @llvm.objectsize.i32(i8* getelementptr ([60 x i8]* @a, i32 1, i32 0), i1 false)
ret i32 %1 ret i32 %1
} }
@@ -42,8 +42,8 @@ define i32 @f() nounwind {
define i1 @baz() nounwind { define i1 @baz() nounwind {
; CHECK: @baz ; CHECK: @baz
; CHECK-NEXT: objectsize ; CHECK-NEXT: ret i1 true
%1 = tail call i32 @llvm.objectsize.i32(i8* getelementptr inbounds ([0 x i8]* @window, i32 0, i32 0), i1 false, i32 0) %1 = tail call i32 @llvm.objectsize.i32(i8* getelementptr inbounds ([0 x i8]* @window, i32 0, i32 0), i1 false)
%2 = icmp eq i32 %1, -1 %2 = icmp eq i32 %1, -1
ret i1 %2 ret i1 %2
} }
@@ -52,7 +52,7 @@ define void @test1(i8* %q, i32 %x) nounwind noinline {
; CHECK: @test1 ; CHECK: @test1
; CHECK: objectsize.i32 ; CHECK: objectsize.i32
entry: entry:
%0 = call i32 @llvm.objectsize.i32(i8* getelementptr inbounds ([0 x i8]* @window, i32 0, i32 10), i1 false, i32 0) ; <i64> [#uses=1] %0 = call i32 @llvm.objectsize.i32(i8* getelementptr inbounds ([0 x i8]* @window, i32 0, i32 10), i1 false) ; <i64> [#uses=1]
%1 = icmp eq i32 %0, -1 ; <i1> [#uses=1] %1 = icmp eq i32 %0, -1 ; <i1> [#uses=1]
br i1 %1, label %"47", label %"46" br i1 %1, label %"47", label %"46"
@@ -68,7 +68,7 @@ entry:
define i32 @test2() nounwind { define i32 @test2() nounwind {
; CHECK: @test2 ; CHECK: @test2
; CHECK-NEXT: ret i32 34 ; CHECK-NEXT: ret i32 34
%1 = call i32 @llvm.objectsize.i32(i8* getelementptr (i8* bitcast ([9 x i32]* @.str5 to i8*), i32 2), i1 false, i32 0) %1 = call i32 @llvm.objectsize.i32(i8* getelementptr (i8* bitcast ([9 x i32]* @.str5 to i8*), i32 2), i1 false)
ret i32 %1 ret i32 %1
} }
@@ -77,7 +77,7 @@ define i32 @test2() nounwind {
declare i8* @__memcpy_chk(i8*, i8*, i32, i32) nounwind declare i8* @__memcpy_chk(i8*, i8*, i32, i32) nounwind
declare i32 @llvm.objectsize.i32(i8*, i1, i32) nounwind readonly declare i32 @llvm.objectsize.i32(i8*, i1) nounwind readonly
declare i8* @__inline_memcpy_chk(i8*, i8*, i32) nounwind inlinehint declare i8* @__inline_memcpy_chk(i8*, i8*, i32) nounwind inlinehint
@@ -89,7 +89,7 @@ entry:
bb11: bb11:
%0 = getelementptr inbounds float* getelementptr inbounds ([480 x float]* @array, i32 0, i32 128), i32 -127 ; <float*> [#uses=1] %0 = getelementptr inbounds float* getelementptr inbounds ([480 x float]* @array, i32 0, i32 128), i32 -127 ; <float*> [#uses=1]
%1 = bitcast float* %0 to i8* ; <i8*> [#uses=1] %1 = bitcast float* %0 to i8* ; <i8*> [#uses=1]
%2 = call i32 @llvm.objectsize.i32(i8* %1, i1 false, i32 0) ; <i32> [#uses=1] %2 = call i32 @llvm.objectsize.i32(i8* %1, i1 false) ; <i32> [#uses=1]
%3 = call i8* @__memcpy_chk(i8* undef, i8* undef, i32 512, i32 %2) nounwind ; <i8*> [#uses=0] %3 = call i8* @__memcpy_chk(i8* undef, i8* undef, i32 512, i32 %2) nounwind ; <i8*> [#uses=0]
; CHECK: unreachable ; CHECK: unreachable
unreachable unreachable
@@ -111,7 +111,7 @@ define i32 @test4() nounwind ssp {
entry: entry:
%0 = alloca %struct.data, align 8 %0 = alloca %struct.data, align 8
%1 = bitcast %struct.data* %0 to i8* %1 = bitcast %struct.data* %0 to i8*
%2 = call i32 @llvm.objectsize.i32(i8* %1, i1 false, i32 0) nounwind %2 = call i32 @llvm.objectsize.i32(i8* %1, i1 false) nounwind
; CHECK-NOT: @llvm.objectsize ; CHECK-NOT: @llvm.objectsize
; CHECK: @llvm.memset.p0i8.i32(i8* %1, i8 0, i32 1824, i32 8, i1 false) ; CHECK: @llvm.memset.p0i8.i32(i8* %1, i8 0, i32 1824, i32 8, i1 false)
%3 = call i8* @__memset_chk(i8* %1, i32 0, i32 1824, i32 %2) nounwind %3 = call i8* @__memset_chk(i8* %1, i32 0, i32 1824, i32 %2) nounwind
@@ -125,7 +125,7 @@ define void @test5(i32 %n) nounwind ssp {
; CHECK: @test5 ; CHECK: @test5
entry: entry:
%0 = tail call noalias i8* @malloc(i32 20) nounwind %0 = tail call noalias i8* @malloc(i32 20) nounwind
%1 = tail call i32 @llvm.objectsize.i32(i8* %0, i1 false, i32 0) %1 = tail call i32 @llvm.objectsize.i32(i8* %0, i1 false)
%2 = load i8** @s, align 8 %2 = load i8** @s, align 8
; CHECK-NOT: @llvm.objectsize ; CHECK-NOT: @llvm.objectsize
; CHECK: @llvm.memcpy.p0i8.p0i8.i32(i8* %0, i8* %1, i32 10, i32 1, i1 false) ; CHECK: @llvm.memcpy.p0i8.p0i8.i32(i8* %0, i8* %1, i32 10, i32 1, i1 false)
@@ -137,7 +137,7 @@ define void @test6(i32 %n) nounwind ssp {
; CHECK: @test6 ; CHECK: @test6
entry: entry:
%0 = tail call noalias i8* @malloc(i32 20) nounwind %0 = tail call noalias i8* @malloc(i32 20) nounwind
%1 = tail call i32 @llvm.objectsize.i32(i8* %0, i1 false, i32 0) %1 = tail call i32 @llvm.objectsize.i32(i8* %0, i1 false)
%2 = load i8** @s, align 8 %2 = load i8** @s, align 8
; CHECK-NOT: @llvm.objectsize ; CHECK-NOT: @llvm.objectsize
; CHECK: @__memcpy_chk(i8* %0, i8* %1, i32 30, i32 20) ; CHECK: @__memcpy_chk(i8* %0, i8* %1, i32 30, i32 20)
@@ -153,7 +153,7 @@ define i32 @test7() {
; CHECK: @test7 ; CHECK: @test7
%alloc = call noalias i8* @malloc(i32 48) nounwind %alloc = call noalias i8* @malloc(i32 48) nounwind
%gep = getelementptr inbounds i8* %alloc, i32 16 %gep = getelementptr inbounds i8* %alloc, i32 16
%objsize = call i32 @llvm.objectsize.i32(i8* %gep, i1 false, i32 0) nounwind readonly %objsize = call i32 @llvm.objectsize.i32(i8* %gep, i1 false) nounwind readonly
; CHECK-NEXT: ret i32 32 ; CHECK-NEXT: ret i32 32
ret i32 %objsize ret i32 %objsize
} }
@@ -164,87 +164,7 @@ define i32 @test8() {
; CHECK: @test8 ; CHECK: @test8
%alloc = call noalias i8* @calloc(i32 5, i32 7) nounwind %alloc = call noalias i8* @calloc(i32 5, i32 7) nounwind
%gep = getelementptr inbounds i8* %alloc, i32 5 %gep = getelementptr inbounds i8* %alloc, i32 5
%objsize = call i32 @llvm.objectsize.i32(i8* %gep, i1 false, i32 0) nounwind readonly %objsize = call i32 @llvm.objectsize.i32(i8* %gep, i1 false) nounwind readonly
; CHECK-NEXT: ret i32 30 ; CHECK-NEXT: ret i32 30
ret i32 %objsize ret i32 %objsize
} }
; CHECK: @test9
define i32 @test9(i32 %x, i32 %y) nounwind {
%a = alloca [3 x [4 x double]], align 8
%1 = getelementptr inbounds [3 x [4 x double]]* %a, i32 0, i32 %x
%2 = getelementptr inbounds [4 x double]* %1, i32 0, i32 %y
%3 = bitcast double* %2 to i8*
%objsize = call i32 @llvm.objectsize.i32(i8* %3, i1 false, i32 2)
ret i32 %objsize
; CHECK-NEXT: shl i32 %x, 5
; CHECK-NEXT: shl i32 %y, 3
; CHECK-NEXT: add i32
; CHECK-NEXT: sub i32 96,
; CHECK-NEXT: icmp ugt i32 {{.*}}, 96
; CHECK-NEXT: select i1 {{.*}}, i32 0,
}
; CHECK: @test10
define i32 @test10(i32 %x, i32 %y) nounwind {
%alloc = call noalias i8* @calloc(i32 %x, i32 %y) nounwind
%gep = getelementptr inbounds i8* %alloc, i32 5
%objsize = call i32 @llvm.objectsize.i32(i8* %gep, i1 false, i32 2)
ret i32 %objsize
; CHECK-NEXT: mul i32
; CHECK-NEXT: add i32 {{.*}}, -5
; CHECK-NEXT: icmp ult i32 {{.*}}, 5
; CHECK-NEXT: select i1
; CHECK-NEXT: ret
}
; CHECK: @test11
define i32 @test11(i32 %x, i32 %y) nounwind {
%alloc = call i8* @malloc(i32 %x)
%allocd = bitcast i8* %alloc to double*
%gep = getelementptr double* %allocd, i32 %y
%gepi8 = bitcast double* %gep to i8*
%objsize = call i32 @llvm.objectsize.i32(i8* %gepi8, i1 false, i32 2)
ret i32 %objsize
; CHECK-NEXT: shl i32
; CHECK-NEXT: sub i32
; CHECK-NEXT: icmp ugt i32
; CHECK-NEXT: select i1
; CHECK-NEXT: ret
}
; CHECK: @test12
define i32 @test12(i32 %x) nounwind {
%alloc = alloca i32, i32 %x, align 16
%gep = getelementptr i32* %alloc, i32 7
%gepi8 = bitcast i32* %gep to i8*
%objsize = call i32 @llvm.objectsize.i32(i8* %gepi8, i1 false, i32 2)
ret i32 %objsize
; CHECK-NEXT: shl i32
; CHECK-NEXT: add i32 {{.*}}, -28
; CHECK-NEXT: icmp ult i32 {{.*}}, 28
; CHECK-NEXT: select i1
; CHECK-NEXT: ret
}
; CHECK: @test13
define i32 @test13(i32 %x, i32 %y) nounwind {
%alloc = call i8* @calloc(i32 %x, i32 %y)
%alloc2 = call i8* @malloc(i32 %x)
%objsize = call i32 @llvm.objectsize.i32(i8* %alloc, i1 false, i32 1)
%objsize2 = call i32 @llvm.objectsize.i32(i8* %alloc2, i1 false, i32 1)
%add = add i32 %objsize, %objsize2
ret i32 %add
; CHECK: objectsize
; CHECK: objectsize
; CHECK: add
}
; CHECK: @overflow
define i32 @overflow() {
%alloc = call noalias i8* @malloc(i32 21) nounwind
%gep = getelementptr inbounds i8* %alloc, i32 50
%objsize = call i32 @llvm.objectsize.i32(i8* %gep, i1 false, i32 0) nounwind readonly
; CHECK-NEXT: ret i32 0
ret i32 %objsize
}

View File

@@ -11,7 +11,7 @@ declare i8* @stpcpy(i8*, i8*)
declare i8* @__stpcpy_chk(i8*, i8*, i32) nounwind declare i8* @__stpcpy_chk(i8*, i8*, i32) nounwind
declare i32 @llvm.objectsize.i32(i8*, i1, i32) nounwind readonly declare i32 @llvm.objectsize.i32(i8*, i1) nounwind readonly
define i32 @t1() { define i32 @t1() {
; CHECK: @t1 ; CHECK: @t1
@@ -28,7 +28,7 @@ define i32 @t2() {
%target = alloca [1024 x i8] %target = alloca [1024 x i8]
%arg1 = getelementptr [1024 x i8]* %target, i32 0, i32 0 %arg1 = getelementptr [1024 x i8]* %target, i32 0, i32 0
%arg2 = getelementptr [6 x i8]* @hello, i32 0, i32 0 %arg2 = getelementptr [6 x i8]* @hello, i32 0, i32 0
%tmp1 = call i32 @llvm.objectsize.i32(i8* %arg1, i1 false, i32 0) %tmp1 = call i32 @llvm.objectsize.i32(i8* %arg1, i1 false)
%rslt1 = call i8* @__stpcpy_chk(i8* %arg1, i8* %arg2, i32 %tmp1) %rslt1 = call i8* @__stpcpy_chk(i8* %arg1, i8* %arg2, i32 %tmp1)
; CHECK: @__memcpy_chk ; CHECK: @__memcpy_chk
ret i32 0 ret i32 0

View File

@@ -11,7 +11,7 @@ declare i8* @strcpy(i8*, i8*)
declare i8* @__strcpy_chk(i8*, i8*, i32) nounwind declare i8* @__strcpy_chk(i8*, i8*, i32) nounwind
declare i32 @llvm.objectsize.i32(i8*, i1, i32) nounwind readonly declare i32 @llvm.objectsize.i32(i8*, i1) nounwind readonly
; rdar://6839935 ; rdar://6839935
@@ -30,7 +30,7 @@ define i32 @t2() {
%target = alloca [1024 x i8] %target = alloca [1024 x i8]
%arg1 = getelementptr [1024 x i8]* %target, i32 0, i32 0 %arg1 = getelementptr [1024 x i8]* %target, i32 0, i32 0
%arg2 = getelementptr [6 x i8]* @hello, i32 0, i32 0 %arg2 = getelementptr [6 x i8]* @hello, i32 0, i32 0
%tmp1 = call i32 @llvm.objectsize.i32(i8* %arg1, i1 false, i32 0) %tmp1 = call i32 @llvm.objectsize.i32(i8* %arg1, i1 false)
%rslt1 = call i8* @__strcpy_chk(i8* %arg1, i8* %arg2, i32 %tmp1) %rslt1 = call i8* @__strcpy_chk(i8* %arg1, i8* %arg2, i32 %tmp1)
; CHECK: @__memcpy_chk ; CHECK: @__memcpy_chk
ret i32 0 ret i32 0