1*c9945492SAndroid Build Coastguard Worker# exp(x) = 2^hi + 2^hi (2^lo - 1) 2*c9945492SAndroid Build Coastguard Worker# where hi+lo = log2e*x with 128bit precision 3*c9945492SAndroid Build Coastguard Worker# exact log2e*x calculation depends on nearest rounding mode 4*c9945492SAndroid Build Coastguard Worker# using the exact multiplication method of Dekker and Veltkamp 5*c9945492SAndroid Build Coastguard Worker 6*c9945492SAndroid Build Coastguard Worker.global expl 7*c9945492SAndroid Build Coastguard Worker.type expl,@function 8*c9945492SAndroid Build Coastguard Workerexpl: 9*c9945492SAndroid Build Coastguard Worker fldt 4(%esp) 10*c9945492SAndroid Build Coastguard Worker 11*c9945492SAndroid Build Coastguard Worker # interesting case: 0x1p-32 <= |x| < 16384 12*c9945492SAndroid Build Coastguard Worker # check if (exponent|0x8000) is in [0xbfff-32, 0xbfff+13] 13*c9945492SAndroid Build Coastguard Worker mov 12(%esp), %ax 14*c9945492SAndroid Build Coastguard Worker or $0x8000, %ax 15*c9945492SAndroid Build Coastguard Worker sub $0xbfdf, %ax 16*c9945492SAndroid Build Coastguard Worker cmp $45, %ax 17*c9945492SAndroid Build Coastguard Worker jbe 2f 18*c9945492SAndroid Build Coastguard Worker test %ax, %ax 19*c9945492SAndroid Build Coastguard Worker fld1 20*c9945492SAndroid Build Coastguard Worker js 1f 21*c9945492SAndroid Build Coastguard Worker # if |x|>=0x1p14 or nan return 2^trunc(x) 22*c9945492SAndroid Build Coastguard Worker fscale 23*c9945492SAndroid Build Coastguard Worker fstp %st(1) 24*c9945492SAndroid Build Coastguard Worker ret 25*c9945492SAndroid Build Coastguard Worker # if |x|<0x1p-32 return 1+x 26*c9945492SAndroid Build Coastguard Worker1: faddp 27*c9945492SAndroid Build Coastguard Worker ret 28*c9945492SAndroid Build Coastguard Worker 29*c9945492SAndroid Build Coastguard Worker # should be 0x1.71547652b82fe178p0L == 0x3fff b8aa3b29 5c17f0bc 30*c9945492SAndroid Build Coastguard Worker # it will be wrong on non-nearest rounding mode 31*c9945492SAndroid Build Coastguard Worker2: fldl2e 32*c9945492SAndroid Build Coastguard Worker subl $44, %esp 33*c9945492SAndroid Build Coastguard Worker # hi = log2e_hi*x 34*c9945492SAndroid Build Coastguard Worker # 2^hi = exp2l(hi) 35*c9945492SAndroid Build Coastguard Worker fmul %st(1),%st 36*c9945492SAndroid Build Coastguard Worker fld %st(0) 37*c9945492SAndroid Build Coastguard Worker fstpt (%esp) 38*c9945492SAndroid Build Coastguard Worker fstpt 16(%esp) 39*c9945492SAndroid Build Coastguard Worker fstpt 32(%esp) 40*c9945492SAndroid Build Coastguard Worker.hidden __exp2l 41*c9945492SAndroid Build Coastguard Worker call __exp2l 42*c9945492SAndroid Build Coastguard Worker # if 2^hi == inf return 2^hi 43*c9945492SAndroid Build Coastguard Worker fld %st(0) 44*c9945492SAndroid Build Coastguard Worker fstpt (%esp) 45*c9945492SAndroid Build Coastguard Worker cmpw $0x7fff, 8(%esp) 46*c9945492SAndroid Build Coastguard Worker je 1f 47*c9945492SAndroid Build Coastguard Worker fldt 32(%esp) 48*c9945492SAndroid Build Coastguard Worker fldt 16(%esp) 49*c9945492SAndroid Build Coastguard Worker # fpu stack: 2^hi x hi 50*c9945492SAndroid Build Coastguard Worker # exact mult: x*log2e 51*c9945492SAndroid Build Coastguard Worker fld %st(1) 52*c9945492SAndroid Build Coastguard Worker # c = 0x1p32+1 53*c9945492SAndroid Build Coastguard Worker pushl $0x41f00000 54*c9945492SAndroid Build Coastguard Worker pushl $0x00100000 55*c9945492SAndroid Build Coastguard Worker fldl (%esp) 56*c9945492SAndroid Build Coastguard Worker # xh = x - c*x + c*x 57*c9945492SAndroid Build Coastguard Worker # xl = x - xh 58*c9945492SAndroid Build Coastguard Worker fmulp 59*c9945492SAndroid Build Coastguard Worker fld %st(2) 60*c9945492SAndroid Build Coastguard Worker fsub %st(1), %st 61*c9945492SAndroid Build Coastguard Worker faddp 62*c9945492SAndroid Build Coastguard Worker fld %st(2) 63*c9945492SAndroid Build Coastguard Worker fsub %st(1), %st 64*c9945492SAndroid Build Coastguard Worker # yh = log2e_hi - c*log2e_hi + c*log2e_hi 65*c9945492SAndroid Build Coastguard Worker pushl $0x3ff71547 66*c9945492SAndroid Build Coastguard Worker pushl $0x65200000 67*c9945492SAndroid Build Coastguard Worker fldl (%esp) 68*c9945492SAndroid Build Coastguard Worker # fpu stack: 2^hi x hi xh xl yh 69*c9945492SAndroid Build Coastguard Worker # lo = hi - xh*yh + xl*yh 70*c9945492SAndroid Build Coastguard Worker fld %st(2) 71*c9945492SAndroid Build Coastguard Worker fmul %st(1), %st 72*c9945492SAndroid Build Coastguard Worker fsubp %st, %st(4) 73*c9945492SAndroid Build Coastguard Worker fmul %st(1), %st 74*c9945492SAndroid Build Coastguard Worker faddp %st, %st(3) 75*c9945492SAndroid Build Coastguard Worker # yl = log2e_hi - yh 76*c9945492SAndroid Build Coastguard Worker pushl $0x3de705fc 77*c9945492SAndroid Build Coastguard Worker pushl $0x2f000000 78*c9945492SAndroid Build Coastguard Worker fldl (%esp) 79*c9945492SAndroid Build Coastguard Worker # fpu stack: 2^hi x lo xh xl yl 80*c9945492SAndroid Build Coastguard Worker # lo += xh*yl + xl*yl 81*c9945492SAndroid Build Coastguard Worker fmul %st, %st(2) 82*c9945492SAndroid Build Coastguard Worker fmulp %st, %st(1) 83*c9945492SAndroid Build Coastguard Worker fxch %st(2) 84*c9945492SAndroid Build Coastguard Worker faddp 85*c9945492SAndroid Build Coastguard Worker faddp 86*c9945492SAndroid Build Coastguard Worker # log2e_lo 87*c9945492SAndroid Build Coastguard Worker pushl $0xbfbe 88*c9945492SAndroid Build Coastguard Worker pushl $0x82f0025f 89*c9945492SAndroid Build Coastguard Worker pushl $0x2dc582ee 90*c9945492SAndroid Build Coastguard Worker fldt (%esp) 91*c9945492SAndroid Build Coastguard Worker addl $36,%esp 92*c9945492SAndroid Build Coastguard Worker # fpu stack: 2^hi x lo log2e_lo 93*c9945492SAndroid Build Coastguard Worker # lo += log2e_lo*x 94*c9945492SAndroid Build Coastguard Worker # return 2^hi + 2^hi (2^lo - 1) 95*c9945492SAndroid Build Coastguard Worker fmulp %st, %st(2) 96*c9945492SAndroid Build Coastguard Worker faddp 97*c9945492SAndroid Build Coastguard Worker f2xm1 98*c9945492SAndroid Build Coastguard Worker fmul %st(1), %st 99*c9945492SAndroid Build Coastguard Worker faddp 100*c9945492SAndroid Build Coastguard Worker1: addl $44, %esp 101*c9945492SAndroid Build Coastguard Worker ret 102