Remove the palignr intrinsics now that we lower them to vector shuffles,

shifts and null vectors. Autoupgrade these to what we'd lower them to.

Add a testcase to exercise this.


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@101851 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Eric Christopher 2010-04-20 00:59:54 +00:00
parent a9056dbec1
commit 6d972fd087
5 changed files with 122 additions and 29 deletions

View File

@ -669,16 +669,6 @@ let TargetPrefix = "x86" in { // All intrinsics start with "llvm.x86.".
Intrinsic<[llvm_v4i32_ty], [llvm_v4i32_ty], [IntrNoMem]>;
}
// Align ops
let TargetPrefix = "x86" in { // All intrinsics start with "llvm.x86.".
def int_x86_ssse3_palign_r :
Intrinsic<[llvm_v1i64_ty], [llvm_v1i64_ty,
llvm_v1i64_ty, llvm_i8_ty], [IntrNoMem]>;
def int_x86_ssse3_palign_r_128 :
Intrinsic<[llvm_v2i64_ty], [llvm_v2i64_ty,
llvm_v2i64_ty, llvm_i8_ty], [IntrNoMem]>;
}
//===----------------------------------------------------------------------===//
// SSE4.1

View File

@ -2939,14 +2939,8 @@ let Constraints = "$src1 = $dst" in {
[]>, OpSize;
}
def : Pat<(int_x86_ssse3_palign_r VR64:$src1, VR64:$src2, (i8 imm:$src3)),
(PALIGNR64rr VR64:$src1, VR64:$src2, (BYTE_imm imm:$src3))>,
Requires<[HasSSSE3]>;
def : Pat<(int_x86_ssse3_palign_r VR64:$src1,
(memop64 addr:$src2),
(i8 imm:$src3)),
(PALIGNR64rm VR64:$src1, addr:$src2, (BYTE_imm imm:$src3))>,
Requires<[HasSSSE3]>;
let AddedComplexity = 5 in {
def : Pat<(v1i64 (palign:$src3 VR64:$src1, VR64:$src2)),
(PALIGNR64rr VR64:$src2, VR64:$src1,
(SHUFFLE_get_palign_imm VR64:$src3))>,
@ -2968,16 +2962,6 @@ def : Pat<(v8i8 (palign:$src3 VR64:$src1, VR64:$src2)),
(SHUFFLE_get_palign_imm VR64:$src3))>,
Requires<[HasSSSE3]>;
def : Pat<(int_x86_ssse3_palign_r_128 VR128:$src1, VR128:$src2, (i8 imm:$src3)),
(PALIGNR128rr VR128:$src1, VR128:$src2, (BYTE_imm imm:$src3))>,
Requires<[HasSSSE3]>;
def : Pat<(int_x86_ssse3_palign_r_128 VR128:$src1,
(memopv2i64 addr:$src2),
(i8 imm:$src3)),
(PALIGNR128rm VR128:$src1, addr:$src2, (BYTE_imm imm:$src3))>,
Requires<[HasSSSE3]>;
let AddedComplexity = 5 in {
def : Pat<(v4i32 (palign:$src3 VR128:$src1, VR128:$src2)),
(PALIGNR128rr VR128:$src2, VR128:$src1,
(SHUFFLE_get_palign_imm VR128:$src3))>,

View File

@ -19,6 +19,7 @@
#include "llvm/IntrinsicInst.h"
#include "llvm/ADT/SmallVector.h"
#include "llvm/Support/ErrorHandling.h"
#include "llvm/Support/IRBuilder.h"
#include <cstring>
using namespace llvm;
@ -277,8 +278,13 @@ static bool UpgradeIntrinsicFunction1(Function *F, Function *&NewFn) {
// Calls to these intrinsics are transformed into vector multiplies.
NewFn = 0;
return true;
} else if (Name.compare(5, 18, "x86.ssse3.palign.r", 18) == 0 ||
Name.compare(5, 22, "x86.ssse3.palign.r.128", 22) == 0) {
// Calls to these intrinsics are transformed into vector shuffles, shifts,
// or 0.
NewFn = 0;
return true;
}
break;
}
@ -420,6 +426,118 @@ void llvm::UpgradeIntrinsicCall(CallInst *CI, Function *NewFn) {
// Remove upgraded multiply.
CI->eraseFromParent();
} else if (F->getName() == "llvm.x86.ssse3.palign.r") {
Value *Op1 = CI->getOperand(1);
Value *Op2 = CI->getOperand(2);
Value *Op3 = CI->getOperand(3);
unsigned shiftVal = cast<ConstantInt>(Op3)->getZExtValue();
Value *Rep;
IRBuilder<> Builder(C);
Builder.SetInsertPoint(CI->getParent(), CI);
// If palignr is shifting the pair of input vectors less than 9 bytes,
// emit a shuffle instruction.
if (shiftVal <= 8) {
const Type *IntTy = Type::getInt32Ty(C);
const Type *EltTy = Type::getInt8Ty(C);
const Type *VecTy = VectorType::get(EltTy, 8);
Op2 = Builder.CreateBitCast(Op2, VecTy);
Op1 = Builder.CreateBitCast(Op1, VecTy);
llvm::SmallVector<llvm::Constant*, 8> Indices;
for (unsigned i = 0; i != 8; ++i)
Indices.push_back(ConstantInt::get(IntTy, shiftVal + i));
Value *SV = ConstantVector::get(Indices.begin(), Indices.size());
Rep = Builder.CreateShuffleVector(Op2, Op1, SV, "palignr");
Rep = Builder.CreateBitCast(Rep, F->getReturnType());
}
// If palignr is shifting the pair of input vectors more than 8 but less
// than 16 bytes, emit a logical right shift of the destination.
else if (shiftVal < 16) {
// MMX has these as 1 x i64 vectors for some odd optimization reasons.
const Type *EltTy = Type::getInt64Ty(C);
const Type *VecTy = VectorType::get(EltTy, 1);
Op1 = Builder.CreateBitCast(Op1, VecTy, "cast");
Op2 = ConstantInt::get(VecTy, (shiftVal-8) * 8);
// create i32 constant
Function *I =
Intrinsic::getDeclaration(F->getParent(), Intrinsic::x86_mmx_psrl_q);
Rep = Builder.CreateCall2(I, Op1, Op2, "palignr");
}
// If palignr is shifting the pair of vectors more than 32 bytes, emit zero.
else {
Rep = Constant::getNullValue(F->getReturnType());
}
// Replace any uses with our new instruction.
if (!CI->use_empty())
CI->replaceAllUsesWith(Rep);
// Remove upgraded instruction.
CI->eraseFromParent();
} else if (F->getName() == "llvm.x86.ssse3.palign.r.128") {
Value *Op1 = CI->getOperand(1);
Value *Op2 = CI->getOperand(2);
Value *Op3 = CI->getOperand(3);
unsigned shiftVal = cast<ConstantInt>(Op3)->getZExtValue();
Value *Rep;
IRBuilder<> Builder(C);
Builder.SetInsertPoint(CI->getParent(), CI);
// If palignr is shifting the pair of input vectors less than 17 bytes,
// emit a shuffle instruction.
if (shiftVal <= 16) {
const Type *IntTy = Type::getInt32Ty(C);
const Type *EltTy = Type::getInt8Ty(C);
const Type *VecTy = VectorType::get(EltTy, 16);
Op2 = Builder.CreateBitCast(Op2, VecTy);
Op1 = Builder.CreateBitCast(Op1, VecTy);
llvm::SmallVector<llvm::Constant*, 16> Indices;
for (unsigned i = 0; i != 16; ++i)
Indices.push_back(ConstantInt::get(IntTy, shiftVal + i));
Value *SV = ConstantVector::get(Indices.begin(), Indices.size());
Rep = Builder.CreateShuffleVector(Op2, Op1, SV, "palignr");
Rep = Builder.CreateBitCast(Rep, F->getReturnType());
}
// If palignr is shifting the pair of input vectors more than 16 but less
// than 32 bytes, emit a logical right shift of the destination.
else if (shiftVal < 32) {
const Type *EltTy = Type::getInt64Ty(C);
const Type *VecTy = VectorType::get(EltTy, 2);
const Type *IntTy = Type::getInt32Ty(C);
Op1 = Builder.CreateBitCast(Op1, VecTy, "cast");
Op2 = ConstantInt::get(IntTy, (shiftVal-16) * 8);
// create i32 constant
Function *I =
Intrinsic::getDeclaration(F->getParent(), Intrinsic::x86_sse2_psrl_dq);
Rep = Builder.CreateCall2(I, Op1, Op2, "palignr");
}
// If palignr is shifting the pair of vectors more than 32 bytes, emit zero.
else {
Rep = Constant::getNullValue(F->getReturnType());
}
// Replace any uses with our new instruction.
if (!CI->use_empty())
CI->replaceAllUsesWith(Rep);
// Remove upgraded instruction.
CI->eraseFromParent();
} else {
llvm_unreachable("Unknown function for CallInst upgrade.");
}

View File

@ -0,0 +1 @@
; RUN: llvm-dis < %s.bc | not grep {@llvm\\.palign}

Binary file not shown.