blob: b3594a1704a75a826af8cf1c554efc7170da11a2 [file] [log] [blame]
Thomas Gleixnerea65cc92019-06-04 10:10:48 +02001// SPDX-License-Identifier: GPL-2.0-only
Markos Chandras4e9561b2015-08-13 09:56:35 +02002/*
3 * IEEE754 floating point arithmetic
4 * double precision: MIN{,A}.f
5 * MIN : Scalar Floating-Point Minimum
6 * MINA: Scalar Floating-Point argument with Minimum Absolute Value
7 *
8 * MIN.D : FPR[fd] = minNum(FPR[fs],FPR[ft])
9 * MINA.D: FPR[fd] = maxNumMag(FPR[fs],FPR[ft])
10 *
11 * MIPS floating point support
12 * Copyright (C) 2015 Imagination Technologies, Ltd.
13 * Author: Markos Chandras <markos.chandras@imgtec.com>
Markos Chandras4e9561b2015-08-13 09:56:35 +020014 */
15
16#include "ieee754dp.h"
17
18union ieee754dp ieee754dp_fmin(union ieee754dp x, union ieee754dp y)
19{
20 COMPXDP;
21 COMPYDP;
22
23 EXPLODEXDP;
24 EXPLODEYDP;
25
26 FLUSHXDP;
27 FLUSHYDP;
28
29 ieee754_clearcx();
30
31 switch (CLPAIR(xc, yc)) {
32 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_SNAN):
33 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_SNAN):
34 case CLPAIR(IEEE754_CLASS_NORM, IEEE754_CLASS_SNAN):
35 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_SNAN):
36 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_SNAN):
37 return ieee754dp_nanxcpt(y);
38
39 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_SNAN):
40 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_QNAN):
41 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_ZERO):
42 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_NORM):
43 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_DNORM):
44 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_INF):
45 return ieee754dp_nanxcpt(x);
46
Aleksandar Markovice78bf0d2017-07-27 18:08:48 +020047 /*
48 * Quiet NaN handling
49 */
50
51 /*
52 * The case of both inputs quiet NaNs
53 */
54 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_QNAN):
55 return x;
56
57 /*
58 * The cases of exactly one input quiet NaN (numbers
59 * are here preferred as returned values to NaNs)
60 */
Markos Chandras4e9561b2015-08-13 09:56:35 +020061 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_QNAN):
62 case CLPAIR(IEEE754_CLASS_NORM, IEEE754_CLASS_QNAN):
63 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_QNAN):
64 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_QNAN):
65 return x;
66
Markos Chandras4e9561b2015-08-13 09:56:35 +020067 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_ZERO):
68 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_NORM):
69 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_DNORM):
70 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_INF):
71 return y;
72
73 /*
74 * Infinity and zero handling
75 */
76 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_ZERO):
77 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_NORM):
78 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_DNORM):
79 case CLPAIR(IEEE754_CLASS_NORM, IEEE754_CLASS_ZERO):
80 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_ZERO):
81 return xs ? x : y;
82
83 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_INF):
84 case CLPAIR(IEEE754_CLASS_NORM, IEEE754_CLASS_INF):
85 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_INF):
86 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_INF):
87 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_NORM):
88 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_DNORM):
89 return ys ? y : x;
90
91 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_ZERO):
Aleksandar Markovic15560a52017-07-27 18:08:49 +020092 return ieee754dp_zero(xs | ys);
Markos Chandras4e9561b2015-08-13 09:56:35 +020093
94 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_DNORM):
95 DPDNORMX;
Aleksandar Markovic2a14b212017-11-02 12:14:05 +010096 /* fall through */
Markos Chandras4e9561b2015-08-13 09:56:35 +020097
98 case CLPAIR(IEEE754_CLASS_NORM, IEEE754_CLASS_DNORM):
99 DPDNORMY;
100 break;
101
102 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_NORM):
103 DPDNORMX;
104 }
105
106 /* Finally get to do some computation */
107
108 assert(xm & DP_HIDDEN_BIT);
109 assert(ym & DP_HIDDEN_BIT);
110
111 /* Compare signs */
112 if (xs > ys)
113 return x;
114 else if (xs < ys)
115 return y;
116
Aleksandar Markovicaabf5cf2017-07-27 18:08:50 +0200117 /* Signs of inputs are the same, let's compare exponents */
118 if (xs == 0) {
119 /* Inputs are both positive */
120 if (xe > ye)
121 return y;
122 else if (xe < ye)
123 return x;
124 } else {
125 /* Inputs are both negative */
126 if (xe > ye)
127 return x;
128 else if (xe < ye)
129 return y;
130 }
Markos Chandras4e9561b2015-08-13 09:56:35 +0200131
Aleksandar Markovicaabf5cf2017-07-27 18:08:50 +0200132 /* Signs and exponents of inputs are equal, let's compare mantissas */
133 if (xs == 0) {
134 /* Inputs are both positive, with equal signs and exponents */
135 if (xm <= ym)
136 return x;
137 return y;
138 }
139 /* Inputs are both negative, with equal signs and exponents */
Markos Chandras4e9561b2015-08-13 09:56:35 +0200140 if (xm <= ym)
Aleksandar Markovicaabf5cf2017-07-27 18:08:50 +0200141 return y;
142 return x;
Markos Chandras4e9561b2015-08-13 09:56:35 +0200143}
144
145union ieee754dp ieee754dp_fmina(union ieee754dp x, union ieee754dp y)
146{
147 COMPXDP;
148 COMPYDP;
149
150 EXPLODEXDP;
151 EXPLODEYDP;
152
153 FLUSHXDP;
154 FLUSHYDP;
155
156 ieee754_clearcx();
157
158 switch (CLPAIR(xc, yc)) {
159 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_SNAN):
160 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_SNAN):
161 case CLPAIR(IEEE754_CLASS_NORM, IEEE754_CLASS_SNAN):
162 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_SNAN):
163 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_SNAN):
164 return ieee754dp_nanxcpt(y);
165
166 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_SNAN):
167 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_QNAN):
168 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_ZERO):
169 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_NORM):
170 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_DNORM):
171 case CLPAIR(IEEE754_CLASS_SNAN, IEEE754_CLASS_INF):
172 return ieee754dp_nanxcpt(x);
173
Aleksandar Markovice78bf0d2017-07-27 18:08:48 +0200174 /*
175 * Quiet NaN handling
176 */
177
178 /*
179 * The case of both inputs quiet NaNs
180 */
181 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_QNAN):
182 return x;
183
184 /*
185 * The cases of exactly one input quiet NaN (numbers
186 * are here preferred as returned values to NaNs)
187 */
Markos Chandras4e9561b2015-08-13 09:56:35 +0200188 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_QNAN):
189 case CLPAIR(IEEE754_CLASS_NORM, IEEE754_CLASS_QNAN):
190 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_QNAN):
191 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_QNAN):
192 return x;
193
Markos Chandras4e9561b2015-08-13 09:56:35 +0200194 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_ZERO):
195 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_NORM):
196 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_DNORM):
197 case CLPAIR(IEEE754_CLASS_QNAN, IEEE754_CLASS_INF):
198 return y;
199
200 /*
201 * Infinity and zero handling
202 */
Aleksandar Markovic3444c4e2017-07-27 18:08:52 +0200203 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_INF):
204 return ieee754dp_inf(xs | ys);
205
Markos Chandras4e9561b2015-08-13 09:56:35 +0200206 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_ZERO):
207 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_NORM):
208 case CLPAIR(IEEE754_CLASS_INF, IEEE754_CLASS_DNORM):
209 case CLPAIR(IEEE754_CLASS_NORM, IEEE754_CLASS_ZERO):
210 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_ZERO):
Aleksandar Markovic304bfe42017-07-27 18:08:53 +0200211 return y;
Markos Chandras4e9561b2015-08-13 09:56:35 +0200212
Markos Chandras4e9561b2015-08-13 09:56:35 +0200213 case CLPAIR(IEEE754_CLASS_NORM, IEEE754_CLASS_INF):
214 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_INF):
215 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_INF):
216 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_NORM):
217 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_DNORM):
Aleksandar Markovic304bfe42017-07-27 18:08:53 +0200218 return x;
Markos Chandras4e9561b2015-08-13 09:56:35 +0200219
220 case CLPAIR(IEEE754_CLASS_ZERO, IEEE754_CLASS_ZERO):
Aleksandar Markovic15560a52017-07-27 18:08:49 +0200221 return ieee754dp_zero(xs | ys);
Markos Chandras4e9561b2015-08-13 09:56:35 +0200222
223 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_DNORM):
224 DPDNORMX;
Aleksandar Markovic2a14b212017-11-02 12:14:05 +0100225 /* fall through */
Markos Chandras4e9561b2015-08-13 09:56:35 +0200226
227 case CLPAIR(IEEE754_CLASS_NORM, IEEE754_CLASS_DNORM):
228 DPDNORMY;
229 break;
230
231 case CLPAIR(IEEE754_CLASS_DNORM, IEEE754_CLASS_NORM):
232 DPDNORMX;
233 }
234
235 /* Finally get to do some computation */
236
237 assert(xm & DP_HIDDEN_BIT);
238 assert(ym & DP_HIDDEN_BIT);
239
240 /* Compare exponent */
241 if (xe > ye)
242 return y;
243 else if (xe < ye)
244 return x;
245
246 /* Compare mantissa */
Aleksandar Markovic1a41b3b2017-07-27 18:08:51 +0200247 if (xm < ym)
248 return x;
249 else if (xm > ym)
250 return y;
251 else if (xs == 1)
Markos Chandras4e9561b2015-08-13 09:56:35 +0200252 return x;
253 return y;
254}