diff options
author | Anton Yartsev <anton.yartsev@gmail.com> | 2011-02-07 02:17:30 +0000 |
---|---|---|
committer | Anton Yartsev <anton.yartsev@gmail.com> | 2011-02-07 02:17:30 +0000 |
commit | 85129b8a8667a0f3b4d96c39f2f4ba5ee2e902df (patch) | |
tree | b5762011ad0371ae59ac0e58ff6259baa8f46190 /clang/lib/CodeGen/CGExprScalar.cpp | |
parent | 202630c6ee80ae7ba13219753b76c5146cb92084 (diff) | |
download | bcm5719-llvm-85129b8a8667a0f3b4d96c39f2f4ba5ee2e902df.tar.gz bcm5719-llvm-85129b8a8667a0f3b4d96c39f2f4ba5ee2e902df.zip |
pre/post ++/-- for AltiVec vectors. (with builtins-ppc-altivec.c failure fixed)
llvm-svn: 125000
Diffstat (limited to 'clang/lib/CodeGen/CGExprScalar.cpp')
-rw-r--r-- | clang/lib/CodeGen/CGExprScalar.cpp | 71 |
1 files changed, 48 insertions, 23 deletions
diff --git a/clang/lib/CodeGen/CGExprScalar.cpp b/clang/lib/CodeGen/CGExprScalar.cpp index 57627be9e39..a46dda94f75 100644 --- a/clang/lib/CodeGen/CGExprScalar.cpp +++ b/clang/lib/CodeGen/CGExprScalar.cpp @@ -302,6 +302,11 @@ public: return EmitScalarPrePostIncDec(E, LV, true, true); } + llvm::Value *EmitAddConsiderOverflowBehavior(const UnaryOperator *E, + llvm::Value *InVal, + llvm::Value *NextVal, + bool IsInc); + llvm::Value *EmitScalarPrePostIncDec(const UnaryOperator *E, LValue LV, bool isInc, bool isPre); @@ -1221,6 +1226,31 @@ Value *ScalarExprEmitter::VisitBlockDeclRefExpr(const BlockDeclRefExpr *E) { //===----------------------------------------------------------------------===// llvm::Value *ScalarExprEmitter:: +EmitAddConsiderOverflowBehavior(const UnaryOperator *E, + llvm::Value *InVal, + llvm::Value *NextVal, bool IsInc) { + switch (CGF.getContext().getLangOptions().getSignedOverflowBehavior()) { + case LangOptions::SOB_Undefined: + return Builder.CreateNSWAdd(InVal, NextVal, IsInc ? "inc" : "dec"); + break; + case LangOptions::SOB_Defined: + return Builder.CreateAdd(InVal, NextVal, IsInc ? "inc" : "dec"); + break; + case LangOptions::SOB_Trapping: + BinOpInfo BinOp; + BinOp.LHS = InVal; + BinOp.RHS = NextVal; + BinOp.Ty = E->getType(); + BinOp.Opcode = BO_Add; + BinOp.E = E; + return EmitOverflowCheckedBinOp(BinOp); + break; + } + assert(false && "Unknown SignedOverflowBehaviorTy"); + return 0; +} + +llvm::Value *ScalarExprEmitter:: EmitScalarPrePostIncDec(const UnaryOperator *E, LValue LV, bool isInc, bool isPre) { @@ -1269,31 +1299,26 @@ EmitScalarPrePostIncDec(const UnaryOperator *E, LValue LV, // An interesting aspect of this is that increment is always true. // Decrement does not have this property. NextVal = llvm::ConstantInt::getTrue(VMContext); + } else if (ValTy->isVectorType()) { + if (ValTy->hasIntegerRepresentation()) { + NextVal = llvm::ConstantInt::get(InVal->getType(), AmountVal); + + NextVal = ValTy->hasSignedIntegerRepresentation() ? + EmitAddConsiderOverflowBehavior(E, InVal, NextVal, isInc) : + Builder.CreateAdd(InVal, NextVal, isInc ? "inc" : "dec"); + } else { + NextVal = Builder.CreateFAdd( + InVal, + llvm::ConstantFP::get(InVal->getType(), AmountVal), + isInc ? "inc" : "dec"); + } } else if (isa<llvm::IntegerType>(InVal->getType())) { NextVal = llvm::ConstantInt::get(InVal->getType(), AmountVal); - - if (!ValTy->isSignedIntegerType()) - // Unsigned integer inc is always two's complement. - NextVal = Builder.CreateAdd(InVal, NextVal, isInc ? "inc" : "dec"); - else { - switch (CGF.getContext().getLangOptions().getSignedOverflowBehavior()) { - case LangOptions::SOB_Undefined: - NextVal = Builder.CreateNSWAdd(InVal, NextVal, isInc ? "inc" : "dec"); - break; - case LangOptions::SOB_Defined: - NextVal = Builder.CreateAdd(InVal, NextVal, isInc ? "inc" : "dec"); - break; - case LangOptions::SOB_Trapping: - BinOpInfo BinOp; - BinOp.LHS = InVal; - BinOp.RHS = NextVal; - BinOp.Ty = E->getType(); - BinOp.Opcode = BO_Add; - BinOp.E = E; - NextVal = EmitOverflowCheckedBinOp(BinOp); - break; - } - } + + NextVal = ValTy->isSignedIntegerType() ? + EmitAddConsiderOverflowBehavior(E, InVal, NextVal, isInc) : + // Unsigned integer inc is always two's complement. + Builder.CreateAdd(InVal, NextVal, isInc ? "inc" : "dec"); } else { // Add the inc/dec to the real part. if (InVal->getType()->isFloatTy()) |