xref: /netbsd-src/external/gpl3/gcc.old/dist/libgfortran/ieee/ieee_helper.c (revision 4c3eb207d36f67d31994830c0a694161fc1ca39b)
1627f7eb2Smrg /* Helper functions in C for IEEE modules
2*4c3eb207Smrg    Copyright (C) 2013-2020 Free Software Foundation, Inc.
3627f7eb2Smrg    Contributed by Francois-Xavier Coudert <fxcoudert@gcc.gnu.org>
4627f7eb2Smrg 
5627f7eb2Smrg This file is part of the GNU Fortran runtime library (libgfortran).
6627f7eb2Smrg 
7627f7eb2Smrg Libgfortran is free software; you can redistribute it and/or
8627f7eb2Smrg modify it under the terms of the GNU General Public
9627f7eb2Smrg License as published by the Free Software Foundation; either
10627f7eb2Smrg version 3 of the License, or (at your option) any later version.
11627f7eb2Smrg 
12627f7eb2Smrg Libgfortran is distributed in the hope that it will be useful,
13627f7eb2Smrg but WITHOUT ANY WARRANTY; without even the implied warranty of
14627f7eb2Smrg MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15627f7eb2Smrg GNU General Public License for more details.
16627f7eb2Smrg 
17627f7eb2Smrg Under Section 7 of GPL version 3, you are granted additional
18627f7eb2Smrg permissions described in the GCC Runtime Library Exception, version
19627f7eb2Smrg 3.1, as published by the Free Software Foundation.
20627f7eb2Smrg 
21627f7eb2Smrg You should have received a copy of the GNU General Public License and
22627f7eb2Smrg a copy of the GCC Runtime Library Exception along with this program;
23627f7eb2Smrg see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
24627f7eb2Smrg <http://www.gnu.org/licenses/>.  */
25627f7eb2Smrg 
26627f7eb2Smrg #include "libgfortran.h"
27627f7eb2Smrg 
28627f7eb2Smrg /* Prototypes.  */
29627f7eb2Smrg 
30627f7eb2Smrg extern int ieee_class_helper_4 (GFC_REAL_4 *);
31627f7eb2Smrg internal_proto(ieee_class_helper_4);
32627f7eb2Smrg 
33627f7eb2Smrg extern int ieee_class_helper_8 (GFC_REAL_8 *);
34627f7eb2Smrg internal_proto(ieee_class_helper_8);
35627f7eb2Smrg 
36627f7eb2Smrg #ifdef HAVE_GFC_REAL_10
37627f7eb2Smrg extern int ieee_class_helper_10 (GFC_REAL_10 *);
38627f7eb2Smrg internal_proto(ieee_class_helper_10);
39627f7eb2Smrg #endif
40627f7eb2Smrg 
41627f7eb2Smrg #ifdef HAVE_GFC_REAL_16
42627f7eb2Smrg extern int ieee_class_helper_16 (GFC_REAL_16 *);
43627f7eb2Smrg internal_proto(ieee_class_helper_16);
44627f7eb2Smrg #endif
45627f7eb2Smrg 
46627f7eb2Smrg /* Enumeration of the possible floating-point types. These values
47627f7eb2Smrg    correspond to the hidden arguments of the IEEE_CLASS_TYPE
48627f7eb2Smrg    derived-type of IEEE_ARITHMETIC.  */
49627f7eb2Smrg 
50627f7eb2Smrg enum { IEEE_OTHER_VALUE = 0, IEEE_SIGNALING_NAN, IEEE_QUIET_NAN,
51627f7eb2Smrg   IEEE_NEGATIVE_INF, IEEE_NEGATIVE_NORMAL, IEEE_NEGATIVE_DENORMAL,
52627f7eb2Smrg   IEEE_NEGATIVE_ZERO, IEEE_POSITIVE_ZERO, IEEE_POSITIVE_DENORMAL,
53627f7eb2Smrg   IEEE_POSITIVE_NORMAL, IEEE_POSITIVE_INF, IEEE_SUBNORMAL,
54627f7eb2Smrg   IEEE_NEGATIVE_SUBNORMAL, IEEE_POSITIVE_SUBNORMAL };
55627f7eb2Smrg 
56627f7eb2Smrg #define CLASSMACRO(TYPE) \
57627f7eb2Smrg   int ieee_class_helper_ ## TYPE (GFC_REAL_ ## TYPE *value) \
58627f7eb2Smrg   { \
59627f7eb2Smrg     int res = __builtin_fpclassify (IEEE_QUIET_NAN, IEEE_POSITIVE_INF, \
60627f7eb2Smrg 				    IEEE_POSITIVE_NORMAL, \
61627f7eb2Smrg 				    IEEE_POSITIVE_DENORMAL, \
62627f7eb2Smrg 				    IEEE_POSITIVE_ZERO, *value); \
63627f7eb2Smrg  \
64627f7eb2Smrg     if (__builtin_signbit (*value)) \
65627f7eb2Smrg     { \
66627f7eb2Smrg       if (res == IEEE_POSITIVE_NORMAL) \
67627f7eb2Smrg 	return IEEE_NEGATIVE_NORMAL; \
68627f7eb2Smrg       else if (res == IEEE_POSITIVE_DENORMAL) \
69627f7eb2Smrg 	return IEEE_NEGATIVE_DENORMAL; \
70627f7eb2Smrg       else if (res == IEEE_POSITIVE_ZERO) \
71627f7eb2Smrg 	return IEEE_NEGATIVE_ZERO; \
72627f7eb2Smrg       else if (res == IEEE_POSITIVE_INF) \
73627f7eb2Smrg 	return IEEE_NEGATIVE_INF; \
74627f7eb2Smrg     } \
75627f7eb2Smrg  \
76627f7eb2Smrg     if (res == IEEE_QUIET_NAN) \
77627f7eb2Smrg     { \
78627f7eb2Smrg       /* TODO: Handle signaling NaNs  */ \
79627f7eb2Smrg       return res; \
80627f7eb2Smrg     } \
81627f7eb2Smrg  \
82627f7eb2Smrg     return res; \
83627f7eb2Smrg   }
84627f7eb2Smrg 
85627f7eb2Smrg CLASSMACRO(4)
86627f7eb2Smrg CLASSMACRO(8)
87627f7eb2Smrg 
88627f7eb2Smrg #ifdef HAVE_GFC_REAL_10
89627f7eb2Smrg CLASSMACRO(10)
90627f7eb2Smrg #endif
91627f7eb2Smrg 
92627f7eb2Smrg #ifdef HAVE_GFC_REAL_16
93627f7eb2Smrg CLASSMACRO(16)
94627f7eb2Smrg #endif
95627f7eb2Smrg 
96627f7eb2Smrg 
97627f7eb2Smrg #define GFC_FPE_ALL (GFC_FPE_INVALID | GFC_FPE_DENORMAL | \
98627f7eb2Smrg 		     GFC_FPE_ZERO | GFC_FPE_OVERFLOW | \
99627f7eb2Smrg 		     GFC_FPE_UNDERFLOW | GFC_FPE_INEXACT)
100627f7eb2Smrg 
101627f7eb2Smrg /* Functions to save and restore floating-point state, clear and restore
102627f7eb2Smrg    exceptions on procedure entry/exit.  The rules we follow are set
103627f7eb2Smrg    in Fortran 2008's 14.3 paragraph 3, note 14.4, 14.4 paragraph 4,
104627f7eb2Smrg    14.5 paragraph 2, and 14.6 paragraph 1.  */
105627f7eb2Smrg 
106627f7eb2Smrg void ieee_procedure_entry (void *);
107627f7eb2Smrg export_proto(ieee_procedure_entry);
108627f7eb2Smrg 
109627f7eb2Smrg void
ieee_procedure_entry(void * state)110627f7eb2Smrg ieee_procedure_entry (void *state)
111627f7eb2Smrg {
112627f7eb2Smrg   /* Save the floating-point state in the space provided by the caller.  */
113627f7eb2Smrg   get_fpu_state (state);
114627f7eb2Smrg 
115627f7eb2Smrg   /* Clear the floating-point exceptions.  */
116627f7eb2Smrg   set_fpu_except_flags (0, GFC_FPE_ALL);
117627f7eb2Smrg }
118627f7eb2Smrg 
119627f7eb2Smrg 
120627f7eb2Smrg void ieee_procedure_exit (void *);
121627f7eb2Smrg export_proto(ieee_procedure_exit);
122627f7eb2Smrg 
123627f7eb2Smrg void
ieee_procedure_exit(void * state)124627f7eb2Smrg ieee_procedure_exit (void *state)
125627f7eb2Smrg {
126627f7eb2Smrg   /* Get the flags currently signaling.  */
127627f7eb2Smrg   int flags = get_fpu_except_flags ();
128627f7eb2Smrg 
129627f7eb2Smrg   /* Restore the floating-point state we had on entry.  */
130627f7eb2Smrg   set_fpu_state (state);
131627f7eb2Smrg 
132627f7eb2Smrg   /* And re-raised the flags that were raised since entry.  */
133627f7eb2Smrg   set_fpu_except_flags (flags, 0);
134627f7eb2Smrg }
135627f7eb2Smrg 
136