rts-slatec.c
1 //!@file rts-slatec.c
2 //!@author J. Marcel van der Veer
3 //
4 // !@section Copyright
5 //
6 // This file is part of VIF - vintage FORTRAN compiler.
7 // Copyright 2020-2025 J. Marcel van der Veer <algol68g@xs4all.nl>.
8 //
9 //! @section License
10 //
11 // This program is free software; you can redistribute it and/or modify it
12 // under the terms of the GNU General Public License as published by the
13 // Free Software Foundation; either version 3 of the License, or
14 // (at your option) any later version.
15 //
16 // This program is distributed in the hope that it will be useful, but
17 // WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
18 // or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
19 // more details. You should have received a copy of the GNU General Public
20 // License along with this program. If not, see <http://www.gnu.org/licenses/>.
21
22 //! @section Synopsis
23 //!
24 //! Runtime support for SLATEC subprograms.
25
26 //!@brief SLATEC stubs for VIF, adapted to C.
27 // SLATEC common mathematical library, version 4.1, July 1993.
28 //
29 // SLATEC was developed at US government research laboratories
30 // and is in the public domain.
31 // Repository: http://www.netlib.org/slatec/
32
33 #include <vif.h>
34
35 // d1mach yields machine-dependent parameters for the
36 // local machine environment.
37 //
38 // d1mach(1) = b**(emin-1), the smallest positive magnitude.
39 // d1mach(2) = b**emax*(1 - b**(-t)), the largest magnitude.
40 // d1mach(3) = b**(-t), the smallest relative spacing.
41 // d1mach(4) = b**(1-t), the largest relative spacing.
42 // d1mach(5) = log10(b)
43
44 real_4 _r1mach (int_4 *i)
45 {
46 switch (*i) {
47 // r1mach(1) = b**(emin-1), the smallest positive magnitude.
48 case 1: return FLT_MIN;
49 // r1mach(2) = b**emax*(1 - b**(-t)), the largest magnitude.
50 case 2: return FLT_MAX;
51 // r1mach(3) = b**(-t), the smallest relative spacing.
52 case 3: return 0.5 * FLT_EPSILON;
53 // r1mach(4) = b**(1-t), the largest relative spacing.
54 case 4: return FLT_EPSILON;
55 // r1mach(5) = log10(b)
56 case 5: return M_LOG10_2;
57 //
58 default: return 0.0;
59 }
60 }
61
62 real_8 _d1mach (int_4 *i)
63 {
64 switch (*i) {
65 // d1mach(1) = b**(emin-1), the smallest positive magnitude.
66 case 1: return DBL_MIN;
67 // d1mach(2) = b**emax*(1 - b**(-t)), the largest magnitude.
68 case 2: return DBL_MAX;
69 // d1mach(3) = b**(-t), the smallest relative spacing.
70 case 3: return 0.5 * DBL_EPSILON;
71 // d1mach(4) = b**(1-t), the largest relative spacing.
72 case 4: return DBL_EPSILON;
73 // d1mach(5) = log10(b)
74 case 5: return M_LOG10_2;
75 //
76 default: return 0.0;
77 }
78 }
79
80 real_8 _dmach (int_4 *i)
81 {
82 return _d1mach (i);
83 }
84
85 // i1mach yields machine-dependent parameters for the
86 // local machine environment.
87 //
88 // i/o unit numbers:
89 // i1mach(1) = the standard input unit.
90 // i1mach(2) = the standard output unit.
91 // i1mach(3) = the standard punch unit.
92 // i1mach(4) = the standard error message unit.
93 //
94 // words:
95 // i1mach(5) = the number of bits per int_4 storage unit.
96 // i1mach(6) = the number of characters per int_4 storage unit.
97 //
98 // integers:
99 // assume int_4s are represented in the s-digit, base-a form
100 //
101 // sign (x(s-1)*a**(s-1) + ... + x(1)*a + x(0) )
102 //
103 // where 0 .le. x(i) .lt. a for i=0,...,s-1.
104 // i1mach(7) = a, the base.
105 // i1mach(8) = s, the number of base-a digits.
106 // i1mach(9) = a**s - 1, the largest magnitude.
107 //
108 // floating-point_4 numbers:
109 // assume floating-point_4 numbers are represented in the t-digit,
110 // base-b form
111 // sign (b**e)*((x(1)/b) + ... + (x(t)/b**t) )
112 //
113 // where 0 .le. x(i) .lt. b for i=1,...,t,
114 // 0 .lt. x(1), and emin .le. e .le. emax.
115 // i1mach(10) = b, the base.
116 //
117 // single-precision:
118 // i1mach(11) = t, the number of base-b digits.
119 // i1mach(12) = emin, the smallest exponent e.
120 // i1mach(13) = emax, the largest exponent e.
121 //
122 // double-precision:
123 // i1mach(14) = t, the number of base-b digits.
124 // i1mach(15) = emin, the smallest exponent e.
125 // i1mach(16) = emax, the largest exponent e.
126
127 int_4 _i1mach (int_4 *i)
128 {
129 switch(*i) {
130 // i1mach(1) = the standard input unit.
131 // i1mach(2) = the standard output unit.
132 // i1mach(3) = the standard punch unit.
133 // i1mach(4) = the standard error message unit.
134 case 1: return STDF_IN;
135 case 2: return STDF_OUT;
136 case 3: return STDF_PUN;
137 case 4: return STDF_ERR;
138 // i1mach(5) = the number of bits per int_4 storage unit.
139 // i1mach(6) = the number of characters per int_4 storage unit.
140 case 5: return CHAR_BIT * sizeof (int_4);
141 case 6: return sizeof (int_4);
142 // i1mach(7) = a, the base.
143 // i1mach(8) = s, the number of base-a digits.
144 // i1mach(9) = a**s - 1, the largest magnitude.
145 case 7: return 2;
146 case 8: return CHAR_BIT * sizeof (int_4) - 1;
147 case 9: return INT_MAX;
148 // i1mach(10) = b, the base.
149 case 10: return FLT_RADIX;
150 // i1mach(11) = t, the number of base-b digits.
151 // i1mach(12) = emin, the smallest exponent e.
152 // i1mach(13) = emax, the largest exponent e.
153 case 11: return FLT_MANT_DIG;
154 case 12: return FLT_MIN_EXP;
155 case 13: return FLT_MAX_EXP;
156 // i1mach(14) = t, the number of base-b digits.
157 // i1mach(15) = emin, the smallest exponent e.
158 // i1mach(16) = emax, the largest exponent e.
159 case 14: return DBL_MANT_DIG;
160 case 15: return DBL_MIN_EXP;
161 case 16: return DBL_MAX_EXP;
162 //
163 default: return 0;
164 }
165 }
166
167 // amach yields machine-dependent parameters for the local environment.
168 //
169 // subroutine amach(mode, i, i1, r1, d1)
170
171 int_4 _amach (int_4 _p_ mode_, int_4 _p_ i_, int_4 _p_ i1_, real_4 _p_ r1_, real_8 _p_ d1_)
172 {
173 if (*mode_ == 0) {
174 *i1_ = _i1mach(i_);
175 }
176 else if (*mode_ == 1) {
177 *r1_ = _r1mach(i_);
178 }
179 else {
180 *d1_ = _d1mach(i_);
181 }
182 return 0;
183 }
184
185 int_4 _xerclr (void)
186 {
187 // This routine simply resets the current error number to zero.
188 // This may be necessary in order to determine that a certain
189 // error has occurred again since the last time NUMXER was
190 // referenced.
191 errno = 0;
192 return 0;
193 }
194
195 static int_4 _kontrl = 0;
196
197 int_4 _xgetf (int_4 *kontrl)
198 {
199 // XGETF returns the current value of the error control flag
200 // in KONTRL. See subroutine XSETF for flag value meanings.
201 // (KONTRL is an output parameter only.)
202 //
203 *kontrl = _kontrl;
204 return 0;
205 }
206
207 int_4 _xsetf (int_4 *kontrl)
208 {
209 // XSETF sets the error control flag value to KONTRL.
210 // (KONTRL is an input parameter only.)
211 // The following table shows how each message is treated,
212 // depending on the values of KONTRL and LEVEL. (See XERMSG
213 // for description of LEVEL.)
214
215 // If KONTRL is zero or negative, no information other than the
216 // message itself (including numeric values, if any) will be
217 // printed. If KONTRL is positive, introductory messages,
218 // trace-backs, etc., will be printed in addition to the message.
219
220 // ABS(KONTRL)
221 // LEVEL 0 1 2
222 // value
223 // 2 fatal fatal fatal
224
225 // 1 not printed printed fatal
226
227 // 0 not printed printed printed
228
229 // -1 not printed printed printed
230 // only only
231 // once once
232 _kontrl = *kontrl;
233 return 0;
234 }
235
236 int_4 _xermsg (char *librar, char *subrou, char *messg, int_4 *nerr, int_4 *level)
237 {
238 // XERMSG processes a diagnostic message in a manner determined by the
239 // value of LEVEL and the current value of the library error control
240 // flag, KONTRL. See subroutine XSETF for details.
241
242 // LIBRAR A character constant (or character variable) with the name
243 // of the library. This will be 'SLATE' for the SLATEC
244 // ommon Math Library. The error handling package is
245 // general enough to be used by many libraries
246 // simultaneously, so it is desirable for the routine that
247 // detects and reports an error to identify the library name
248 // as well as the routine name.
249
250 // SUBROU A character constant (or character variable) with the name
251 // of the routine that detected the error. Usually it is the
252 // name of the routine that is calling XERMSG. There are
253 // some instances where a user callable library routine calls
254 // lower level subsidiary routines where the error is
255 // detected. In such cases it may be more informative to
256 // supply the name of the routine the user called rather than
257 // the name of the subsidiary routine that detected the
258 // error.
259
260 // MESSG A character constant (or character variable) with the text
261 // of the error or warning message. In the example below,
262 // the message is a character constant that contains a
263 // generic message.
264
265 // ALL XERMSG ('SLATEC', 'MMPY',
266 // *'THE ORDER OF THE MATRIX EXEEDS THE ROW DIMENSION',
267 // *3, 1)
268
269 // It is possible (and is sometimes desirable) to generate a
270 // specific message--e.g., one that contains actual numeric
271 // values. Specific numeric values can be converted into
272 // character strings using formatted WRITE statements into
273 // character variables. This is called standard Fortran
274 // internal file I/O and is exemplified in the first three
275 // lines of the following example. You can also catenate
276 // substrings of characters to construct the error message.
277 // Here is an example showing the use of both writing to
278 // an internal file and catenating character strings.
279
280 // CHARACTER*5 CHARN, CHARL
281 // WRITE (HARN,10) N
282 // WRITE (HARL,10) LDA
283 // 10 FORMAT(I5)
284 // ALL XERMSG ('SLATEC', 'MMPY', 'THE ORDER'//CHARN//
285 // * ' OF THE MATRIX EXEEDS ITS ROW DIMENSION OF'//
286 // * HARL, 3, 1)
287
288 // There are two subtleties worth mentioning. One is that
289 // the // for character catenation is used to construct the
290 // error message so that no single character constant is
291 // continued to the next line. This avoids confusion as to
292 // whether there are trailing blanks at the end of the line.
293 // The second is that by catenating the parts of the message
294 // as an actual argument rather than encoding the entire
295 // message into one large character variable, we avoid
296 // having to know how long the message will be in order to
297 // declare an adequate length for that large character
298 // variable. XERMSG calls XERPRN to print_4 the message using
299 // multiple lines if necessary. If the message is very long,
300 // XERPRN will break it into pieces of 72 characters (as
301 // requested by XERMSG) for printing on multiple lines.
302 // Also, XERMSG asks XERPRN to prefix each line with ' * '
303 // so that the total line length could be 76 characters.
304 // Note also that XERPRN scans the error message backwards
305 // to ignore trailing blanks. Another feature is that
306 // the substring '$$' is treated as a new line sentinel
307 // by XERPRN. If you want to construct a multiline
308 // message without having to count out multiples of 72
309 // characters, just use '$$' as a separator. '$$'
310 // obviously must occur within 72 characters of the
311 // start of each line to have its intended effect since
312 // XERPRN is asked to wrap around at 72 characters in
313 // addition to looking for '$$'.
314
315 // NERR An integer value that is chosen by the library routine's
316 // author. It must be in the range -99 to 999 (three
317 // printable digits). Each distinct error should have its
318 // own error number. These error numbers should be described
319 // in the machine readable documentation for the routine.
320 // The error numbers need be unique only within each routine,
321 // so it is reasonable for each routine to start enumerating
322 // errors from 1 and proceeding to the next integer.
323
324 // LEVEL An integer value in the range 0 to 2 that indicates the
325 // level (severity) of the error. Their meanings are
326
327 // -1 A warning message. This is used if it is not clear
328 // that there really is an error, but the user's attention
329 // may be needed. An attempt is made to only print_4 this
330 // message once.
331
332 // 0 A warning message. This is used if it is not clear
333 // that there really is an error, but the user's attention
334 // may be needed.
335
336 // 1 A recoverable error. This is used even if the error is
337 // so serious that the routine cannot return any useful
338 // answer. If the user has told the error package to
339 // return after recoverable errors, then XERMSG will
340 // return to the Library routine which can then return to
341 // the user's routine. The user may also permit the error
342 // package to terminate the program upon encountering a
343 // recoverable error.
344
345 // 2 A fatal error. XERMSG will not return to its caller
346 // after it receives a fatal error. This level should
347 // hardly ever be used; it is much better to allow the
348 // user a chance to recover. An example of one of the few
349 // cases in which it is permissible to declare a level 2
350 // error is a reverse communication Library routine that
351 // is likely to be called repeatedly until it integrates
352 // across some interval. If there is a serious error in
353 // the input such that another step cannot be taken and
354 // the Library routine is called again without the input
355 // error having been corrected by the caller, the Library
356 // routine will probably be called forever with improper
357 // input. In this case, it is reasonable to declare the
358 // error to be fatal.
359
360 // Each of the arguments to XERMSG is input; none will be modified by
361 // XERMSG. A routine may make multiple calls to XERMSG with warning
362 // level messages; however, after a call to XERMSG with a recoverable
363 // error, the routine should return to the user. Do not try to call
364 // XERMSG with a second recoverable error after the first recoverable
365 // error because the error package saves the error number. The user
366 // can retrieve this error number by calling another entry point_4 in
367 // the error handling package and then clear the error number when
368 // recovering from the error. Calling XERMSG in succession causes the
369 // old error number to be overwritten by the latest error number.
370 // This is considered harmless for error numbers associated with
371 // warning messages but must not be done for error numbers of serious
372 // errors. After a call to XERMSG with a recoverable error, the user
373 // must be given a chance to call NUMXER or XERLR to retrieve or
374 // clear the error number.
375 //
376
377 // Note that this stub ignores _kontrl.
378 fprintf (stderr, "** slatec ** error: %s: %s: %s\n", strlower (librar), strlower (subrou), strlower (messg));
379 errno = *nerr;
380 if (*level == 2) {
381 exit (EXIT_FAILURE);
382 }
383 return 0;
384 }
© 2002-2025 J.M. van der Veer (jmvdveer@xs4all.nl)
|