mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-21 19:32:16 +00:00
f8d179ba76
This adds support for the QPX vector instruction set, which is used by the enhanced A2 cores on the IBM BG/Q supercomputers. QPX vectors are 256 bytes wide, holding 4 double-precision floating-point values. Boolean values, modeled here as <4 x i1> are actually also represented as floating-point values (essentially { -1, 1 } for { false, true }). QPX shares many features with Altivec and VSX, but is distinct from both of them. One major difference is that, instead of adding completely-separate vector registers, QPX vector registers are extensions of the scalar floating-point registers (lane 0 is the corresponding scalar floating-point value). The operations supported on QPX vectors mirrors that supported on the scalar floating-point values (with some additional ones for permutations and logical/comparison operations). I've been maintaining this support out-of-tree, as part of the bgclang project, for several years. This is not the entire bgclang patch set, but is most of the subset that can be cleanly integrated into LLVM proper at this time. Adding this to the LLVM backend is part of my efforts to rebase bgclang to the current LLVM trunk, but is independently useful (especially for codes that use LLVM as a JIT in library form). The assembler/disassembler test coverage is complete. The CodeGen test coverage is not, but I've included some tests, and more will be added as follow-up work. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@230413 91177308-0d34-0410-b5e6-96231b3b80d8
195 lines
4.5 KiB
LLVM
195 lines
4.5 KiB
LLVM
; RUN: llc < %s -mtriple=powerpc64-unknown-linux-gnu -mcpu=a2q -enable-unsafe-fp-math | FileCheck %s
|
|
; RUN: llc < %s -mtriple=powerpc64-unknown-linux-gnu -mcpu=a2q | FileCheck -check-prefix=CHECK-SAFE %s
|
|
target datalayout = "E-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-f128:128:128-v128:128:128-n32:64"
|
|
target triple = "powerpc64-unknown-linux-gnu"
|
|
|
|
declare <4 x double> @llvm.sqrt.v4f64(<4 x double>)
|
|
declare <4 x float> @llvm.sqrt.v4f32(<4 x float>)
|
|
|
|
define <4 x double> @foo(<4 x double> %a, <4 x double> %b) nounwind {
|
|
entry:
|
|
%x = call <4 x double> @llvm.sqrt.v4f64(<4 x double> %b)
|
|
%r = fdiv <4 x double> %a, %x
|
|
ret <4 x double> %r
|
|
|
|
; CHECK-LABEL: @foo
|
|
; CHECK: qvfrsqrte
|
|
; CHECK: qvfmul
|
|
; FIXME: We're currently loading two constants here (1.5 and -1.5), and using
|
|
; an qvfmadd instead of a qvfnmsub
|
|
; CHECK: qvfmadd
|
|
; CHECK: qvfmadd
|
|
; CHECK: qvfmul
|
|
; CHECK: qvfmul
|
|
; CHECK: qvfmadd
|
|
; CHECK: qvfmul
|
|
; CHECK: qvfmul
|
|
; CHECK: blr
|
|
|
|
; CHECK-SAFE-LABEL: @foo
|
|
; CHECK-SAFE: fsqrt
|
|
; CHECK-SAFE: fdiv
|
|
; CHECK-SAFE: blr
|
|
}
|
|
|
|
define <4 x double> @foof(<4 x double> %a, <4 x float> %b) nounwind {
|
|
entry:
|
|
%x = call <4 x float> @llvm.sqrt.v4f32(<4 x float> %b)
|
|
%y = fpext <4 x float> %x to <4 x double>
|
|
%r = fdiv <4 x double> %a, %y
|
|
ret <4 x double> %r
|
|
|
|
; CHECK-LABEL: @foof
|
|
; CHECK: qvfrsqrtes
|
|
; CHECK: qvfmuls
|
|
; FIXME: We're currently loading two constants here (1.5 and -1.5), and using
|
|
; an qvfmadd instead of a qvfnmsubs
|
|
; CHECK: qvfmadds
|
|
; CHECK: qvfmadds
|
|
; CHECK: qvfmuls
|
|
; CHECK: qvfmul
|
|
; CHECK: blr
|
|
|
|
; CHECK-SAFE-LABEL: @foof
|
|
; CHECK-SAFE: fsqrts
|
|
; CHECK-SAFE: fdiv
|
|
; CHECK-SAFE: blr
|
|
}
|
|
|
|
define <4 x float> @food(<4 x float> %a, <4 x double> %b) nounwind {
|
|
entry:
|
|
%x = call <4 x double> @llvm.sqrt.v4f64(<4 x double> %b)
|
|
%y = fptrunc <4 x double> %x to <4 x float>
|
|
%r = fdiv <4 x float> %a, %y
|
|
ret <4 x float> %r
|
|
|
|
; CHECK-LABEL: @food
|
|
; CHECK: qvfrsqrte
|
|
; CHECK: qvfmul
|
|
; FIXME: We're currently loading two constants here (1.5 and -1.5), and using
|
|
; an qvfmadd instead of a qvfnmsub
|
|
; CHECK: qvfmadd
|
|
; CHECK: qvfmadd
|
|
; CHECK: qvfmul
|
|
; CHECK: qvfmul
|
|
; CHECK: qvfmadd
|
|
; CHECK: qvfmul
|
|
; CHECK: qvfrsp
|
|
; CHECK: qvfmuls
|
|
; CHECK: blr
|
|
|
|
; CHECK-SAFE-LABEL: @food
|
|
; CHECK-SAFE: fsqrt
|
|
; CHECK-SAFE: fdivs
|
|
; CHECK-SAFE: blr
|
|
}
|
|
|
|
define <4 x float> @goo(<4 x float> %a, <4 x float> %b) nounwind {
|
|
entry:
|
|
%x = call <4 x float> @llvm.sqrt.v4f32(<4 x float> %b)
|
|
%r = fdiv <4 x float> %a, %x
|
|
ret <4 x float> %r
|
|
|
|
; CHECK-LABEL: @goo
|
|
; CHECK: qvfrsqrtes
|
|
; CHECK: qvfmuls
|
|
; FIXME: We're currently loading two constants here (1.5 and -1.5), and using
|
|
; an qvfmadd instead of a qvfnmsubs
|
|
; CHECK: qvfmadds
|
|
; CHECK: qvfmadds
|
|
; CHECK: qvfmuls
|
|
; CHECK: qvfmuls
|
|
; CHECK: blr
|
|
|
|
; CHECK-SAFE-LABEL: @goo
|
|
; CHECK-SAFE: fsqrts
|
|
; CHECK-SAFE: fdivs
|
|
; CHECK-SAFE: blr
|
|
}
|
|
|
|
define <4 x double> @foo2(<4 x double> %a, <4 x double> %b) nounwind {
|
|
entry:
|
|
%r = fdiv <4 x double> %a, %b
|
|
ret <4 x double> %r
|
|
|
|
; CHECK-LABEL: @foo2
|
|
; CHECK: qvfre
|
|
; CHECK: qvfnmsub
|
|
; CHECK: qvfmadd
|
|
; CHECK: qvfnmsub
|
|
; CHECK: qvfmadd
|
|
; CHECK: qvfmul
|
|
; CHECK: blr
|
|
|
|
; CHECK-SAFE-LABEL: @foo2
|
|
; CHECK-SAFE: fdiv
|
|
; CHECK-SAFE: blr
|
|
}
|
|
|
|
define <4 x float> @goo2(<4 x float> %a, <4 x float> %b) nounwind {
|
|
entry:
|
|
%r = fdiv <4 x float> %a, %b
|
|
ret <4 x float> %r
|
|
|
|
; CHECK-LABEL: @goo2
|
|
; CHECK: qvfres
|
|
; CHECK: qvfnmsubs
|
|
; CHECK: qvfmadds
|
|
; CHECK: qvfmuls
|
|
; CHECK: blr
|
|
|
|
; CHECK-SAFE-LABEL: @goo2
|
|
; CHECK-SAFE: fdivs
|
|
; CHECK-SAFE: blr
|
|
}
|
|
|
|
define <4 x double> @foo3(<4 x double> %a) nounwind {
|
|
entry:
|
|
%r = call <4 x double> @llvm.sqrt.v4f64(<4 x double> %a)
|
|
ret <4 x double> %r
|
|
|
|
; CHECK-LABEL: @foo3
|
|
; CHECK: qvfrsqrte
|
|
; CHECK: qvfmul
|
|
; FIXME: We're currently loading two constants here (1.5 and -1.5), and using
|
|
; an qvfmadd instead of a qvfnmsub
|
|
; CHECK-DAG: qvfmadd
|
|
; CHECK-DAG: qvfcmpeq
|
|
; CHECK-DAG: qvfmadd
|
|
; CHECK-DAG: qvfmul
|
|
; CHECK-DAG: qvfmul
|
|
; CHECK-DAG: qvfmadd
|
|
; CHECK-DAG: qvfmul
|
|
; CHECK-DAG: qvfmul
|
|
; CHECK: qvfsel
|
|
; CHECK: blr
|
|
|
|
; CHECK-SAFE-LABEL: @foo3
|
|
; CHECK-SAFE: fsqrt
|
|
; CHECK-SAFE: blr
|
|
}
|
|
|
|
define <4 x float> @goo3(<4 x float> %a) nounwind {
|
|
entry:
|
|
%r = call <4 x float> @llvm.sqrt.v4f32(<4 x float> %a)
|
|
ret <4 x float> %r
|
|
|
|
; CHECK-LABEL: @goo3
|
|
; CHECK: qvfrsqrtes
|
|
; CHECK: qvfmuls
|
|
; FIXME: We're currently loading two constants here (1.5 and -1.5), and using
|
|
; an qvfmadds instead of a qvfnmsubs
|
|
; CHECK-DAG: qvfmadds
|
|
; CHECK-DAG: qvfcmpeq
|
|
; CHECK-DAG: qvfmadds
|
|
; CHECK-DAG: qvfmuls
|
|
; CHECK-DAG: qvfmuls
|
|
; CHECK: qvfsel
|
|
; CHECK: blr
|
|
|
|
; CHECK-SAFE-LABEL: @goo3
|
|
; CHECK-SAFE: fsqrts
|
|
; CHECK-SAFE: blr
|
|
}
|
|
|