1 /*
2  * Copyright (c) 1994 Cygnus Support.
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms are permitted
6  * provided that the above copyright notice and this paragraph are
7  * duplicated in all such forms and that any documentation,
8  * and/or other materials related to such
9  * distribution and use acknowledge that the software was developed
10  * at Cygnus Support, Inc.  Cygnus Support, Inc. may not be used to
11  * endorse or promote products derived from this software without
12  * specific prior written permission.
13  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
14  * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
15  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
16  */
17 #include <stdlib.h>
18 #include <signal.h>
19 #include  "test.h"
20 #include <math.h>
21 #include <string.h>
22 #include <stdint.h>
23 #include <malloc.h>
24 int verbose;
25 static int count;
26 int inacc;
27 
28 int redo = 0;
29 extern int calc;
30 
31 #ifdef MALLOC_DEBUG
32 extern char **environ;
33 extern int _malloc_test_fail;
34 #endif
35 
36 int
main(int ac,char ** av)37 main (int ac,
38       char **av)
39 {
40 #ifdef MALLOC_DEBUG
41   environ = en;
42 #endif
43   int i;
44   int math2 = 1;
45   int string= 1;
46   int is = 1;
47   int math= 1;
48   int cvt = 1;
49 #ifdef _HAVE_IEEEFP_FUNCS
50   int ieee= 1;
51 #endif
52   int vector=0;
53   for (i = 1; i < ac; i++)
54   {
55     if (strcmp(av[i],"-v")==0)
56      verbose ++;
57     if (strcmp(av[i],"-nomath2") == 0)
58      math2 = 0;
59     (void) math2;
60     if (strcmp(av[i],"-nostrin") == 0)
61      string= 0;
62     (void) string;
63     if (strcmp(av[i],"-nois") == 0)
64      is = 0;
65     (void) is;
66     if (strcmp(av[i],"-nomath") == 0)
67      math= 0;
68     (void) math;
69     if (strcmp(av[i],"-nocvt") == 0)
70      cvt = 0;
71 #ifdef _HAVE_IEEEFP_FUNCS
72     if (strcmp(av[i],"-noiee") == 0)
73      ieee= 0;
74   (void) ieee;
75 #endif
76     if (strcmp(av[i],"-generate") == 0) {
77      vector = 1;
78      calc = 1;
79     }
80   }
81   if (cvt)
82    test_cvt();
83 
84 #if TEST_PART == 0 || TEST_PART == -1
85   if (math2)
86    test_math2();
87   if (string)
88    test_string();
89 #endif
90   if (math)
91    test_math(vector);
92 #if TEST_PART == 0 || TEST_PART == -1
93   if (is)
94    test_is();
95 #ifdef _HAVE_IEEEFP_FUNCS
96   if (ieee)
97    test_ieee();
98 #endif
99 #endif
100   printf("Tested %d functions, %d errors detected\n", count, inacc);
101   exit(inacc != 0);
102 }
103 
104 static const char *iname = "foo";
newfunc(const char * string)105 void newfunc (const char *string)
106 {
107   if (strcmp(iname, string))
108   {
109 #ifdef MALLOC_DEBUG
110     char *memcheck = getenv("CHECK_NAME");
111     if (memcheck && !strcmp(string, memcheck))
112       _malloc_test_fail = atoi(getenv("CHECK_COUNT"));
113     if (memcheck && !strcmp(iname, memcheck)) {
114       if (_malloc_test_fail) {
115 	printf("malloc test fail remain %d\n", _malloc_test_fail);
116 	_malloc_test_fail = 0;
117       }
118     }
119 #endif
120     printf("testing %s\n", string);
121     fflush(stdout);
122     iname = string;
123   }
124 
125 }
126 
127 
128 static int theline;
129 
line(int li)130 void line(int li)
131 {
132   if (verbose)
133   {
134     printf("  %d\n", li);
135   }
136   theline = li;
137 
138   count++;
139 }
140 
141 
142 
143 int reduce = 0;
144 
145 int strtod_vector = 0;
146 
147 static int
bigger(__ieee_double_shape_type * a,__ieee_double_shape_type * b)148 bigger (__ieee_double_shape_type *a,
149 	   __ieee_double_shape_type *b)
150 {
151 
152   if (a->parts.msw > b->parts.msw)
153     {
154 
155       return 1;
156     }
157   else if (a->parts.msw == b->parts.msw)
158     {
159       if (a->parts.lsw > b->parts.lsw)
160 	{
161 	  return 1;
162 	}
163     }
164   return 0;
165 }
166 
167 static int
fbigger(__ieee_float_shape_type * a,__ieee_float_shape_type * b)168 fbigger (__ieee_float_shape_type *a,
169 	   __ieee_float_shape_type *b)
170 {
171 
172   if (a->p1 > b->p1)
173     {
174       return 1;
175     }
176   return 0;
177 }
178 
179 
180 
181 /* Return the first bit different between two double numbers */
182 int
mag_of_error(double is,double shouldbe)183 mag_of_error (double is,
184        double shouldbe)
185 {
186   __ieee_double_shape_type a,b;
187   int i;
188   int a_big;
189   uint32_t mask;
190   uint32_t __x;
191   uint32_t msw, lsw;
192   a.value = is;
193 
194   b.value = shouldbe;
195 
196   if (a.parts.msw == b.parts.msw
197       && a.parts.lsw== b.parts.lsw) return 64;
198 
199 
200   /* Subtract the larger from the smaller number */
201 
202   a_big = bigger(&a, &b);
203 
204   if (!a_big) {
205     uint32_t t;
206     t = a.parts.msw;
207     a.parts.msw = b.parts.msw;
208     b.parts.msw = t;
209 
210     t = a.parts.lsw;
211     a.parts.lsw = b.parts.lsw;
212     b.parts.lsw = t;
213   }
214 
215 
216 
217   __x = (a.parts.lsw) - (b.parts.lsw);
218   msw = (a.parts.msw) - (b.parts.msw) - (__x > (a.parts.lsw));
219   lsw = __x;
220 
221 
222 
223 
224   /* Find out which bit the difference is in */
225   mask = 0x80000000UL;
226   for (i = 0; i < 32; i++)
227   {
228     if (((msw) & mask)!=0) return i;
229     mask >>=1;
230   }
231 
232   mask = 0x80000000UL;
233   for (i = 0; i < 32; i++)
234   {
235 
236     if (((lsw) & mask)!=0) return i+32;
237     mask >>=1;
238   }
239 
240   return 64;
241 
242 }
243 
244 /* Return the first bit different between two double numbers */
245 int
fmag_of_error(float is,float shouldbe)246 fmag_of_error (float is,
247        float shouldbe)
248 {
249   __ieee_float_shape_type a,b;
250   int i;
251   int a_big;
252   uint32_t mask;
253   uint32_t sw;
254   a.value = is;
255 
256   b.value = shouldbe;
257 
258   if (a.p1 == b.p1) return 32;
259 
260   /* Subtract the larger from the smaller number */
261 
262   a_big = fbigger(&a, &b);
263 
264   if (!a_big) {
265     uint32_t t;
266     t = a.p1;
267     a.p1 = b.p1;
268     b.p1 = t;
269   }
270 
271   sw = (a.p1) - (b.p1);
272 
273   mask = 0x80000000UL;
274   for (i = 0; i < 32; i++)
275   {
276 	  if (((sw) & mask)!=0) return i;
277 	  mask >>=1;
278   }
279   return 32;
280 }
281 
282  int ok_mag;
283 
284 
285 
286 void
test_sok(char * is,char * shouldbe)287 test_sok (char *is,
288        char *shouldbe)
289 {
290   if (strcmp(is,shouldbe))
291     {
292     printf("%s:%d, inacurate answer: (%s should be %s)\n",
293 	   iname,
294 	   theline,
295 	   is, shouldbe);
296     inacc++;
297   }
298 }
299 void
test_iok(int is,int shouldbe)300 test_iok (int is,
301        int shouldbe)
302 {
303   if (is != shouldbe){
304     printf("%s:%d, inacurate answer: (%08x should be %08x)\n",
305 	   iname,
306 	   theline,
307 	   is, shouldbe);
308     inacc++;
309   }
310 }
311 
312 
313 /* Compare counted strings upto a certain length - useful to test single
314    prec float conversions against double results
315 */
316 void
test_scok(char * is,char * shouldbe,int count)317 test_scok (char *is,
318        char *shouldbe,
319        int count)
320 {
321   if (strncmp(is,shouldbe, count))
322     {
323     printf("%s:%d, inacurate answer: (%s should be %s)\n",
324 	   iname,
325 	   theline,
326 	   is, shouldbe);
327     inacc++;
328   }
329 }
330 
331 /* Compare counted strings upto a certain length, allowing two forms - useful to test single
332    prec float conversions against double results
333 */
334 void
test_scok2(char * is,char * maybe1,char * maybe2,int count)335 test_scok2 (char *is,
336        char *maybe1,
337        char *maybe2,
338        int count)
339 {
340   if (strncmp(is,maybe1, count) && (maybe2 == NULL || strncmp(is,maybe2,count)))
341     {
342     printf("%s:%d, inacurate answer: (%s may be %s or %s)\n",
343 	   iname,
344 	   theline,
345 	   is, maybe1, maybe2 ? maybe2 : "(nothing)");
346     inacc++;
347   }
348 }
349 
350 void
test_eok(int is,int shouldbe)351 test_eok (int is,
352        int shouldbe)
353 {
354   if (is != shouldbe){
355     printf("%s:%d, bad errno answer: (%d should be %d)\n",
356 	   iname,
357 	   theline,
358 	   is, shouldbe);
359     inacc++;
360   }
361 }
362 
363 void
test_mok(double value,double shouldbe,int okmag)364 test_mok (double value,
365        double shouldbe,
366        int okmag)
367 {
368   __ieee_double_shape_type a,b;
369   int mag = mag_of_error(value, shouldbe);
370   if (mag == 0)
371   {
372     /* error in the first bit is ok if the numbers are both 0 */
373     if (value == 0.0 && shouldbe == 0.0)
374      return;
375 
376   }
377   a.value = shouldbe;
378   b.value = value;
379 
380   if (mag < okmag)
381   {
382     printf("%s:%d, wrong answer: bit %d ",
383 	   iname,
384 	   theline,
385 	   mag);
386      printf("%08lx%08lx %08lx%08lx) ",
387 	    (unsigned long) a.parts.msw,	     (unsigned long) a.parts.lsw,
388 	    (unsigned long) b.parts.msw,	     (unsigned long) b.parts.lsw);
389     printf("(%g %g)\n",   a.value, b.value);
390     inacc++;
391   }
392 }
393 
394 void
test_mfok(float value,float shouldbe,int okmag)395 test_mfok (float value,
396 	   float shouldbe,
397 	   int okmag)
398 {
399   __ieee_float_shape_type a,b;
400   int mag = fmag_of_error(value, shouldbe);
401   if (mag == 0)
402   {
403     /* error in the first bit is ok if the numbers are both 0 */
404     if (value == 0.0f && shouldbe == 0.0f)
405      return;
406 
407   }
408   a.value = shouldbe;
409   b.value = value;
410 
411   if (mag < okmag)
412   {
413     printf("%s:%d, wrong answer: bit %d ",
414 	   iname,
415 	   theline,
416 	   mag);
417      printf("%08lx %08lx) ",
418 	    (unsigned long) a.p1,
419 	    (unsigned long) b.p1);
420      printf("(%g %g)\n",   (double) a.value, (double) b.value);
421     inacc++;
422   }
423 }
424 
425 #ifdef __PCCNECV70__
kill()426 kill() {}
getpid()427 getpid() {}
428 #endif
429 
430 #if 0
431 static void bt(void){
432 
433   double f1,f2;
434   f1 = 0.0;
435   f2 = 0.0/f1;
436   printf("(%g)\n", f2);
437 
438 }
439 #endif
440