Disable inlining between sanitized and non-sanitized functions.

Inlining between functions with different values of sanitize_* attributes
leads to over- or under-sanitizing, which is always bad.

llvm-svn: 187967
This commit is contained in:
Evgeniy Stepanov 2013-08-08 08:22:39 +00:00
parent 056b647d1f
commit 2ad3698b70
2 changed files with 135 additions and 5 deletions

View File

@ -1171,6 +1171,22 @@ InlineCost InlineCostAnalysis::getInlineCost(CallSite CS, int Threshold) {
return getInlineCost(CS, CS.getCalledFunction(), Threshold); return getInlineCost(CS, CS.getCalledFunction(), Threshold);
} }
/// \brief Test that two functions either have or have not the given attribute
/// at the same time.
static bool attributeMatches(Function *F1, Function *F2,
Attribute::AttrKind Attr) {
return F1->hasFnAttribute(Attr) == F2->hasFnAttribute(Attr);
}
/// \brief Test that there are no attribute conflicts between Caller and Callee
/// that prevent inlining.
static bool functionsHaveCompatibleAttributes(Function *Caller,
Function *Callee) {
return attributeMatches(Caller, Callee, Attribute::SanitizeAddress) &&
attributeMatches(Caller, Callee, Attribute::SanitizeMemory) &&
attributeMatches(Caller, Callee, Attribute::SanitizeThread);
}
InlineCost InlineCostAnalysis::getInlineCost(CallSite CS, Function *Callee, InlineCost InlineCostAnalysis::getInlineCost(CallSite CS, Function *Callee,
int Threshold) { int Threshold) {
// Cannot inline indirect calls. // Cannot inline indirect calls.
@ -1179,20 +1195,22 @@ InlineCost InlineCostAnalysis::getInlineCost(CallSite CS, Function *Callee,
// Calls to functions with always-inline attributes should be inlined // Calls to functions with always-inline attributes should be inlined
// whenever possible. // whenever possible.
if (Callee->getAttributes().hasAttribute(AttributeSet::FunctionIndex, if (Callee->hasFnAttribute(Attribute::AlwaysInline)) {
Attribute::AlwaysInline)) {
if (isInlineViable(*Callee)) if (isInlineViable(*Callee))
return llvm::InlineCost::getAlways(); return llvm::InlineCost::getAlways();
return llvm::InlineCost::getNever(); return llvm::InlineCost::getNever();
} }
// Never inline functions with conflicting attributes (unless callee has
// always-inline attribute).
if (!functionsHaveCompatibleAttributes(CS.getCaller(), Callee))
return llvm::InlineCost::getNever();
// Don't inline functions which can be redefined at link-time to mean // Don't inline functions which can be redefined at link-time to mean
// something else. Don't inline functions marked noinline or call sites // something else. Don't inline functions marked noinline or call sites
// marked noinline. // marked noinline.
if (Callee->mayBeOverridden() || if (Callee->mayBeOverridden() ||
Callee->getAttributes().hasAttribute(AttributeSet::FunctionIndex, Callee->hasFnAttribute(Attribute::NoInline) || CS.isNoInline())
Attribute::NoInline) ||
CS.isNoInline())
return llvm::InlineCost::getNever(); return llvm::InlineCost::getNever();
DEBUG(llvm::dbgs() << " Analyzing call of " << Callee->getName() DEBUG(llvm::dbgs() << " Analyzing call of " << Callee->getName()

View File

@ -0,0 +1,112 @@
; RUN: opt < %s -inline -S | FileCheck %s
target datalayout = "E-p:64:64:64-a0:0:8-f32:32:32-f64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:32:64-v64:64:64-v128:128:128"
define i32 @noattr_callee(i32 %i) {
ret i32 %i
}
define i32 @sanitize_address_callee(i32 %i) sanitize_address {
ret i32 %i
}
define i32 @sanitize_thread_callee(i32 %i) sanitize_thread {
ret i32 %i
}
define i32 @sanitize_memory_callee(i32 %i) sanitize_memory {
ret i32 %i
}
define i32 @alwaysinline_callee(i32 %i) alwaysinline {
ret i32 %i
}
define i32 @alwaysinline_sanitize_address_callee(i32 %i) alwaysinline sanitize_address {
ret i32 %i
}
define i32 @alwaysinline_sanitize_thread_callee(i32 %i) alwaysinline sanitize_thread {
ret i32 %i
}
define i32 @alwaysinline_sanitize_memory_callee(i32 %i) alwaysinline sanitize_memory {
ret i32 %i
}
; Check that:
; * noattr callee is inlined into noattr caller,
; * sanitize_(address|memory|thread) callee is not inlined into noattr caller,
; * alwaysinline callee is always inlined no matter what sanitize_* attributes are present.
define i32 @test_no_sanitize_address(i32 %arg) {
%x1 = call i32 @noattr_callee(i32 %arg)
%x2 = call i32 @sanitize_address_callee(i32 %x1)
%x3 = call i32 @alwaysinline_callee(i32 %x2)
%x4 = call i32 @alwaysinline_sanitize_address_callee(i32 %x3)
ret i32 %x4
; CHECK-LABEL: @test_no_sanitize_address(
; CHECK-NEXT: @sanitize_address_callee
; CHECK-NEXT: ret i32
}
define i32 @test_no_sanitize_memory(i32 %arg) {
%x1 = call i32 @noattr_callee(i32 %arg)
%x2 = call i32 @sanitize_memory_callee(i32 %x1)
%x3 = call i32 @alwaysinline_callee(i32 %x2)
%x4 = call i32 @alwaysinline_sanitize_memory_callee(i32 %x3)
ret i32 %x4
; CHECK-LABEL: @test_no_sanitize_memory(
; CHECK-NEXT: @sanitize_memory_callee
; CHECK-NEXT: ret i32
}
define i32 @test_no_sanitize_thread(i32 %arg) {
%x1 = call i32 @noattr_callee(i32 %arg)
%x2 = call i32 @sanitize_thread_callee(i32 %x1)
%x3 = call i32 @alwaysinline_callee(i32 %x2)
%x4 = call i32 @alwaysinline_sanitize_thread_callee(i32 %x3)
ret i32 %x4
; CHECK-LABEL: @test_no_sanitize_thread(
; CHECK-NEXT: @sanitize_thread_callee
; CHECK-NEXT: ret i32
}
; Check that:
; * noattr callee is not inlined into sanitize_(address|memory|thread) caller,
; * sanitize_(address|memory|thread) callee is inlined into the caller with the same attribute,
; * alwaysinline callee is always inlined no matter what sanitize_* attributes are present.
define i32 @test_sanitize_address(i32 %arg) sanitize_address {
%x1 = call i32 @noattr_callee(i32 %arg)
%x2 = call i32 @sanitize_address_callee(i32 %x1)
%x3 = call i32 @alwaysinline_callee(i32 %x2)
%x4 = call i32 @alwaysinline_sanitize_address_callee(i32 %x3)
ret i32 %x4
; CHECK-LABEL: @test_sanitize_address(
; CHECK-NEXT: @noattr_callee
; CHECK-NEXT: ret i32
}
define i32 @test_sanitize_memory(i32 %arg) sanitize_memory {
%x1 = call i32 @noattr_callee(i32 %arg)
%x2 = call i32 @sanitize_memory_callee(i32 %x1)
%x3 = call i32 @alwaysinline_callee(i32 %x2)
%x4 = call i32 @alwaysinline_sanitize_memory_callee(i32 %x3)
ret i32 %x4
; CHECK-LABEL: @test_sanitize_memory(
; CHECK-NEXT: @noattr_callee
; CHECK-NEXT: ret i32
}
define i32 @test_sanitize_thread(i32 %arg) sanitize_thread {
%x1 = call i32 @noattr_callee(i32 %arg)
%x2 = call i32 @sanitize_thread_callee(i32 %x1)
%x3 = call i32 @alwaysinline_callee(i32 %x2)
%x4 = call i32 @alwaysinline_sanitize_thread_callee(i32 %x3)
ret i32 %x4
; CHECK-LABEL: @test_sanitize_thread(
; CHECK-NEXT: @noattr_callee
; CHECK-NEXT: ret i32
}