From 5b7cfb02f7186dde37234e692a5f848adb68cc18 Mon Sep 17 00:00:00 2001 From: Duncan Sands Date: Mon, 2 Mar 2009 09:18:21 +0000 Subject: [PATCH] Fix PR3694: add an instcombine micro-optimization that helps clean up when using variable length arrays in llvm-gcc. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@65832 91177308-0d34-0410-b5e6-96231b3b80d8 --- .../Scalar/InstructionCombining.cpp | 22 ++++++++++++++----- .../2009-03-02-VarLengthArrayGEP.ll | 14 ++++++++++++ 2 files changed, 31 insertions(+), 5 deletions(-) create mode 100644 test/Transforms/InstCombine/2009-03-02-VarLengthArrayGEP.ll diff --git a/lib/Transforms/Scalar/InstructionCombining.cpp b/lib/Transforms/Scalar/InstructionCombining.cpp index 8e91b36d5d1..7ddbb4ca7cb 100644 --- a/lib/Transforms/Scalar/InstructionCombining.cpp +++ b/lib/Transforms/Scalar/InstructionCombining.cpp @@ -10766,15 +10766,25 @@ Instruction *InstCombiner::visitGetElementPtrInst(GetElementPtrInst &GEP) { // transform: GEP (bitcast [10 x i8]* X to [0 x i8]*), i32 0, ... // into : GEP [10 x i8]* X, i32 0, ... // + // Likewise, transform: GEP (bitcast i8* X to [0 x i8]*), i32 0, ... + // into : GEP i8* X, ... + // // This occurs when the program declares an array extern like "int X[];" - // const PointerType *CPTy = cast(PtrOp->getType()); const PointerType *XTy = cast(X->getType()); - if (const ArrayType *XATy = - dyn_cast(XTy->getElementType())) - if (const ArrayType *CATy = - dyn_cast(CPTy->getElementType())) + if (const ArrayType *CATy = + dyn_cast(CPTy->getElementType())) { + // GEP (bitcast i8* X to [0 x i8]*), i32 0, ... ? + if (CATy->getElementType() == XTy->getElementType()) { + // -> GEP i8* X, ... + SmallVector Indices(GEP.idx_begin()+1, GEP.idx_end()); + return GetElementPtrInst::Create(X, Indices.begin(), Indices.end(), + GEP.getName()); + } else if (const ArrayType *XATy = + dyn_cast(XTy->getElementType())) { + // GEP (bitcast [10 x i8]* X to [0 x i8]*), i32 0, ... ? if (CATy->getElementType() == XATy->getElementType()) { + // -> GEP [10 x i8]* X, i32 0, ... // At this point, we know that the cast source type is a pointer // to an array of the same type as the destination pointer // array. Because the array type is never stepped over (there @@ -10782,6 +10792,8 @@ Instruction *InstCombiner::visitGetElementPtrInst(GetElementPtrInst &GEP) { GEP.setOperand(0, X); return &GEP; } + } + } } else if (GEP.getNumOperands() == 2) { // Transform things like: // %t = getelementptr i32* bitcast ([2 x i32]* %str to i32*), i32 %V diff --git a/test/Transforms/InstCombine/2009-03-02-VarLengthArrayGEP.ll b/test/Transforms/InstCombine/2009-03-02-VarLengthArrayGEP.ll new file mode 100644 index 00000000000..1e136f5eab3 --- /dev/null +++ b/test/Transforms/InstCombine/2009-03-02-VarLengthArrayGEP.ll @@ -0,0 +1,14 @@ +; RUN: llvm-as < %s | opt -instcombine | llvm-dis | grep {getelementptr i32} +; PR3694 + +define i32 @e(i32 %m, i32 %n) nounwind { +entry: + %0 = alloca i32, i32 %n, align 4 ; [#uses=2] + %1 = bitcast i32* %0 to [0 x i32]* ; <[0 x i32]*> [#uses=1] + call void @f(i32* %0) nounwind + %2 = getelementptr [0 x i32]* %1, i32 0, i32 %m ; [#uses=1] + %3 = load i32* %2, align 4 ; [#uses=1] + ret i32 %3 +} + +declare void @f(i32*) -- 2.34.1