exp387.asm
来自「开放源码的编译器open watcom 1.6.0版的源代码」· 汇编 代码 · 共 196 行
ASM
196 行
;*****************************************************************************
;*
;* Open Watcom Project
;*
;* Portions Copyright (c) 1983-2002 Sybase, Inc. All Rights Reserved.
;*
;* ========================================================================
;*
;* This file contains Original Code and/or Modifications of Original
;* Code as defined in and that are subject to the Sybase Open Watcom
;* Public License version 1.0 (the 'License'). You may not use this file
;* except in compliance with the License. BY USING THIS FILE YOU AGREE TO
;* ALL TERMS AND CONDITIONS OF THE LICENSE. A copy of the License is
;* provided with the Original Code and Modifications, and is also
;* available at www.sybase.com/developer/opensource.
;*
;* The Original Code and all software distributed under the License are
;* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
;* EXPRESS OR IMPLIED, AND SYBASE AND ALL CONTRIBUTORS HEREBY DISCLAIM
;* ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF
;* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR
;* NON-INFRINGEMENT. Please see the License for the specific language
;* governing rights and limitations under the License.
;*
;* ========================================================================
;*
;* Description: WHEN YOU FIGURE OUT WHAT THIS FILE DOES, PLEASE
;* DESCRIBE IT HERE!
;*
;*****************************************************************************
;
; interface to floating point library for trig functions
;
ifdef __386__
.387
else
.8087
endif
include mdef.inc
include struct.inc
include math387.inc
xref __8087 ; indicate that NDP instructions are present
modstart exp387
;datasegment
;enddata
xref "C",__math87_err
xdefp "C",exp ; calc exp(fac1)
ifndef __386__
if _MODEL and _BIG_CODE
argx equ 6
else
argx equ 4
endif
endif
ifdef __386__
datasegment
endif
MaxExpVal dq 709.7827128933839
;MaxExpVal DB 0eeH,39H,0faH,0feH,42H,2eH,86H,040H
MinExpVal dq -709.7827128933839
;MinExpVal DB 0eeH,39H,0faH,0feH,42H,2eH,86H,0c0H
ifdef __386__
enddata
endif
public __@DEXP1
defp __@DEXP1 ; calc. exp(x) with no check
prolog
sub _SP,16 ; allocate space
jmp short calc_exp ; go calculate exp(x)
endproc __@DEXP1
public IF@EXP
public IF@DEXP
defp IF@DEXP
defp IF@EXP
; calculate exp(st(0)) using 387
mov AL,FUNC_EXP ; indicate exp function
public __@DEXP
defp __@DEXP
prolog
sub _SP,16 ; allocate space
fcom qword ptr MaxExpVal ; compare with largest valid argument
fstsw word ptr -16+0[_BP] ; get status
fwait ; wait for it
mov AH,-16+1[_BP] ; get status word
sahf ; set flags
_if a ; if number is too large
cmp AL,FUNC_POW ; - if not "pow" function
_if ne ; - then
fstp qword ptr -16+0[_BP] ; - - push argument on stack
mov -16+8[_BP],_AX ; - - set func code
mov -16+12[_BP],_DX ; - - save DX (-3s) clobbers EDX 17-mar-92
call __math87_err ; - - error (func code in AL)
ifdef __386__
ifdef __STACK__
push EDX ; - - load result into 8087
push EAX ; - - ...
fld qword ptr 0[ESP] ; - - ...
mov EDX,-16+12[EBP] ; - - restore EDX 17-mar-92
fwait ; - - ...
endif
endif
_endif ; - endif
mov AL,1 ; - indicate error
mov _SP,_BP ; - reset stack pointer
epilog
ret ; - return
_endif ; endif
calc_exp:
_guess ; - guess: argument too small
fcom qword ptr MinExpVal ; - - compare with smallest valid arg
fstsw word ptr -16+0[_BP] ; - - get status
fwait ; - - wait for it
mov AH,-16+1[_BP] ; - - get status word
sahf ; - - set flags
_quif a ; - - quit if number is not too small
fldz ; - - set result to 0.0
_admit ; - guess: 387
ifndef __FPI87__
chk387
_quif ne ; - - quit if not 387
endif ; __FPI87__
fldl2e ; - - load log2(e)
fmulp st(1),st(0) ; - - calculate argument * log2(e)
fld st(0) ; - - duplicate fac1
frndint ; - - round to an integer
fxch st(1) ; - - flip around
fsub st,st(1) ; - - calc. x - int(x)
f2xm1 ; - - calc. 2 ** ( x - int(x) )
fld1 ; - - load 1.0
faddp st(1),st(0) ; - - add 1 to result
fscale ; - - multiply by 2 ** int(x)
ifndef __FPI87__
_admit ; - admit: argument in valid range
fldl2e ; - - load log2(e)
fmulp st(1),st(0) ; - - calculate argument * log2(e)
fld1 ; - - load -1 for dividing by 2
fchs ; - - ...
fld st(1) ; - - duplicate x
fstcw word ptr -16+0[_BP] ; - - save control word
fwait ; - - wait
mov AL,-16+1[_BP] ; - - save it
and byte ptr -16+1[_BP],0F3h; - - mask out rounding control bits
or byte ptr -16+1[_BP],04h ; - - set to round down to - infinity
fldcw word ptr -16+0[_BP] ; - - load new control word
frndint ; - - round to an integer
mov -16+1[_BP],AL ; - - set original control word
fldcw word ptr -16+0[_BP] ; - - restore original control word
fxch st(2) ; - - exchange int(x) with x
fsub st,st(2) ; - - calc. x - int(x)
fscale ; - - divide by 2
f2xm1 ; - - calc 2 ** x
fsubrp st(1),st ; - - subtract -1 (or add 1)
fmul st,st(0) ; - - square the result
fscale ; - - multiply by 2 ** int(x)
endif ; __FPI87__
_endguess ; - endguess
fstp st(1) ; - copy answer up
mov AL,0 ; - indicate no error
mov _SP,_BP ; reset stack pointer
epilog
ret ; return
endproc __@DEXP
endproc IF@EXP
endproc IF@DEXP
defp exp
ifdef __386__
fld qword ptr 4[ESP]; load argument x
call IF@DEXP ; calculate exp(x)
loadres ; load result
else
prolog
fld qword ptr argx[BP]; load argument x
lcall IF@DEXP ; calculate exp(x)
epilog
endif
ret_pop 8 ; return
endproc exp
endmod
end
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?