1 // SPDX-License-Identifier: GPL-2.0-or-later 2 /* 3 NetWinder Floating Point Emulator 4 (c) Rebel.COM, 1998,1999 5 6 Direct questions, comments to Scott Bambrough <scottb@netwinder.org> 7 8 */ 9 10 #include "fpa11.h" 11 #include "softfloat.h" 12 #include "fpopcode.h" 13 14 union float64_components { 15 float64 f64; 16 unsigned int i[2]; 17 }; 18 19 float64 float64_exp(float64 Fm); 20 float64 float64_ln(float64 Fm); 21 float64 float64_sin(float64 rFm); 22 float64 float64_cos(float64 rFm); 23 float64 float64_arcsin(float64 rFm); 24 float64 float64_arctan(float64 rFm); 25 float64 float64_log(float64 rFm); 26 float64 float64_tan(float64 rFm); 27 float64 float64_arccos(float64 rFm); 28 float64 float64_pow(float64 rFn, float64 rFm); 29 float64 float64_pol(float64 rFn, float64 rFm); 30 31 static float64 float64_rsf(struct roundingData *roundData, float64 rFn, float64 rFm) 32 { 33 return float64_sub(roundData, rFm, rFn); 34 } 35 36 static float64 float64_rdv(struct roundingData *roundData, float64 rFn, float64 rFm) 37 { 38 return float64_div(roundData, rFm, rFn); 39 } 40 41 static float64 (*const dyadic_double[16])(struct roundingData*, float64 rFn, float64 rFm) = { 42 [ADF_CODE >> 20] = float64_add, 43 [MUF_CODE >> 20] = float64_mul, 44 [SUF_CODE >> 20] = float64_sub, 45 [RSF_CODE >> 20] = float64_rsf, 46 [DVF_CODE >> 20] = float64_div, 47 [RDF_CODE >> 20] = float64_rdv, 48 [RMF_CODE >> 20] = float64_rem, 49 50 /* strictly, these opcodes should not be implemented */ 51 [FML_CODE >> 20] = float64_mul, 52 [FDV_CODE >> 20] = float64_div, 53 [FRD_CODE >> 20] = float64_rdv, 54 }; 55 56 static float64 float64_mvf(struct roundingData *roundData,float64 rFm) 57 { 58 return rFm; 59 } 60 61 static float64 float64_mnf(struct roundingData *roundData,float64 rFm) 62 { 63 union float64_components u; 64 65 u.f64 = rFm; 66 #ifdef __ARMEB__ 67 u.i[0] ^= 0x80000000; 68 #else 69 u.i[1] ^= 0x80000000; 70 #endif 71 72 return u.f64; 73 } 74 75 static float64 float64_abs(struct roundingData *roundData,float64 rFm) 76 { 77 union float64_components u; 78 79 u.f64 = rFm; 80 #ifdef __ARMEB__ 81 u.i[0] &= 0x7fffffff; 82 #else 83 u.i[1] &= 0x7fffffff; 84 #endif 85 86 return u.f64; 87 } 88 89 static float64 (*const monadic_double[16])(struct roundingData *, float64 rFm) = { 90 [MVF_CODE >> 20] = float64_mvf, 91 [MNF_CODE >> 20] = float64_mnf, 92 [ABS_CODE >> 20] = float64_abs, 93 [RND_CODE >> 20] = float64_round_to_int, 94 [URD_CODE >> 20] = float64_round_to_int, 95 [SQT_CODE >> 20] = float64_sqrt, 96 [NRM_CODE >> 20] = float64_mvf, 97 }; 98 99 unsigned int DoubleCPDO(struct roundingData *roundData, const unsigned int opcode, FPREG * rFd) 100 { 101 FPA11 *fpa11 = GET_FPA11(); 102 float64 rFm; 103 unsigned int Fm, opc_mask_shift; 104 105 Fm = getFm(opcode); 106 if (CONSTANT_FM(opcode)) { 107 rFm = getDoubleConstant(Fm); 108 } else { 109 switch (fpa11->fType[Fm]) { 110 case typeSingle: 111 rFm = float32_to_float64(fpa11->fpreg[Fm].fSingle); 112 break; 113 114 case typeDouble: 115 rFm = fpa11->fpreg[Fm].fDouble; 116 break; 117 118 default: 119 return 0; 120 } 121 } 122 123 opc_mask_shift = (opcode & MASK_ARITHMETIC_OPCODE) >> 20; 124 if (!MONADIC_INSTRUCTION(opcode)) { 125 unsigned int Fn = getFn(opcode); 126 float64 rFn; 127 128 switch (fpa11->fType[Fn]) { 129 case typeSingle: 130 rFn = float32_to_float64(fpa11->fpreg[Fn].fSingle); 131 break; 132 133 case typeDouble: 134 rFn = fpa11->fpreg[Fn].fDouble; 135 break; 136 137 default: 138 return 0; 139 } 140 141 if (dyadic_double[opc_mask_shift]) { 142 rFd->fDouble = dyadic_double[opc_mask_shift](roundData, rFn, rFm); 143 } else { 144 return 0; 145 } 146 } else { 147 if (monadic_double[opc_mask_shift]) { 148 rFd->fDouble = monadic_double[opc_mask_shift](roundData, rFm); 149 } else { 150 return 0; 151 } 152 } 153 154 return 1; 155 } 156
Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.