summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorBradley Smith <bradley.smith@arm.com>2015-03-23 15:59:54 +0000
committerBradley Smith <bradley.smith@arm.com>2015-03-23 15:59:54 +0000
commitbc0f0d8c49232506b2985304feb56fa5e982289a (patch)
tree5bbf41e393e2df35b8db9ae9187a6fe625769a27
parente13f47e8aa38a977c7dbfa950c4fb55ffdad5f1c (diff)
downloadbcm5719-llvm-bc0f0d8c49232506b2985304feb56fa5e982289a.tar.gz
bcm5719-llvm-bc0f0d8c49232506b2985304feb56fa5e982289a.zip
[ARM] Add more pattern matching for f16 <-> f64 conversions
Specifically when the conversion is done in two steps, f16 -> f32 -> f64. For example: %1 = tail call float @llvm.convert.from.fp16.f32(i16 %0) %conv = fpext float %1 to double to: vcvtb.f64.f16 llvm-svn: 232954
-rw-r--r--llvm/lib/Target/ARM/ARMInstrVFP.td8
-rw-r--r--llvm/test/CodeGen/ARM/fp16-64.ll31
2 files changed, 39 insertions, 0 deletions
diff --git a/llvm/lib/Target/ARM/ARMInstrVFP.td b/llvm/lib/Target/ARM/ARMInstrVFP.td
index e0a93149916..cc8a2b0b4c6 100644
--- a/llvm/lib/Target/ARM/ARMInstrVFP.td
+++ b/llvm/lib/Target/ARM/ARMInstrVFP.td
@@ -627,6 +627,14 @@ def : Pat<(f16_to_fp GPR:$a),
def : Pat<(f64 (f16_to_fp GPR:$a)),
(VCVTBHD (COPY_TO_REGCLASS GPR:$a, SPR))>;
+def : Pat<(f64 (fextend (f16_to_fp GPR:$a))),
+ (VCVTBHD (COPY_TO_REGCLASS GPR:$a, SPR))>,
+ Requires<[HasFPARMv8, HasDPVFP]>;
+
+def : Pat<(fp_to_f16 (fround (f64 DPR:$a))),
+ (i32 (COPY_TO_REGCLASS (VCVTBDH DPR:$a), GPR))>,
+ Requires<[HasFPARMv8, HasDPVFP]>;
+
multiclass vcvt_inst<string opc, bits<2> rm,
SDPatternOperator node = null_frag> {
let PostEncoderMethod = "", DecoderNamespace = "VFPV8" in {
diff --git a/llvm/test/CodeGen/ARM/fp16-64.ll b/llvm/test/CodeGen/ARM/fp16-64.ll
new file mode 100644
index 00000000000..854ba9adb84
--- /dev/null
+++ b/llvm/test/CodeGen/ARM/fp16-64.ll
@@ -0,0 +1,31 @@
+; RUN: llc -mtriple=arm -mattr=+fp-armv8 < %s | \
+; RUN: FileCheck --check-prefix=CHECK --check-prefix=V8 %s
+; RUN: llc -mtriple=arm -mattr=+vfp3,+d16 < %s | \
+; RUN: FileCheck --check-prefix=CHECK --check-prefix=NOV8 %s
+
+declare float @llvm.convert.from.fp16.f32(i16) nounwind readnone
+declare i16 @llvm.convert.to.fp16.f32(float) nounwind readnone
+
+define void @vcvt_f64_f16(i16* %x, double* %y) nounwind {
+entry:
+; CHECK-LABEL: vcvt_f64_f16
+ %0 = load i16, i16* %x, align 2
+ %1 = tail call float @llvm.convert.from.fp16.f32(i16 %0)
+ %conv = fpext float %1 to double
+; CHECK-V8: vcvtb.f64.f16
+; CHECK-NOV8-NOT: vcvtb.f64.f16
+ store double %conv, double* %y, align 8
+ ret void
+}
+
+define void @vcvt_f16_f64(i16* %x, double* %y) nounwind {
+entry:
+; CHECK-LABEL: vcvt_f16_f64
+ %0 = load double, double* %y, align 8
+ %conv = fptrunc double %0 to float
+; CHECK-V8: vcvtb.f16.f64
+; CHECK-NOV8-NOT: vcvtb.f16.f64
+ %1 = tail call i16 @llvm.convert.to.fp16.f32(float %conv)
+ store i16 %1, i16* %x, align 2
+ ret void
+}
OpenPOWER on IntegriCloud