aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorXi Ruoyao <xry111@xry111.site>2022-08-16 15:34:36 +0800
committerXi Ruoyao <xry111@xry111.site>2022-08-17 16:08:05 +0800
commit3cab897a67af120aa18efa7ddd7ee49b9a29e5dd (patch)
treeb4e16fddf5234b3fd5e61c1a3980b736259878a5
parent80f78716c2c7ce1b7f96077c35c1dd474a2086a2 (diff)
downloadgcc-3cab897a67af120aa18efa7ddd7ee49b9a29e5dd.zip
gcc-3cab897a67af120aa18efa7ddd7ee49b9a29e5dd.tar.gz
gcc-3cab897a67af120aa18efa7ddd7ee49b9a29e5dd.tar.bz2
LoongArch: Provide fmin/fmax RTL pattern
We already had smin/smax RTL pattern using fmin/fmax instruction. But for smin/smax, it's unspecified what will happen if either operand is NaN. So we would generate calls to libc fmin/fmax functions with -fno-finite-math-only (the default for all optimization levels expect -Ofast). But, LoongArch fmin/fmax instruction is IEEE-754-2008 conformant so we can also use the instruction for fmin/fmax pattern and avoid the library function call. gcc/ChangeLog: * config/loongarch/loongarch.md (fmax<mode>3): New RTL pattern. (fmin<mode>3): Likewise. gcc/testsuite/ChangeLog: * gcc.target/loongarch/fmax-fmin.c: New test.
-rw-r--r--gcc/config/loongarch/loongarch.md18
-rw-r--r--gcc/testsuite/gcc.target/loongarch/fmax-fmin.c30
2 files changed, 48 insertions, 0 deletions
diff --git a/gcc/config/loongarch/loongarch.md b/gcc/config/loongarch/loongarch.md
index 6b6df22..8e8868d 100644
--- a/gcc/config/loongarch/loongarch.md
+++ b/gcc/config/loongarch/loongarch.md
@@ -1023,6 +1023,24 @@
[(set_attr "type" "fmove")
(set_attr "mode" "<MODE>")])
+(define_insn "fmax<mode>3"
+ [(set (match_operand:ANYF 0 "register_operand" "=f")
+ (smax:ANYF (match_operand:ANYF 1 "register_operand" "f")
+ (match_operand:ANYF 2 "register_operand" "f")))]
+ ""
+ "fmax.<fmt>\t%0,%1,%2"
+ [(set_attr "type" "fmove")
+ (set_attr "mode" "<MODE>")])
+
+(define_insn "fmin<mode>3"
+ [(set (match_operand:ANYF 0 "register_operand" "=f")
+ (smin:ANYF (match_operand:ANYF 1 "register_operand" "f")
+ (match_operand:ANYF 2 "register_operand" "f")))]
+ ""
+ "fmin.<fmt>\t%0,%1,%2"
+ [(set_attr "type" "fmove")
+ (set_attr "mode" "<MODE>")])
+
(define_insn "smaxa<mode>3"
[(set (match_operand:ANYF 0 "register_operand" "=f")
(if_then_else:ANYF
diff --git a/gcc/testsuite/gcc.target/loongarch/fmax-fmin.c b/gcc/testsuite/gcc.target/loongarch/fmax-fmin.c
new file mode 100644
index 0000000..92cf8a1
--- /dev/null
+++ b/gcc/testsuite/gcc.target/loongarch/fmax-fmin.c
@@ -0,0 +1,30 @@
+/* { dg-do compile } */
+/* { dg-options "-mdouble-float -fno-finite-math-only" } */
+/* { dg-final { scan-assembler "fmin\\.s" } } */
+/* { dg-final { scan-assembler "fmin\\.d" } } */
+/* { dg-final { scan-assembler "fmax\\.s" } } */
+/* { dg-final { scan-assembler "fmax\\.d" } } */
+
+double
+_fmax(double a, double b)
+{
+ return __builtin_fmax(a, b);
+}
+
+float
+_fmaxf(float a, float b)
+{
+ return __builtin_fmaxf(a, b);
+}
+
+double
+_fmin(double a, double b)
+{
+ return __builtin_fmin(a, b);
+}
+
+float
+_fminf(float a, float b)
+{
+ return __builtin_fminf(a, b);
+}