mirror of
				https://sourceware.org/git/glibc.git
				synced 2025-10-24 13:33:08 +03:00 
			
		
		
		
	* sysdeps/powerpc/powerpc32/fpu/s_lround.S (__lround): Fixed erroneous result when x is +/-nextafter(+/-0.5,-/+1) i.e. all 1's in the mantissa. * sysdeps/powerpc/powerpc32/power4/fpu/s_llround.S (__llround): Likewise. Also account for when x is an odd number between 2^52 and 2^53-1. * sysdeps/powerpc/powerpc64/fpu/s_llround.S (__llround): Likewise. * sysdeps/powerpc/powerpc64/fpu/s_llroundf.S (__llroundf): Likewise. * math/libm-test.inc (lround_test, llround_test): Added test cases to detect aforementioned erroneous conditions.
		
			
				
	
	
		
			93 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
			
		
		
	
	
			93 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
| /* llround function.  PowerPC64 version.
 | |
|    Copyright (C) 2004, 2006, 2007 Free Software Foundation, Inc.
 | |
|    This file is part of the GNU C Library.
 | |
| 
 | |
|    The GNU C Library is free software; you can redistribute it and/or
 | |
|    modify it under the terms of the GNU Lesser General Public
 | |
|    License as published by the Free Software Foundation; either
 | |
|    version 2.1 of the License, or (at your option) any later version.
 | |
| 
 | |
|    The GNU C Library is distributed in the hope that it will be useful,
 | |
|    but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | |
|    Lesser General Public License for more details.
 | |
| 
 | |
|    You should have received a copy of the GNU Lesser General Public
 | |
|    License along with the GNU C Library; if not, write to the Free
 | |
|    Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
 | |
|    02111-1307 USA.  */
 | |
| 
 | |
| #include <sysdep.h>
 | |
| #include <math_ldbl_opt.h>
 | |
| 
 | |
| 	.section	".toc","aw"
 | |
| .LC0:	/* 2^52 */
 | |
| 	.tc FD_43300000_0[TC],0x4330000000000000
 | |
| .LC1:	/* 0.5 */
 | |
| 	.tc FD_3fe00000_0[TC],0x3fe0000000000000
 | |
| 	.section	".text"
 | |
| 	
 | |
| /* long long [r3] llround (double x [fp1])
 | |
|    IEEE 1003.1 llround function.  IEEE specifies "round to the nearest 
 | |
|    integer value, rounding halfway cases away from zero, regardless of
 | |
|    the current rounding mode."  However PowerPC Architecture defines
 | |
|    "round to Nearest" as "Choose the best approximation. In case of a 
 | |
|    tie, choose the one that is even (least significant bit o).". 
 | |
|    So we can't use the PowerPC "round to Nearest" mode. Instead we set
 | |
|    "round toward Zero" mode and round by adding +-0.5 before rounding
 | |
|    to the integer value.
 | |
| 
 | |
|    It is necessary to detect when x is (+-)0x1.fffffffffffffp-2
 | |
|    because adding +-0.5 in this case will cause an erroneous shift,
 | |
|    carry and round.  We simply return 0 if 0.5 > x > -0.5.  Likewise
 | |
|    if x is and odd number between +-(2^52 and 2^53-1) a shift and
 | |
|    carry will erroneously round if biased with +-0.5.  Therefore if x
 | |
|    is greater/less than +-2^52 we don't need to bias the number with
 | |
|    +-0.5.  */
 | |
| 
 | |
| ENTRY (__llround)
 | |
| 	CALL_MCOUNT 0
 | |
| 	lfd	fp9,.LC0@toc(2)	/* Load 2^52 into fpr9.  */
 | |
| 	lfd	fp10,.LC1@toc(2)/* Load 0.5 into fpr10.  */
 | |
| 	fabs	fp2,fp1		/* Get the absolute value of x.  */
 | |
| 	fsub	fp12,fp10,fp10	/* Compute 0.0 into fp12.  */
 | |
| 	fcmpu	cr6,fp2,fp10	/* if |x| < 0.5  */
 | |
| 	fcmpu	cr4,fp2,fp9	/* if |x| >= 2^52  */
 | |
| 	fcmpu	cr3,fp1,fp12	/* x is negative? x < 0.0  */
 | |
| 	blt-	cr6,.Lretzero	/* 0.5 > x < -0.5 so just return 0.  */
 | |
| 	bge-	cr4,.Lnobias	/* 2^52 > x < -2^52 just convert with no bias.  */
 | |
| 	fadd	fp3,fp2,fp10	/* |x|+=0.5 bias to prepare to round.  */
 | |
| 	bge	cr3,.Lconvert	/* x is positive so don't negate x.  */
 | |
| 	fnabs	fp3,fp3		/* -(|x|+=0.5)  */
 | |
| .Lconvert:
 | |
| 	fctidz	fp4,fp3		/* Convert to Integer double word round toward 0.  */
 | |
| 	stfd	fp4,-16(r1)
 | |
| 	nop
 | |
| 	nop
 | |
| 	nop
 | |
| 	ld	r3,-16(r1)	/* Load return as integer.  */
 | |
| .Lout:
 | |
| 	blr
 | |
| .Lretzero:			/* 0.5 > x > -0.5  */
 | |
| 	li	r3,0		/* return 0.  */
 | |
| 	b	.Lout
 | |
| .Lnobias:
 | |
| 	fmr	fp3,fp1
 | |
| 	b	.Lconvert
 | |
| 	END (__llround)
 | |
| 
 | |
| strong_alias (__llround, __lround)
 | |
| weak_alias (__llround, llround)
 | |
| weak_alias (__lround, lround)
 | |
| 
 | |
| #ifdef NO_LONG_DOUBLE
 | |
| weak_alias (__llround, llroundl)
 | |
| strong_alias (__llround, __llroundl)
 | |
| weak_alias (__lround, lroundl)
 | |
| strong_alias (__lround, __lroundl)
 | |
| #endif
 | |
| #if LONG_DOUBLE_COMPAT(libm, GLIBC_2_1)
 | |
| compat_symbol (libm, __llround, llroundl, GLIBC_2_1)
 | |
| compat_symbol (libm, __lround, lroundl, GLIBC_2_1)
 | |
| #endif
 |