diff options
author | Nick Lewycky <nicholas@mxc.ca> | 2014-05-02 04:11:45 +0000 |
---|---|---|
committer | Nick Lewycky <nicholas@mxc.ca> | 2014-05-02 04:11:45 +0000 |
commit | 26ad3eb69d21232c204e76728936f5845ed67474 (patch) | |
tree | 0fa5fe6e091dff47b3377284c75d1c609570283c /lib/Transforms/Utils | |
parent | 30e4655a8aa326c9356a6e44c516b3717ddf771b (diff) | |
download | llvm-26ad3eb69d21232c204e76728936f5845ed67474.tar.gz llvm-26ad3eb69d21232c204e76728936f5845ed67474.tar.bz2 llvm-26ad3eb69d21232c204e76728936f5845ed67474.tar.xz |
Fold strlen(expr ? "str1" : "str2") to x ? len1 : len2. This fires about 330 times in a bootstrap of clang.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@207828 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/Transforms/Utils')
-rw-r--r-- | lib/Transforms/Utils/SimplifyLibCalls.cpp | 15 |
1 files changed, 15 insertions, 0 deletions
diff --git a/lib/Transforms/Utils/SimplifyLibCalls.cpp b/lib/Transforms/Utils/SimplifyLibCalls.cpp index 903fbb5899..0fd185816f 100644 --- a/lib/Transforms/Utils/SimplifyLibCalls.cpp +++ b/lib/Transforms/Utils/SimplifyLibCalls.cpp @@ -784,10 +784,25 @@ struct StrLenOpt : public LibCallOptimization { if (uint64_t Len = GetStringLength(Src)) return ConstantInt::get(CI->getType(), Len-1); + // strlen(x?"foo":"bars") --> x ? 3 : 4 + if (SelectInst *SI = dyn_cast<SelectInst>(Src)) { + uint64_t LenTrue = GetStringLength(SI->getTrueValue()); + uint64_t LenFalse = GetStringLength(SI->getFalseValue()); + if (LenTrue && LenFalse) { + Context->emitOptimizationRemark( + "simplify-libcalls", *Caller, SI->getDebugLoc(), + "folded strlen(select) to select of constants"); + return B.CreateSelect(SI->getCondition(), + ConstantInt::get(CI->getType(), LenTrue-1), + ConstantInt::get(CI->getType(), LenFalse-1)); + } + } + // strlen(x) != 0 --> *x != 0 // strlen(x) == 0 --> *x == 0 if (isOnlyUsedInZeroEqualityComparison(CI)) return B.CreateZExt(B.CreateLoad(Src, "strlenfirst"), CI->getType()); + return nullptr; } }; |