diff options
| author | Chandler Carruth <chandlerc@gmail.com> | 2014-08-15 17:42:00 +0000 |
|---|---|---|
| committer | Chandler Carruth <chandlerc@gmail.com> | 2014-08-15 17:42:00 +0000 |
| commit | 03f456abbe523fdaeb7243ffe7ac14036f069265 (patch) | |
| tree | 14bfdb1bf9ee46d2141292d826bb245e9da21528 /llvm/lib/Target/X86/X86ISelLowering.cpp | |
| parent | 4eda2c00861cfdbc20b64b623a8fa32f03cbe699 (diff) | |
| download | bcm5719-llvm-03f456abbe523fdaeb7243ffe7ac14036f069265.tar.gz bcm5719-llvm-03f456abbe523fdaeb7243ffe7ac14036f069265.zip | |
[x86] Teach the new AVX v4f64 shuffle lowering to use UNPCK instructions
where applicable for blending.
llvm-svn: 215737
Diffstat (limited to 'llvm/lib/Target/X86/X86ISelLowering.cpp')
| -rw-r--r-- | llvm/lib/Target/X86/X86ISelLowering.cpp | 42 |
1 files changed, 42 insertions, 0 deletions
diff --git a/llvm/lib/Target/X86/X86ISelLowering.cpp b/llvm/lib/Target/X86/X86ISelLowering.cpp index 327cc296fe6..b31aa446998 100644 --- a/llvm/lib/Target/X86/X86ISelLowering.cpp +++ b/llvm/lib/Target/X86/X86ISelLowering.cpp @@ -7055,6 +7055,35 @@ static bool isSingleInputShuffleMask(ArrayRef<int> Mask) { return true; } +/// \brief Implementation of the \c isShuffleEquivalent variadic functor. +/// +/// See its documentation for details. +static bool isShuffleEquivalentImpl(ArrayRef<int> Mask, + ArrayRef<const int *> Args) { + if (Mask.size() != Args.size()) + return false; + for (int i = 0, e = Mask.size(); i < e; ++i) { + assert(*Args[i] >= 0 && "Arguments must be positive integers!"); + assert(*Args[i] < (int)Args.size() * 2 && + "Argument outside the range of possible shuffle inputs!"); + if (Mask[i] != -1 && Mask[i] != *Args[i]) + return false; + } + return true; +} +/// \brief Checks whether a shuffle mask is equivalent to an explicit list of +/// arguments. +/// +/// This is a fast way to test a shuffle mask against a fixed pattern: +/// +/// if (isShuffleEquivalent(Mask, 3, 2, 1, 0)) { ... } +/// +/// It returns true if the mask is exactly as wide as the argument list, and +/// each element of the mask is either -1 (signifying undef) or the value given +/// in the argument. +static const VariadicFunction1< + bool, ArrayRef<int>, int, isShuffleEquivalentImpl> isShuffleEquivalent = {}; + /// \brief Get a 4-lane 8-bit shuffle immediate for a mask. /// /// This helper function produces an 8-bit shuffle immediate corresponding to @@ -8440,6 +8469,19 @@ static SDValue lowerV4F64VectorShuffle(SDValue Op, SDValue V1, SDValue V2, DAG.getConstant(VPERMILPMask, MVT::i8)); } + // X86 has dedicated unpack instructions that can handle specific blend + // operations: UNPCKH and UNPCKL. + if (isShuffleEquivalent(Mask, 0, 4, 2, 6)) + return DAG.getNode(X86ISD::UNPCKL, DL, MVT::v4f64, V1, V2); + if (isShuffleEquivalent(Mask, 1, 5, 3, 7)) + return DAG.getNode(X86ISD::UNPCKH, DL, MVT::v4f64, V1, V2); + // FIXME: It would be nice to find a way to get canonicalization to commute + // these patterns. + if (isShuffleEquivalent(Mask, 4, 0, 6, 2)) + return DAG.getNode(X86ISD::UNPCKL, DL, MVT::v4f64, V2, V1); + if (isShuffleEquivalent(Mask, 5, 1, 7, 3)) + return DAG.getNode(X86ISD::UNPCKH, DL, MVT::v4f64, V2, V1); + // Check if the blend happens to exactly fit that of SHUFPD. if (Mask[0] < 4 && (Mask[1] == -1 || Mask[1] >= 4) && Mask[2] < 4 && (Mask[3] == -1 || Mask[3] >= 4)) { |

