| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114 |
- //go:build !tinygo && !noasm
- // Copyright 2014 Xuanyi Chew. All rights reserved.
- // Use of this source code is governed by a BSD-style
- // license that can be found in the LICENSE file.
- //
- // The original code is lifted from the Go standard library which is governed by
- // a BSD-style licence which can be found here: https://golang.org/LICENSE
- #include "textflag.h"
- // The method is based on a paper by Naoki Shibata: "Efficient evaluation
- // methods of elementary functions suitable for SIMD computation", Proc.
- // of International Supercomputing Conference 2010 (ISC'10), pp. 25 -- 32
- // (May 2010). The paper is available at
- // http://www.springerlink.com/content/340228x165742104/
- //
- // The original code and the constants below are from the author's
- // implementation available at http://freshmeat.net/projects/sleef.
- // The README file says, "The software is in public domain.
- // You can use the software without any obligation."
- //
- // This code is a simplified version of the original.
- // The magic numbers for the float32 are lifted from the same project
-
- #define LN2 0.693147182464599609375 // log_e(2)
- #define LOG2E 1.44269502162933349609375 // 1/LN2
- #define LN2U 0.693145751953125 // upper half LN2
- #define LN2L 1.428606765330187045e-06 // lower half LN2
- #define T0 1.0
- #define T1 0.5
- #define T2 0.166665524244308471679688
- #define T3 0.0416710823774337768554688
- #define T4 0.00836596917361021041870117
- #define PosInf 0x7F800000
- #define NegInf 0xFF800000
- // func archExp(x float32) float32
- TEXT ·archExp(SB),NOSPLIT,$0
- // test bits for not-finite
- MOVL x+0(FP), BX
- MOVQ $~(1<<31), AX // sign bit mask
- MOVL BX, DX
- ANDL AX, DX
- MOVL $PosInf, AX
- CMPL AX, DX
- JLE notFinite
- MOVL BX, X0
- MOVSS $LOG2E, X1
- MULSS X0, X1
- CVTSS2SL X1, BX // BX = exponent
- CVTSL2SS BX, X1
- MOVSS $LN2U, X2
- MULSS X1, X2
- SUBSS X2, X0
- MOVSS $LN2L, X2
- MULSS X1, X2
- SUBSS X2, X0
- // reduce argument
- MULSS $0.0625, X0
- // Taylor series evaluation
- ADDSS $T4, X1
- MULSS X0, X1
- ADDSS $T3, X1
- MULSS X0, X1
- ADDSS $T2, X1
- MULSS X0, X1
- ADDSS $T1, X1
- MULSS X0, X1
- ADDSS $T0, X1
- MULSS X1, X0
- MOVSS $2.0, X1
- ADDSS X0, X1
- MULSS X1, X0
- MOVSS $2.0, X1
- ADDSS X0, X1
- MULSS X1, X0
- MOVSS $2.0, X1
- ADDSS X0, X1
- MULSS X1, X0
- MOVSS $2.0, X1
- ADDSS X0, X1
- MULSS X1, X0
- ADDSS $1.0, X0
- // return fr * 2**exponent
- MOVL $0x7F, AX // bias
- ADDL AX, BX
- JLE underflow
- CMPL BX, $0xFF
- JGE overflow
- MOVL $23, CX
- SHLQ CX, BX
- MOVL BX, X1
- MULSS X1, X0
- MOVSS X0, ret+8(FP)
- RET
- notFinite:
- // test bits for -Inf
- MOVL $NegInf, AX
- CMPQ AX, BX
- JNE notNegInf
- // -Inf, return 0
- underflow: // return 0
- MOVL $0, AX
- MOVL AX, ret+8(FP)
- RET
- overflow: // return +Inf
- MOVL $PosInf, BX
- notNegInf: // NaN or +Inf, return x
- MOVL BX, ret+8(FP)
- RET
- TEXT ·archExp2(SB),NOSPLIT,$0
- JMP ·exp2(SB)
|