summaryrefslogtreecommitdiffstats
path: root/llvm
diff options
context:
space:
mode:
Diffstat (limited to 'llvm')
-rw-r--r--llvm/lib/Transforms/IPO/SimplifyLibCalls.cpp234
1 files changed, 217 insertions, 17 deletions
diff --git a/llvm/lib/Transforms/IPO/SimplifyLibCalls.cpp b/llvm/lib/Transforms/IPO/SimplifyLibCalls.cpp
index 8cc8e124b8f..985d8856945 100644
--- a/llvm/lib/Transforms/IPO/SimplifyLibCalls.cpp
+++ b/llvm/lib/Transforms/IPO/SimplifyLibCalls.cpp
@@ -69,7 +69,8 @@ public:
: func_name(fname)
#ifndef NDEBUG
, stat_name(std::string("simplify-libcalls:")+fname)
- , occurrences(stat_name.c_str(),"Number of calls simplified")
+ , stat_desc(std::string("Number of ")+fname+"(...) calls simplified")
+ , occurrences(stat_name.c_str(),stat_desc.c_str())
#endif
{
// Register this call optimizer in the optlist (a hash_map)
@@ -118,6 +119,7 @@ private:
const char* func_name; ///< Name of the library call we optimize
#ifndef NDEBUG
std::string stat_name; ///< Holder for debug statistic name
+ std::string stat_desc; ///< Holder for debug statistic description
Statistic<> occurrences; ///< debug statistic (-debug-only=simplify-libcalls)
#endif
};
@@ -203,10 +205,69 @@ public:
}
/// @brief Return the *current* module we're working on.
- Module* getModule() { return M; }
+ Module* getModule() const { return M; }
/// @brief Return the *current* target data for the module we're working on.
- TargetData* getTargetData() { return TD; }
+ TargetData* getTargetData() const { return TD; }
+
+ /// @brief Return the size_t type -- syntactic shortcut
+ const Type* getIntPtrType() const { return TD->getIntPtrType(); }
+
+ /// @brief Return a Function* for the fputc libcall
+ Function* get_fputc()
+ {
+ if (!fputc_func)
+ {
+ std::vector<const Type*> args;
+ args.push_back(Type::IntTy);
+ const Type* FILE_type = M->getTypeByName("struct._IO_FILE");
+ if (!FILE_type)
+ FILE_type = M->getTypeByName("struct._FILE");
+ if (!FILE_type)
+ return 0;
+ args.push_back(PointerType::get(FILE_type));
+ FunctionType* fputc_type =
+ FunctionType::get(Type::IntTy, args, false);
+ fputc_func = M->getOrInsertFunction("fputc",fputc_type);
+ }
+ return fputc_func;
+ }
+
+ /// @brief Return a Function* for the fwrite libcall
+ Function* get_fwrite()
+ {
+ if (!fwrite_func)
+ {
+ std::vector<const Type*> args;
+ args.push_back(PointerType::get(Type::SByteTy));
+ args.push_back(TD->getIntPtrType());
+ args.push_back(TD->getIntPtrType());
+ const Type* FILE_type = M->getTypeByName("struct._IO_FILE");
+ if (!FILE_type)
+ FILE_type = M->getTypeByName("struct._FILE");
+ if (!FILE_type)
+ return 0;
+ args.push_back(PointerType::get(FILE_type));
+ FunctionType* fwrite_type =
+ FunctionType::get(TD->getIntPtrType(), args, false);
+ fwrite_func = M->getOrInsertFunction("fwrite",fwrite_type);
+ }
+ return fwrite_func;
+ }
+
+ /// @brief Return a Function* for the sqrt libcall
+ Function* get_sqrt()
+ {
+ if (!sqrt_func)
+ {
+ std::vector<const Type*> args;
+ args.push_back(Type::DoubleTy);
+ FunctionType* sqrt_type =
+ FunctionType::get(Type::DoubleTy, args, false);
+ sqrt_func = M->getOrInsertFunction("sqrt",sqrt_type);
+ }
+ return sqrt_func;
+ }
/// @brief Return a Function* for the strlen libcall
Function* get_strlen()
@@ -245,12 +306,18 @@ private:
{
M = &mod;
TD = &getAnalysis<TargetData>();
+ fputc_func = 0;
+ fwrite_func = 0;
memcpy_func = 0;
+ sqrt_func = 0;
strlen_func = 0;
}
private:
+ Function* fputc_func; ///< Cached fputc function
+ Function* fwrite_func; ///< Cached fwrite function
Function* memcpy_func; ///< Cached llvm.memcpy function
+ Function* sqrt_func; ///< Cached sqrt function
Function* strlen_func; ///< Cached strlen function
Module* M; ///< Cached Module
TargetData* TD; ///< Cached TargetData
@@ -399,7 +466,6 @@ public:
// terminator as well.
len++;
-
// We need to find the end of the destination string. That's where the
// memory is to be moved to. We just generate a call to strlen (further
// optimized in another pass). Note that the SLC.get_strlen() call
@@ -609,7 +675,7 @@ public:
switch (len)
{
case 0:
- // The memcpy is a no-op so just dump its call.
+ // memcpy(d,s,0,a) -> noop
ci->eraseFromParent();
return true;
case 1: castType = Type::SByteTy; break;
@@ -643,6 +709,152 @@ struct MemMoveOptimization : public MemCpyOptimization
} MemMoveOptimizer;
+/// This LibCallOptimization will simplify calls to the "pow" library
+/// function. It looks for cases where the result of pow is well known and
+/// substitutes the appropriate value.
+/// @brief Simplify the pow library function.
+struct PowOptimization : public LibCallOptimization
+{
+public:
+ /// @brief Default Constructor
+ PowOptimization() : LibCallOptimization("pow") {}
+ /// @brief Destructor
+ virtual ~PowOptimization() {}
+
+ /// @brief Make sure that the "pow" function has the right prototype
+ virtual bool ValidateCalledFunction(const Function* f, SimplifyLibCalls& SLC)
+ {
+ // Just make sure this has 2 arguments
+ return (f->arg_size() == 2);
+ }
+
+ /// @brief Perform the pow optimization.
+ virtual bool OptimizeCall(CallInst* ci, SimplifyLibCalls& SLC)
+ {
+ const Type *Ty = cast<Function>(ci->getOperand(0))->getReturnType();
+ Value* base = ci->getOperand(1);
+ Value* expn = ci->getOperand(2);
+ if (ConstantFP *Op1 = dyn_cast<ConstantFP>(base)) {
+ double Op1V = Op1->getValue();
+ if (Op1V == 1.0)
+ {
+ // pow(1.0,x) -> 1.0
+ ci->replaceAllUsesWith(ConstantFP::get(Ty,1.0));
+ ci->eraseFromParent();
+ return true;
+ }
+ }
+ else if (ConstantFP* Op2 = dyn_cast<ConstantFP>(expn))
+ {
+ double Op2V = Op2->getValue();
+ if (Op2V == 0.0)
+ {
+ // pow(x,0.0) -> 1.0
+ ci->replaceAllUsesWith(ConstantFP::get(Ty,1.0));
+ ci->eraseFromParent();
+ return true;
+ }
+ else if (Op2V == 0.5)
+ {
+ // pow(x,0.5) -> sqrt(x)
+ CallInst* sqrt_inst = new CallInst(SLC.get_sqrt(), base,
+ ci->getName()+".pow",ci);
+ ci->replaceAllUsesWith(sqrt_inst);
+ ci->eraseFromParent();
+ return true;
+ }
+ else if (Op2V == 1.0)
+ {
+ // pow(x,1.0) -> x
+ ci->replaceAllUsesWith(base);
+ ci->eraseFromParent();
+ return true;
+ }
+ else if (Op2V == -1.0)
+ {
+ // pow(x,-1.0) -> 1.0/x
+ BinaryOperator* div_inst= BinaryOperator::create(Instruction::Div,
+ ConstantFP::get(Ty,1.0), base, ci->getName()+".pow", ci);
+ ci->replaceAllUsesWith(div_inst);
+ ci->eraseFromParent();
+ return true;
+ }
+ }
+ return false; // opt failed
+ }
+} PowOptimizer;
+
+/// This LibCallOptimization will simplify calls to the "fputs" library
+/// function. It looks for cases where the result of fputs is not used and the
+/// operation can be reduced to something simpler.
+/// @brief Simplify the pow library function.
+struct PutsOptimization : public LibCallOptimization
+{
+public:
+ /// @brief Default Constructor
+ PutsOptimization() : LibCallOptimization("fputs") {}
+
+ /// @brief Destructor
+ virtual ~PutsOptimization() {}
+
+ /// @brief Make sure that the "fputs" function has the right prototype
+ virtual bool ValidateCalledFunction(const Function* f, SimplifyLibCalls& SLC)
+ {
+ // Just make sure this has 2 arguments
+ return (f->arg_size() == 2);
+ }
+
+ /// @brief Perform the fputs optimization.
+ virtual bool OptimizeCall(CallInst* ci, SimplifyLibCalls& SLC)
+ {
+ // If the result is used, none of these optimizations work
+ if (!ci->hasNUses(0))
+ return false;
+
+ // All the optimizations depend on the length of the first argument and the
+ // fact that it is a constant string array. Check that now
+ uint64_t len = 0;
+ if (!getConstantStringLength(ci->getOperand(1), len))
+ return false;
+
+ switch (len)
+ {
+ case 0:
+ // fputs("",F) -> noop
+ break;
+ case 1:
+ {
+ // fputs(s,F) -> fputc(s[0],F) (if s is constant and strlen(s) == 1)
+ Function* fputc_func = SLC.get_fputc();
+ if (!fputc_func)
+ return false;
+ LoadInst* loadi = new LoadInst(ci->getOperand(1),
+ ci->getOperand(1)->getName()+".byte",ci);
+ CastInst* casti = new CastInst(loadi,Type::IntTy,
+ loadi->getName()+".int",ci);
+ new CallInst(fputc_func,casti,ci->getOperand(2),"",ci);
+ break;
+ }
+ default:
+ {
+ // fputs(s,F) -> fwrite(s,1,len,F) (if s is constant and strlen(s) > 1)
+ Function* fwrite_func = SLC.get_fwrite();
+ if (!fwrite_func)
+ return false;
+ std::vector<Value*> parms;
+ parms.push_back(ci->getOperand(1));
+ parms.push_back(ConstantUInt::get(SLC.getIntPtrType(),len));
+ parms.push_back(ConstantUInt::get(SLC.getIntPtrType(),1));
+ parms.push_back(ci->getOperand(2));
+ new CallInst(fwrite_func,parms,"",ci);
+ break;
+ }
+ }
+ ci->eraseFromParent();
+ return true; // success
+ }
+} PutsOptimizer;
+
/// A function to compute the length of a null-terminated constant array of
/// integers. This function can't rely on the size of the constant array
/// because there could be a null terminator in the middle of the array.
@@ -755,7 +967,6 @@ bool getConstantStringLength(Value* V, uint64_t& len )
// * cos(-x) -> cos(x)
//
// exp, expf, expl:
-// * exp(int) -> contant'
// * exp(log(x)) -> x
//
// ffs, ffsl, ffsll:
@@ -768,11 +979,6 @@ bool getConstantStringLength(Value* V, uint64_t& len )
// (only if the fprintf result is not used)
// * fprintf(file,"%c",chr) -> fputc(chr,file)
//
-// fputs: (only if the result is not used)
-// * fputs("",F) -> noop
-// * fputs(s,F) -> fputc(s[0],F) (if s is constant and strlen(s) == 1)
-// * fputs(s,F) -> fwrite(s, 1, len, F) (if s is constant and strlen(s) > 1)
-//
// isascii:
// * isascii(c) -> ((c & ~0x7f) == 0)
//
@@ -798,9 +1004,6 @@ bool getConstantStringLength(Value* V, uint64_t& len )
// (if all arguments are constant and strlen(x) <= l and strlen(y) <= l)
// * memcpy(x,y,1) -> *x - *y
//
-// memcpy:
-// * memcpy(d,s,0,a) -> d
-//
// memmove:
// * memmove(d,s,l,a) -> memcpy(d,s,l,a)
// (if s is a global constant array)
@@ -811,9 +1014,6 @@ bool getConstantStringLength(Value* V, uint64_t& len )
// (for n=1,2,4,8)
//
// pow, powf, powl:
-// * pow(x,-1.0) -> 1.0/x
-// * pow(x,0.5) -> sqrt(x)
-// * pow(cst1,cst2) -> const1**const2
// * pow(exp(x),y) -> exp(x*y)
// * pow(sqrt(x),y) -> pow(x,y*0.5)
// * pow(pow(x,y),z)-> pow(x,y*z)
OpenPOWER on IntegriCloud