Annotation of coherent/b/kernel/emulator/poly_tan.c, revision 1.1

1.1     ! root        1: /*---------------------------------------------------------------------------+
        !             2:  |  poly_tan.c                                                               |
        !             3:  |                                                                           |
        !             4:  | Compute the tan of a FPU_REG, using a polynomial approximation.           |
        !             5:  |                                                                           |
        !             6:  | Copyright (C) 1992    W. Metzenthen, 22 Parker St, Ormond, Vic 3163,      |
        !             7:  |                       Australia.  E-mail [email protected]    |
        !             8:  |                                                                           |
        !             9:  |                                                                           |
        !            10:  +---------------------------------------------------------------------------*/
        !            11: 
        !            12: #include "fpu_system.h"
        !            13: #include "exception.h"
        !            14: #include "reg_constant.h"
        !            15: #include "fpu_emu.h"
        !            16: 
        !            17: 
        !            18: #define        HIPOWERop       3       /* odd poly, positive terms */
        !            19: static unsigned short  oddplterms[HIPOWERop][4] =
        !            20:        {
        !            21:        { 0x846a, 0x42d1, 0xb544, 0x921f},
        !            22:        { 0x6fb2, 0x0215, 0x95c0, 0x099c},
        !            23:        { 0xfce6, 0x0cc8, 0x1c9a, 0x0000}
        !            24:        };
        !            25: 
        !            26: #define        HIPOWERon       2       /* odd poly, negative terms */
        !            27: static unsigned short  oddnegterms[HIPOWERon][4] =
        !            28:        {
        !            29:        { 0x6906, 0xe205, 0x25c8, 0x8838},
        !            30:        { 0x1dd7, 0x3fe3, 0x944e, 0x002c}
        !            31:        };
        !            32: 
        !            33: #define        HIPOWERep       2       /* even poly, positive terms */
        !            34: static unsigned short  evenplterms[HIPOWERep][4] =
        !            35:        {
        !            36:        { 0xdb8f, 0x3761, 0x1432, 0x2acf},
        !            37:        { 0x16eb, 0x13c1, 0x3099, 0x0003}
        !            38:        };
        !            39: 
        !            40: #define        HIPOWERen       2       /* even poly, negative terms */
        !            41: static unsigned short  evennegterms[HIPOWERen][4] =
        !            42:        {
        !            43:        { 0x3a7c, 0xe4c5, 0x7f87, 0x2945},
        !            44:        { 0x572b, 0x664c, 0xc543, 0x018c}
        !            45:        };
        !            46: 
        !            47: 
        !            48: /*--- poly_tan() ------------------------------------------------------------+
        !            49:  |                                                                           |
        !            50:  +---------------------------------------------------------------------------*/
        !            51: void   poly_tan(FPU_REG *arg, FPU_REG *y_reg)
        !            52: {
        !            53:   char         invert = 0;
        !            54:   short                exponent;
        !            55:   FPU_REG       odd_poly, even_poly, pos_poly, neg_poly;
        !            56:   FPU_REG       argSq;
        !            57:   long long     arg_signif, argSqSq;
        !            58:   
        !            59: 
        !            60:   exponent = arg->exp - EXP_BIAS;
        !            61:   
        !            62:   if ( arg->tag == TW_Zero )
        !            63:     {
        !            64:       /* Return 0.0 */
        !            65:       reg_move(&CONST_Z, y_reg);
        !            66:       return;
        !            67:     }
        !            68: 
        !            69:   if ( exponent >= -1 )
        !            70:     {
        !            71:       /* argument is in the range  [0.5 .. 1.0] */
        !            72:       if ( exponent >= 0 )
        !            73:        {
        !            74: #ifdef PARANOID
        !            75:          if ( (exponent == 0) && 
        !            76:              (arg->sigl == 0) && (arg->sigh == 0x80000000) )
        !            77: #endif PARANOID
        !            78:            {
        !            79:              arith_overflow(y_reg);
        !            80:              return;
        !            81:            }
        !            82: #ifdef PARANOID
        !            83:          EXCEPTION(EX_INTERNAL|0x104); /* There must be a logic error */
        !            84: #endif PARANOID
        !            85:        }
        !            86:       /* The argument is in the range  [0.5 .. 1.0) */
        !            87:       /* Convert the argument to a number in the range  (0.0 .. 0.5] */
        !            88:       *((long long *)(&arg->sigl)) = - *((long long *)(&arg->sigl));
        !            89:       normalize(arg);  /* Needed later */
        !            90:       exponent = arg->exp - EXP_BIAS;
        !            91:       invert = 1;
        !            92:     }
        !            93: 
        !            94: #ifdef PARANOID
        !            95:   if ( arg->sign != 0 )        /* Can't hack a number < 0.0 */
        !            96:     { arith_invalid(y_reg); return; }
        !            97: #endif PARANOID
        !            98: 
        !            99:   *(long long *)&arg_signif = *(long long *)&(arg->sigl);
        !           100:   if ( exponent < -1 )
        !           101:     {
        !           102:       /* shift the argument right by the required places */
        !           103:       if ( shrx(&arg_signif, -1-exponent) >= 0x80000000U )
        !           104:        arg_signif++;   /* round up */
        !           105:     }
        !           106: 
        !           107:   mul64(&arg_signif, &arg_signif, (long long *)(&argSq.sigl));
        !           108:   mul64((long long *)(&argSq.sigl), (long long *)(&argSq.sigl), &argSqSq);
        !           109: 
        !           110:   /* will be a valid positive nr with expon = 0 */
        !           111:   *(short *)&(pos_poly.sign) = 0;
        !           112:   pos_poly.exp = EXP_BIAS;
        !           113: 
        !           114:   /* Do the basic fixed point polynomial evaluation */
        !           115:   polynomial(&pos_poly.sigl, (unsigned *)&argSqSq, oddplterms, HIPOWERop-1);
        !           116: 
        !           117:   /* will be a valid positive nr with expon = 0 */
        !           118:   *(short *)&(neg_poly.sign) = 0;
        !           119:   neg_poly.exp = EXP_BIAS;
        !           120: 
        !           121:   /* Do the basic fixed point polynomial evaluation */
        !           122:   polynomial(&neg_poly.sigl, (unsigned *)&argSqSq, oddnegterms, HIPOWERon-1);
        !           123:   mul64((long long *)(&argSq.sigl), (long long *)(&neg_poly.sigl),
        !           124:        (long long *)(&neg_poly.sigl));
        !           125: 
        !           126:   /* Subtract the mantissas */
        !           127:   *((long long *)(&pos_poly.sigl)) -= *((long long *)(&neg_poly.sigl));
        !           128: 
        !           129:   /* Convert to 64 bit signed-compatible */
        !           130:   pos_poly.exp -= 1;
        !           131: 
        !           132:   reg_move(&pos_poly, &odd_poly);
        !           133:   normalize(&odd_poly);
        !           134:   
        !           135:   reg_mul(&odd_poly, arg, &odd_poly);
        !           136:   reg_u_add(&odd_poly, arg, &odd_poly);        /* This is just the odd polynomial */
        !           137: 
        !           138: 
        !           139:   /* will be a valid positive nr with expon = 0 */
        !           140:   *(short *)&(pos_poly.sign) = 0;
        !           141:   pos_poly.exp = EXP_BIAS;
        !           142:   
        !           143:   /* Do the basic fixed point polynomial evaluation */
        !           144:   polynomial(&pos_poly.sigl, (unsigned *)&argSqSq, evenplterms, HIPOWERep-1);
        !           145:   mul64((long long *)(&argSq.sigl),
        !           146:        (long long *)(&pos_poly.sigl), (long long *)(&pos_poly.sigl));
        !           147:   
        !           148:   /* will be a valid positive nr with expon = 0 */
        !           149:   *(short *)&(neg_poly.sign) = 0;
        !           150:   neg_poly.exp = EXP_BIAS;
        !           151: 
        !           152:   /* Do the basic fixed point polynomial evaluation */
        !           153:   polynomial(&neg_poly.sigl, (unsigned *)&argSqSq, evennegterms, HIPOWERen-1);
        !           154: 
        !           155:   /* Subtract the mantissas */
        !           156:   *((long long *)(&neg_poly.sigl)) -= *((long long *)(&pos_poly.sigl));
        !           157:   /* and multiply by argSq */
        !           158: 
        !           159:   /* Convert argSq to a valid reg number */
        !           160:   *(short *)&(argSq.sign) = 0;
        !           161:   argSq.exp = EXP_BIAS - 1;
        !           162:   normalize(&argSq);
        !           163: 
        !           164:   /* Convert to 64 bit signed-compatible */
        !           165:   neg_poly.exp -= 1;
        !           166: 
        !           167:   reg_move(&neg_poly, &even_poly);
        !           168:   normalize(&even_poly);
        !           169: 
        !           170:   reg_mul(&even_poly, &argSq, &even_poly);
        !           171:   reg_add(&even_poly, &argSq, &even_poly);
        !           172:   reg_sub(&CONST_1, &even_poly, &even_poly);  /* This is just the even polynomial */
        !           173: 
        !           174:   /* Now ready to copy the results */
        !           175:   if ( invert )
        !           176:     { reg_div(&even_poly, &odd_poly, y_reg); }
        !           177:   else
        !           178:     { reg_div(&odd_poly, &even_poly, y_reg); }
        !           179: 
        !           180: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.