mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2025-01-09 13:21:30 +00:00
Optimize printf -> iprintf if there are no floating point arguments
and iprintf is available on the target. Currently iprintf is only marked as being available on the XCore. llvm-svn: 126935
This commit is contained in:
parent
ab6450ec63
commit
df829ddcb7
@ -26,6 +26,9 @@ namespace llvm {
|
|||||||
/// void memset_pattern16(void *b, const void *pattern16, size_t len);
|
/// void memset_pattern16(void *b, const void *pattern16, size_t len);
|
||||||
memset_pattern16,
|
memset_pattern16,
|
||||||
|
|
||||||
|
/// int iprintf(const char *format, ...);
|
||||||
|
iprintf,
|
||||||
|
|
||||||
NumLibFuncs
|
NumLibFuncs
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -30,7 +30,10 @@ static void initialize(TargetLibraryInfo &TLI, const Triple &T) {
|
|||||||
// memset_pattern16 is only available on iOS 3.0 and Mac OS/X 10.5 and later.
|
// memset_pattern16 is only available on iOS 3.0 and Mac OS/X 10.5 and later.
|
||||||
if (T.getOS() != Triple::Darwin || T.getDarwinMajorNumber() < 9)
|
if (T.getOS() != Triple::Darwin || T.getDarwinMajorNumber() < 9)
|
||||||
TLI.setUnavailable(LibFunc::memset_pattern16);
|
TLI.setUnavailable(LibFunc::memset_pattern16);
|
||||||
|
|
||||||
|
// iprintf is only available on XCore.
|
||||||
|
if (T.getArch() != Triple::xcore)
|
||||||
|
TLI.setUnavailable(LibFunc::iprintf);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -49,6 +49,7 @@ class LibCallOptimization {
|
|||||||
protected:
|
protected:
|
||||||
Function *Caller;
|
Function *Caller;
|
||||||
const TargetData *TD;
|
const TargetData *TD;
|
||||||
|
const TargetLibraryInfo *TLI;
|
||||||
LLVMContext* Context;
|
LLVMContext* Context;
|
||||||
public:
|
public:
|
||||||
LibCallOptimization() { }
|
LibCallOptimization() { }
|
||||||
@ -62,9 +63,11 @@ public:
|
|||||||
virtual Value *CallOptimizer(Function *Callee, CallInst *CI, IRBuilder<> &B)
|
virtual Value *CallOptimizer(Function *Callee, CallInst *CI, IRBuilder<> &B)
|
||||||
=0;
|
=0;
|
||||||
|
|
||||||
Value *OptimizeCall(CallInst *CI, const TargetData *TD, IRBuilder<> &B) {
|
Value *OptimizeCall(CallInst *CI, const TargetData *TD,
|
||||||
|
const TargetLibraryInfo *TLI, IRBuilder<> &B) {
|
||||||
Caller = CI->getParent()->getParent();
|
Caller = CI->getParent()->getParent();
|
||||||
this->TD = TD;
|
this->TD = TD;
|
||||||
|
this->TLI = TLI;
|
||||||
if (CI->getCalledFunction())
|
if (CI->getCalledFunction())
|
||||||
Context = &CI->getCalledFunction()->getContext();
|
Context = &CI->getCalledFunction()->getContext();
|
||||||
|
|
||||||
@ -97,6 +100,15 @@ static bool IsOnlyUsedInZeroEqualityComparison(Value *V) {
|
|||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool CallHasFloatingPointArgument(const CallInst *CI) {
|
||||||
|
for (CallInst::const_op_iterator it = CI->op_begin(), e = CI->op_end();
|
||||||
|
it != e; ++it) {
|
||||||
|
if ((*it)->getType()->isFloatingPointTy())
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
/// IsOnlyUsedInEqualityComparison - Return true if it is only used in equality
|
/// IsOnlyUsedInEqualityComparison - Return true if it is only used in equality
|
||||||
/// comparisons with With.
|
/// comparisons with With.
|
||||||
@ -1075,14 +1087,8 @@ struct ToAsciiOpt : public LibCallOptimization {
|
|||||||
// 'printf' Optimizations
|
// 'printf' Optimizations
|
||||||
|
|
||||||
struct PrintFOpt : public LibCallOptimization {
|
struct PrintFOpt : public LibCallOptimization {
|
||||||
virtual Value *CallOptimizer(Function *Callee, CallInst *CI, IRBuilder<> &B) {
|
Value *OptimizeFixedFormatString(Function *Callee, CallInst *CI,
|
||||||
// Require one fixed pointer argument and an integer/void result.
|
IRBuilder<> &B) {
|
||||||
const FunctionType *FT = Callee->getFunctionType();
|
|
||||||
if (FT->getNumParams() < 1 || !FT->getParamType(0)->isPointerTy() ||
|
|
||||||
!(FT->getReturnType()->isIntegerTy() ||
|
|
||||||
FT->getReturnType()->isVoidTy()))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
// Check for a fixed format string.
|
// Check for a fixed format string.
|
||||||
std::string FormatStr;
|
std::string FormatStr;
|
||||||
if (!GetConstantStringInfo(CI->getArgOperand(0), FormatStr))
|
if (!GetConstantStringInfo(CI->getArgOperand(0), FormatStr))
|
||||||
@ -1138,6 +1144,32 @@ struct PrintFOpt : public LibCallOptimization {
|
|||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual Value *CallOptimizer(Function *Callee, CallInst *CI, IRBuilder<> &B) {
|
||||||
|
// Require one fixed pointer argument and an integer/void result.
|
||||||
|
const FunctionType *FT = Callee->getFunctionType();
|
||||||
|
if (FT->getNumParams() < 1 || !FT->getParamType(0)->isPointerTy() ||
|
||||||
|
!(FT->getReturnType()->isIntegerTy() ||
|
||||||
|
FT->getReturnType()->isVoidTy()))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (Value *V = OptimizeFixedFormatString(Callee, CI, B)) {
|
||||||
|
return V;
|
||||||
|
}
|
||||||
|
|
||||||
|
// printf(format, ...) -> iprintf(format, ...) if no floating point
|
||||||
|
// arguments.
|
||||||
|
if (TLI->has(LibFunc::iprintf) && !CallHasFloatingPointArgument(CI)) {
|
||||||
|
Module *M = B.GetInsertBlock()->getParent()->getParent();
|
||||||
|
Constant *IPrintFFn =
|
||||||
|
M->getOrInsertFunction("iprintf", FT, Callee->getAttributes());
|
||||||
|
CallInst *New = cast<CallInst>(CI->clone());
|
||||||
|
New->setCalledFunction(IPrintFFn);
|
||||||
|
B.Insert(New);
|
||||||
|
return New;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
//===---------------------------------------===//
|
//===---------------------------------------===//
|
||||||
@ -1545,7 +1577,7 @@ bool SimplifyLibCalls::runOnFunction(Function &F) {
|
|||||||
Builder.SetInsertPoint(BB, I);
|
Builder.SetInsertPoint(BB, I);
|
||||||
|
|
||||||
// Try to optimize this call.
|
// Try to optimize this call.
|
||||||
Value *Result = LCO->OptimizeCall(CI, TD, Builder);
|
Value *Result = LCO->OptimizeCall(CI, TD, TLI, Builder);
|
||||||
if (Result == 0) continue;
|
if (Result == 0) continue;
|
||||||
|
|
||||||
DEBUG(dbgs() << "SimplifyLibCalls simplified: " << *CI;
|
DEBUG(dbgs() << "SimplifyLibCalls simplified: " << *CI;
|
||||||
|
29
test/Transforms/SimplifyLibCalls/iprintf.ll
Normal file
29
test/Transforms/SimplifyLibCalls/iprintf.ll
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
; RUN: opt < %s -simplify-libcalls -S -o %t
|
||||||
|
; RUN: FileCheck < %t %s
|
||||||
|
target datalayout = "e-p:32:32:32-i1:8:32-i8:8:32-i16:16:32-i32:32:32-i64:32:32-f32:32:32-f64:32:32-v64:64:64-v128:128:128-a0:0:32"
|
||||||
|
target triple = "xcore-xmos-elf"
|
||||||
|
|
||||||
|
@.str = internal constant [4 x i8] c"%f\0A\00" ; <[4 x i8]*> [#uses=1]
|
||||||
|
@.str1 = internal constant [4 x i8] c"%d\0A\00" ; <[4 x i8]*> [#uses=1]
|
||||||
|
|
||||||
|
; Verify printf with no floating point arguments is transformed to iprintf
|
||||||
|
define i32 @f0(i32 %x) nounwind {
|
||||||
|
entry:
|
||||||
|
; CHECK: define i32 @f0
|
||||||
|
; CHECK: @iprintf
|
||||||
|
; CHECK: }
|
||||||
|
%0 = tail call i32 (i8*, ...)* @printf(i8* getelementptr ([4 x i8]* @.str1, i32 0, i32 0), i32 %x) ; <i32> [#uses=0]
|
||||||
|
ret i32 %0
|
||||||
|
}
|
||||||
|
|
||||||
|
; Verify we don't turn this into an iprintf call
|
||||||
|
define void @f1(double %x) nounwind {
|
||||||
|
entry:
|
||||||
|
; CHECK: define void @f1
|
||||||
|
; CHECK: @printf
|
||||||
|
; CHECK: }
|
||||||
|
%0 = tail call i32 (i8*, ...)* @printf(i8* getelementptr ([4 x i8]* @.str, i32 0, i32 0), double %x) nounwind ; <i32> [#uses=0]
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
declare i32 @printf(i8* nocapture, ...) nounwind
|
Loading…
Reference in New Issue
Block a user