mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-01 15:33:33 +00:00
050d187bc8
x86 backend. This sketches out a new code path for vector lowering, hidden behind an off-by-default flag while it is under development. The fundamental idea behind the new code path is to aggressively break down the problem space in ways that ease selecting the odd set of instructions available on x86, and carefully avoid scalarizing code even when forced to use older ISAs. Notably, this starts off restricting itself to SSE2 and implements the complete vector shuffle and blend space for 128-bit vectors in SSE2 without scalarizing. The plan is to layer on top of this ISA extensions where we can bail out of the complex SSE2 lowering and opt for a cheaper, specialized instruction (or set of instructions). It also needs to be generalized to AVX and AVX512 vector widths. Currently, this does a decent but not perfect job for SSE2. There are some specific shortcomings that I plan to address: - We need a peephole combine to fold together shuffles where possible. There are cases where a previous shuffle could be modified slightly to arrange for elements to be in the correct position and a later shuffle eliminated. Doing this eagerly added quite a bit of complexity, and so my plan is to combine away these redundancies afterward. - There are a lot more clever ways to use unpck and pack that need to be added. This is essential for real world shuffles as it turns out... Once SSE2 is polished a bit I should be able to get interesting numbers on performance improvements on benchmarks conducive to vectorization. All of this will be off by default until it is functionally equivalent of course. Differential Revision: http://reviews.llvm.org/D4225 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@211888 91177308-0d34-0410-b5e6-96231b3b80d8
220 lines
9.3 KiB
LLVM
220 lines
9.3 KiB
LLVM
; RUN: llc < %s -mtriple=x86_64-unknown-unknown -mcpu=x86-64 -x86-experimental-vector-shuffle-lowering | FileCheck %s --check-prefix=CHECK-SSE2
|
|
|
|
target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128"
|
|
target triple = "x86_64-unknown-unknown"
|
|
|
|
define <2 x i64> @shuffle_v2i64_00(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_00
|
|
; CHECK-SSE2: pshufd {{.*}} # xmm0 = xmm0[0,1,0,1]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 0, i32 0>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_10(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_10
|
|
; CHECK-SSE2: pshufd {{.*}} # xmm0 = xmm0[2,3,0,1]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 1, i32 0>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_11(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_11
|
|
; CHECK-SSE2: pshufd {{.*}} # xmm0 = xmm0[2,3,2,3]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 1, i32 1>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_22(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_22
|
|
; CHECK-SSE2: pshufd {{.*}} # xmm0 = xmm1[0,1,0,1]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 2, i32 2>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_32(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_32
|
|
; CHECK-SSE2: pshufd {{.*}} # xmm0 = xmm1[2,3,0,1]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 3, i32 2>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_33(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_33
|
|
; CHECK-SSE2: pshufd {{.*}} # xmm0 = xmm1[2,3,2,3]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 3, i32 3>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
|
|
define <2 x double> @shuffle_v2f64_00(<2 x double> %a, <2 x double> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2f64_00
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm0 = xmm0[0,0]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x double> %a, <2 x double> %b, <2 x i32> <i32 0, i32 0>
|
|
ret <2 x double> %shuffle
|
|
}
|
|
define <2 x double> @shuffle_v2f64_10(<2 x double> %a, <2 x double> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2f64_10
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm0 = xmm0[1,0]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x double> %a, <2 x double> %b, <2 x i32> <i32 1, i32 0>
|
|
ret <2 x double> %shuffle
|
|
}
|
|
define <2 x double> @shuffle_v2f64_11(<2 x double> %a, <2 x double> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2f64_11
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm0 = xmm0[1,1]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x double> %a, <2 x double> %b, <2 x i32> <i32 1, i32 1>
|
|
ret <2 x double> %shuffle
|
|
}
|
|
define <2 x double> @shuffle_v2f64_22(<2 x double> %a, <2 x double> %b) {
|
|
; FIXME: Should these use movapd + shufpd to remove a domain change at the cost
|
|
; of a mov?
|
|
;
|
|
; CHECK-SSE2-LABEL: @shuffle_v2f64_22
|
|
; CHECK-SSE2: pshufd {{.*}} # xmm0 = xmm1[0,1,0,1]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x double> %a, <2 x double> %b, <2 x i32> <i32 2, i32 2>
|
|
ret <2 x double> %shuffle
|
|
}
|
|
define <2 x double> @shuffle_v2f64_32(<2 x double> %a, <2 x double> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2f64_32
|
|
; CHECK-SSE2: pshufd {{.*}} # xmm0 = xmm1[2,3,0,1]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x double> %a, <2 x double> %b, <2 x i32> <i32 3, i32 2>
|
|
ret <2 x double> %shuffle
|
|
}
|
|
define <2 x double> @shuffle_v2f64_33(<2 x double> %a, <2 x double> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2f64_33
|
|
; CHECK-SSE2: pshufd {{.*}} # xmm0 = xmm1[2,3,2,3]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x double> %a, <2 x double> %b, <2 x i32> <i32 3, i32 3>
|
|
ret <2 x double> %shuffle
|
|
}
|
|
|
|
|
|
define <2 x i64> @shuffle_v2i64_02(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_02
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm0 = xmm0[0],xmm1[0]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 0, i32 2>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_02_copy(<2 x i64> %nonce, <2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_02_copy
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm1 = xmm1[0],xmm2[0]
|
|
; CHECK-SSE2-NEXT: movapd %xmm1, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 0, i32 2>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_03(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_03
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm0 = xmm0[0],xmm1[1]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 0, i32 3>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_03_copy(<2 x i64> %nonce, <2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_03_copy
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm1 = xmm1[0],xmm2[1]
|
|
; CHECK-SSE2-NEXT: movapd %xmm1, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 0, i32 3>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_12(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_12
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm0 = xmm0[1],xmm1[0]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 1, i32 2>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_12_copy(<2 x i64> %nonce, <2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_12_copy
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm1 = xmm1[1],xmm2[0]
|
|
; CHECK-SSE2-NEXT: movapd %xmm1, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 1, i32 2>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_13(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_13
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm0 = xmm0[1],xmm1[1]
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 1, i32 3>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_13_copy(<2 x i64> %nonce, <2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_13_copy
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm1 = xmm1[1],xmm2[1]
|
|
; CHECK-SSE2-NEXT: movapd %xmm1, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 1, i32 3>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_20(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_20
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm1 = xmm1[0],xmm0[0]
|
|
; CHECK-SSE2-NEXT: movapd %xmm1, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 2, i32 0>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_20_copy(<2 x i64> %nonce, <2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_20_copy
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm2 = xmm2[0],xmm1[0]
|
|
; CHECK-SSE2-NEXT: movapd %xmm2, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 2, i32 0>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_21(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_21
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm1 = xmm1[0],xmm0[1]
|
|
; CHECK-SSE2-NEXT: movapd %xmm1, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 2, i32 1>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_21_copy(<2 x i64> %nonce, <2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_21_copy
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm2 = xmm2[0],xmm1[1]
|
|
; CHECK-SSE2-NEXT: movapd %xmm2, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 2, i32 1>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_30(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_30
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm1 = xmm1[1],xmm0[0]
|
|
; CHECK-SSE2-NEXT: movapd %xmm1, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 3, i32 0>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_30_copy(<2 x i64> %nonce, <2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_30_copy
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm2 = xmm2[1],xmm1[0]
|
|
; CHECK-SSE2-NEXT: movapd %xmm2, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 3, i32 0>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_31(<2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_31
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm1 = xmm1[1],xmm0[1]
|
|
; CHECK-SSE2-NEXT: movapd %xmm1, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 3, i32 1>
|
|
ret <2 x i64> %shuffle
|
|
}
|
|
define <2 x i64> @shuffle_v2i64_31_copy(<2 x i64> %nonce, <2 x i64> %a, <2 x i64> %b) {
|
|
; CHECK-SSE2-LABEL: @shuffle_v2i64_31_copy
|
|
; CHECK-SSE2: shufpd {{.*}} # xmm2 = xmm2[1],xmm1[1]
|
|
; CHECK-SSE2-NEXT: movapd %xmm2, %xmm0
|
|
; CHECK-SSE2-NEXT: retq
|
|
%shuffle = shufflevector <2 x i64> %a, <2 x i64> %b, <2 x i32> <i32 3, i32 1>
|
|
ret <2 x i64> %shuffle
|
|
}
|