diff options
Diffstat (limited to 'llvm/lib/Transforms/IPO/SimplifyLibCalls.cpp')
-rw-r--r-- | llvm/lib/Transforms/IPO/SimplifyLibCalls.cpp | 234 |
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) |