aboutsummaryrefslogtreecommitdiffstats
path: root/lib
diff options
context:
space:
mode:
authorDevang Patel <dpatel@apple.com>2007-10-11 17:21:57 +0000
committerDevang Patel <dpatel@apple.com>2007-10-11 17:21:57 +0000
commit6ad3bd1689f2fc9732c982a8b8429b82856858fd (patch)
treee659b315862b3d2c198f74ba92d0679be5dd60b1 /lib
parentc004023b0dc1fe12bdcfcd771d98674ac2d8a41f (diff)
downloadexternal_llvm-6ad3bd1689f2fc9732c982a8b8429b82856858fd.zip
external_llvm-6ad3bd1689f2fc9732c982a8b8429b82856858fd.tar.gz
external_llvm-6ad3bd1689f2fc9732c982a8b8429b82856858fd.tar.bz2
Lower memcpy if it makes sense.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@42864 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib')
-rw-r--r--lib/Transforms/Scalar/InstructionCombining.cpp46
1 files changed, 46 insertions, 0 deletions
diff --git a/lib/Transforms/Scalar/InstructionCombining.cpp b/lib/Transforms/Scalar/InstructionCombining.cpp
index 2311f1e..5ff9714 100644
--- a/lib/Transforms/Scalar/InstructionCombining.cpp
+++ b/lib/Transforms/Scalar/InstructionCombining.cpp
@@ -7669,6 +7669,52 @@ Instruction *InstCombiner::visitCallInst(CallInst &CI) {
MI->setAlignment(ConstantInt::get(Type::Int32Ty, Align));
Changed = true;
}
+
+ // If MemCpyInst length is 1/2/4/8 bytes then replace memcpy with load/store
+ ConstantInt *MemOpLength = dyn_cast<ConstantInt>(CI.getOperand(3));
+ if (isa<MemCpyInst>(MI))
+ if (MemOpLength) {
+ unsigned Size = MemOpLength->getZExtValue();
+ unsigned Align = cast<ConstantInt>(CI.getOperand(4))->getZExtValue();
+ const PointerType *PTy = cast<PointerType>(CI.getOperand(1)->getType());
+ const Type *MTy = PTy->getElementType();
+ PointerType *NewPtrTy = NULL;
+ if (MTy == Type::Int8Ty) {
+ if (Size == 8)
+ NewPtrTy = PointerType::get(Type::Int64Ty);
+ else if (Size == 4)
+ NewPtrTy = PointerType::get(Type::Int32Ty);
+ else if (Size == 2)
+ NewPtrTy = PointerType::get(Type::Int16Ty);
+ else if (Size == 1)
+ NewPtrTy = PointerType::get(Type::Int8Ty);
+ } else if (MTy == Type::Int16Ty) {
+ if (Size == 4)
+ NewPtrTy = PointerType::get(Type::Int64Ty);
+ else if (Size == 2)
+ NewPtrTy = PointerType::get(Type::Int32Ty);
+ else if (Size == 1)
+ NewPtrTy = PointerType::get(Type::Int16Ty);
+ } else if (MTy == Type::Int32Ty) {
+ if (Size == 2)
+ NewPtrTy = PointerType::get(Type::Int64Ty);
+ else if (Size == 1)
+ NewPtrTy = PointerType::get(Type::Int32Ty);
+ } else if (MTy == Type::Int64Ty) {
+ if (Size == 1)
+ NewPtrTy = PointerType::get(Type::Int64Ty);
+ }
+ if (NewPtrTy)
+ {
+ Value *Src = InsertCastBefore(Instruction::BitCast, CI.getOperand(2), NewPtrTy, CI);
+ Value *Dest = InsertCastBefore(Instruction::BitCast, CI.getOperand(1), NewPtrTy, CI);
+ Value *L = new LoadInst(Src, "tmp", false, Align, &CI);
+ Value *NS = new StoreInst(L, Dest, false, Align, &CI);
+ CI.replaceAllUsesWith(NS);
+ Changed = true;
+ return EraseInstFromFunction(CI);
+ }
+ }
} else if (isa<MemSetInst>(MI)) {
unsigned Alignment = GetOrEnforceKnownAlignment(MI->getDest(), TD);
if (MI->getAlignment()->getZExtValue() < Alignment) {