Home | History | Annotate | Download | only in ArithChk
      1 /** @file
      2     Program to generate an arith.h for use with the gdtoa binary to decimal and decimal to binary
      3     conversion library.
      4 
      5   Copyright (c) 2014, Intel Corporation. All rights reserved.<BR>
      6   This program and the accompanying materials
      7   are licensed and made available under the terms and conditions of the BSD License
      8   which accompanies this distribution. The full text of the license may be found at
      9   http://opensource.org/licenses/bsd-license.
     10 
     11   THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
     12   WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
     13 
     14   Copyright (C) 1997, 1998 Lucent Technologies
     15   All Rights Reserved
     16 
     17   Permission to use, copy, modify, and distribute this software and
     18   its documentation for any purpose and without fee is hereby
     19   granted, provided that the above copyright notice appear in all
     20   copies and that both that the copyright notice and this
     21   permission notice and warranty disclaimer appear in supporting
     22   documentation, and that the name of Lucent or any of its entities
     23   not be used in advertising or publicity pertaining to
     24   distribution of the software without specific, written prior
     25   permission.
     26 
     27   LUCENT DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
     28   INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS.
     29   IN NO EVENT SHALL LUCENT OR ANY OF ITS ENTITIES BE LIABLE FOR ANY
     30   SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
     31   WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
     32   IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
     33   ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF
     34   THIS SOFTWARE.
     35 
     36   NetBSD: arithchk.c,v 1.2 2006/01/25 15:27:42 kleink Exp
     37 ****************************************************************/
     38 #include  <sys/EfiCdefs.h>
     39 
     40 /* Try to deduce arith.h from arithmetic properties. */
     41 
     42 #include <stdio.h>
     43 
     44 static int dalign;
     45 
     46 typedef struct  Akind {
     47   char *name;
     48   int   kind;
     49 } Akind;
     50 
     51 static Akind  IEEE_LITTLE_ENDIAN  = { "IEEE_LITTLE_ENDIAN", 1 };
     52 static Akind  IEEE_BIG_ENDIAN     = { "IEEE_BIG_ENDIAN", 2 };
     53 static Akind  IBM                 = { "IBM", 3 };
     54 static Akind  VAX                 = { "VAX", 4 };
     55 static Akind  CRAY                = { "CRAY", 5};
     56 
     57 static Akind *
     58 Lcheck()
     59 {
     60   union {
     61     double d;
     62     long L[2];
     63   } u;
     64   struct {
     65     double d;
     66     long L;
     67   } x[2];
     68 
     69   if (sizeof(x) > 2*(sizeof(double) + sizeof(long)))
     70     dalign = 1;
     71   u.L[0] = u.L[1] = 0;
     72   u.d = 1e13;
     73   if (u.L[0] == 1117925532 && u.L[1] == -448790528)
     74     return &IEEE_BIG_ENDIAN;
     75   if (u.L[1] == 1117925532 && u.L[0] == -448790528)
     76     return &IEEE_LITTLE_ENDIAN;
     77   if (u.L[0] == -2065213935 && u.L[1] == 10752)
     78     return &VAX;
     79   if (u.L[0] == 1267827943 && u.L[1] == 704643072)
     80     return &IBM;
     81   return 0;
     82   }
     83 
     84  static Akind *
     85 icheck()
     86 {
     87   union {
     88     double d;
     89     int L[2];
     90     } u;
     91   struct {
     92     double d;
     93     int L;
     94     } x[2];
     95 
     96   if (sizeof(x) > 2*(sizeof(double) + sizeof(int)))
     97     dalign = 1;
     98   u.L[0] = u.L[1] = 0;
     99   u.d = 1e13;
    100   if (u.L[0] == 1117925532 && u.L[1] == -448790528)
    101     return &IEEE_BIG_ENDIAN;
    102   if (u.L[1] == 1117925532 && u.L[0] == -448790528)
    103     return &IEEE_LITTLE_ENDIAN;
    104   if (u.L[0] == -2065213935 && u.L[1] == 10752)
    105     return &VAX;
    106   if (u.L[0] == 1267827943 && u.L[1] == 704643072)
    107     return &IBM;
    108   return 0;
    109   }
    110 
    111 char *emptyfmt = "";  /* avoid possible warning message with printf("") */
    112 
    113  static Akind *
    114 ccheck()
    115 {
    116   union {
    117     double d;
    118     long L;
    119     } u;
    120   long Cray1;
    121 
    122   /* Cray1 = 4617762693716115456 -- without overflow on non-Crays */
    123   Cray1 = printf(emptyfmt) < 0 ? 0 : 4617762;
    124   if (printf(emptyfmt, Cray1) >= 0)
    125     Cray1 = 1000000*Cray1 + 693716;
    126   if (printf(emptyfmt, Cray1) >= 0)
    127     Cray1 = 1000000*Cray1 + 115456;
    128   u.d = 1e13;
    129   if (u.L == Cray1)
    130     return &CRAY;
    131   return 0;
    132   }
    133 
    134  static int
    135 fzcheck()
    136 {
    137   double a, b;
    138   int i;
    139 
    140   a = 1.;
    141   b = .1;
    142   for(i = 155;; b *= b, i >>= 1) {
    143     if (i & 1) {
    144       a *= b;
    145       if (i == 1)
    146         break;
    147       }
    148     }
    149   b = a * a;
    150   return b == 0.;
    151   }
    152 
    153  int
    154 main()
    155 {
    156   Akind *a = 0;
    157   int Ldef = 0;
    158   FILE *f;
    159 
    160 #ifdef WRITE_ARITH_H  /* for Symantec's buggy "make" */
    161   f = fopen("arith.h", "w");
    162   if (!f) {
    163     printf("Cannot open arith.h\n");
    164     return 1;
    165     }
    166 #else
    167   f = stdout;
    168 #endif
    169 
    170   if (sizeof(double) == 2*sizeof(long))
    171     a = Lcheck();
    172   else if (sizeof(double) == 2*sizeof(int)) {
    173     Ldef = 1;
    174     a = icheck();
    175     }
    176   else if (sizeof(double) == sizeof(long))
    177     a = ccheck();
    178   if (a) {
    179     fprintf(f, "#define %s\n#define Arith_Kind_ASL %d\n",
    180       a->name, a->kind);
    181     if (Ldef)
    182       fprintf(f, "#define Long int\n#define Intcast (int)(long)\n");
    183     if (dalign)
    184       fprintf(f, "#define Double_Align\n");
    185     if (sizeof(char*) == 8)
    186       fprintf(f, "#define X64_bit_pointers\n");
    187 #ifndef NO_LONG_LONG
    188     if (sizeof(long long) < 8)
    189 #endif
    190       fprintf(f, "#define NO_LONG_LONG\n");
    191     if (a->kind <= 2 && fzcheck())
    192       fprintf(f, "#define Sudden_Underflow\n");
    193     return 0;
    194     }
    195   fprintf(f, "/* Unknown arithmetic */\n");
    196   return 1;
    197   }
    198