diff options
| author | Chris Lattner <sabre@nondot.org> | 2008-11-12 08:55:54 +0000 | 
|---|---|---|
| committer | Chris Lattner <sabre@nondot.org> | 2008-11-12 08:55:54 +0000 | 
| commit | 3fd91f83fd38d4e86dfd46a4d0e73c0db1731d39 (patch) | |
| tree | 7d5f81a38d3c896454499f59aad92a93370a9fe9 /clang/lib/CodeGen | |
| parent | 35710d18fe769a73d0661bad23ba9b6883d57ecc (diff) | |
| download | bcm5719-llvm-3fd91f83fd38d4e86dfd46a4d0e73c0db1731d39.tar.gz bcm5719-llvm-3fd91f83fd38d4e86dfd46a4d0e73c0db1731d39.zip | |
Lower ?: into select when the selected values are cheap and side-effect-free.
This cuts another 200 lines off expr.ll, forming 23 selects.
llvm-svn: 59124
Diffstat (limited to 'clang/lib/CodeGen')
| -rw-r--r-- | clang/lib/CodeGen/CGExprScalar.cpp | 37 | 
1 files changed, 37 insertions, 0 deletions
| diff --git a/clang/lib/CodeGen/CGExprScalar.cpp b/clang/lib/CodeGen/CGExprScalar.cpp index a3964ee2fc2..918d3bbde89 100644 --- a/clang/lib/CodeGen/CGExprScalar.cpp +++ b/clang/lib/CodeGen/CGExprScalar.cpp @@ -1119,6 +1119,30 @@ Value *ScalarExprEmitter::VisitBinComma(const BinaryOperator *E) {  //                             Other Operators  //===----------------------------------------------------------------------===// +/// isCheapEnoughToEvaluateUnconditionally - Return true if the specified +/// expression is cheap enough and side-effect-free enough to evaluate +/// unconditionally instead of conditionally.  This is used to convert control +/// flow into selects in some cases. +static bool isCheapEnoughToEvaluateUnconditionally(const Expr *E) { +  if (const ParenExpr *PE = dyn_cast<ParenExpr>(E)) +    return isCheapEnoughToEvaluateUnconditionally(PE->getSubExpr()); +   +  // TODO: Allow anything we can constant fold to an integer or fp constant. +  if (isa<IntegerLiteral>(E) || isa<CharacterLiteral>(E) || +      isa<FloatingLiteral>(E)) +    return true; +   +  // Non-volatile automatic variables too, to get "cond ? X : Y" where +  // X and Y are local variables. +  if (const DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E)) +    if (const VarDecl *VD = dyn_cast<VarDecl>(DRE->getDecl())) +      if (VD->hasLocalStorage() && !VD->getType().isVolatileQualified()) +        return true; +   +  return false; +} + +  Value *ScalarExprEmitter::  VisitConditionalOperator(const ConditionalOperator *E) {    // If the condition constant folds and can be elided, try to avoid emitting @@ -1136,6 +1160,19 @@ VisitConditionalOperator(const ConditionalOperator *E) {        return Visit(Live);    } +   +  // If this is a really simple expression (like x ? 4 : 5), emit this as a +  // select instead of as control flow.  We can only do this if it is cheap and +  // safe to  +  if (E->getLHS() && isCheapEnoughToEvaluateUnconditionally(E->getLHS()) && +      isCheapEnoughToEvaluateUnconditionally(E->getRHS())) { +    llvm::Value *CondV = CGF.EvaluateExprAsBool(E->getCond()); +    llvm::Value *LHS = Visit(E->getLHS()); +    llvm::Value *RHS = Visit(E->getRHS()); +    return Builder.CreateSelect(CondV, LHS, RHS, "cond"); +  } +   +      llvm::BasicBlock *LHSBlock = CGF.createBasicBlock("cond.?");    llvm::BasicBlock *RHSBlock = CGF.createBasicBlock("cond.:");    llvm::BasicBlock *ContBlock = CGF.createBasicBlock("cond.cont"); | 

