69 lines
2.2 KiB
ArmAsm
69 lines
2.2 KiB
ArmAsm
|
/*
|
||
|
* Copyright (c) 1992 Helsinki University of Technology
|
||
|
* All Rights Reserved.
|
||
|
*
|
||
|
* Permission to use, copy, modify and distribute this software and its
|
||
|
* documentation is hereby granted, provided that both the copyright
|
||
|
* notice and this permission notice appear in all copies of the
|
||
|
* software, derivative works or modified versions, and any portions
|
||
|
* thereof, and that both notices appear in supporting documentation.
|
||
|
*
|
||
|
* HELSINKI UNIVERSITY OF TECHNOLOGY ALLOWS FREE USE OF THIS SOFTWARE IN
|
||
|
* ITS "AS IS" CONDITION. HELSINKI UNIVERSITY OF TECHNOLOGY DISCLAIMS ANY
|
||
|
* LIABILITY OF ANY KIND FOR ANY DAMAGES WHATSOEVER RESULTING FROM THE
|
||
|
* USE OF THIS SOFTWARE.
|
||
|
*/
|
||
|
/*
|
||
|
* HISTORY
|
||
|
* 29-Apr-92 Tero Kivinen (kivinen) at Helsinki University of Technology
|
||
|
* Created.
|
||
|
*
|
||
|
* $Id: modf.s,v 1.1.1.1 1993/09/17 18:43:47 phil Exp $
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* double modf (value, iptr)
|
||
|
* double value, *iptr;
|
||
|
*
|
||
|
* Modf returns the fractional part of "value",
|
||
|
* and stores the integer part indirectly through "iptr".
|
||
|
*/
|
||
|
|
||
|
#include <machine/asm.h>
|
||
|
|
||
|
ENTRY(modf)
|
||
|
FRAME
|
||
|
movl 8(fp),f0 /* value */
|
||
|
movd 12(fp),r0 /* higher 32 bit of value */
|
||
|
lshd -20,r0 /* extract exponent */
|
||
|
andd 0x7ff,r0 /* 11 lower bits */
|
||
|
cmpd r0,1023+30 /* compare if it's int part can fit in int */
|
||
|
bgt modf_overflow /* nope else it's ok to truncld it to int*/
|
||
|
truncld f0,r0 /* get integer part */
|
||
|
movdl r0,f2 /* convert back to float */
|
||
|
movl f2,0(16(fp)) /* move integer part to *iptr */
|
||
|
subl f2,f0 /* return fract. part = value - *iptr */
|
||
|
EMARF
|
||
|
ret 0
|
||
|
modf_overflow:
|
||
|
subd 1023+20,r0 /* bias 1023, and upper part of
|
||
|
floating point mantissa part */
|
||
|
movqd -1,r2 /* generate mask to get fract. part */
|
||
|
cmpd r0,32 /* if value > 2^52 (20+32) then all int part */
|
||
|
bhi modf_all_ip
|
||
|
negd r0,r0 /* shift right */
|
||
|
lshd r0,r2 /* here */
|
||
|
comd r2,r2 /* get fractional part complement mask */
|
||
|
movd 8(fp),r1 /* get lower 32 bit of value */
|
||
|
andd r2,r1 /* mask fractional part off leave ip part */
|
||
|
movd r1,0(16(fp)) /* store ip part to *iptr */
|
||
|
movd 12(fp),4(16(fp)) /* store other half to *iptr */
|
||
|
subl 0(16(fp)),f0 /* return fract. part = value - *iptr */
|
||
|
EMARF
|
||
|
ret 0
|
||
|
modf_all_ip:
|
||
|
movl 8(fp),0(16(fp)) /* copy value to *iptr */
|
||
|
movdl 0,f0 /* return 0 for fract. part */
|
||
|
EMARF
|
||
|
ret 0
|