libgcc1.asm   [plain text]


/* Assembly support functions for libgcc.
 *
 *   Copyright (C) 1997 Free Software Foundation, Inc.
 *   Contributed by Cygnus Support
 * 
 * This file is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the
 * Free Software Foundation; either version 2, or (at your option) any
 * later version.
 * 
 * In addition to the permissions in the GNU General Public License, the
 * Free Software Foundation gives you unlimited permission to link the
 * compiled version of this file into combinations with other programs,
 * and to distribute those combinations without any restriction coming
 * from the use of this file.  (The General Public License restrictions
 * do apply in other respects; for example, they cover modification of
 * the file, and distribution when not linked into a combine
 * executable.)
 * 
 * This file 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
 * General Public License for more details.
 * 
 * You should have received a copy of the GNU General Public License
 * along with this program; see the file COPYING.  If not, write to
 * the Free Software Foundation, 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
 * 
 */


#ifdef L_udivsi3

/* For division, we use the following algorithm:
 *
 *	unsigned
 *	__divsi3 (unsigned a, unsigned b)
 *	{
 *	  unsigned al = a;
 *	  unsigned ah = 0;
 *	  unsigned tmpf;
 *	  int i;
 *
 *	  for (i = 32; i > 0; i--)
 *	    {
 *	      ah = (ah << 1) | (al >> 31);
 *	      tmpf = (ah >= b) ? 1 : 0;
 *	      ah -= ((tmpf) ? b : 0);
 *	      al = (al << 1) | tmpf;
 *	    }
 *
 *	  return al;	// for __udivsi3
 *	  return ah;	// for __umodsi3
 *	}
 */

	.file	"_udivsi3"
	.text
	.globl	__umodsi3
	.globl	__udivsi3
	.type	__umodsi3,@function
	.type	__udivsi3,@function
	.stabs	"libgcc1.asm",100,0,0,__umodsi3
	.stabs	"int:t(0,1)=r(0,1);-2147483648;2147483647;",128,0,0,0
	.stabs	"__umodsi3:F(0,1)",36,0,1,__umodsi3
	.stabs	"a:P(0,1)",64,0,1,2
	.stabs	"b:P(0,1)",64,0,1,3

__umodsi3:
	bra.s .Lmerge		|| orfg	f1,f1,1		; indicate this is __umodsi3
.Lumod:
	.size	__umodsi3,.Lumod-__umodsi3
	.stabs	"",36,0,0,.Lumod-__umodsi3

	.stabs	"__udivsi3:F(0,1)",36,0,1,__udivsi3
	.stabs	"a:P(0,1)",64,0,1,2
	.stabs	"b:P(0,1)",64,0,1,3
__udivsi3:
	andfg	f1,f1,0		|| nop			; indicate this is __udivsi3

.Lmerge:
	; r2 = al
	; r3 = b
	; r4 = ah
	; r5 = loop counter
	; f0 = tmpf
	; f1 = 1 if this is mod, 0 if this is div
	or	 r4,r0,0	|| sub	 r5,r0,-32	; ah = 0, loop = 32

.Lloop:
	src	 r4,r2,-1	|| sub	 r5,r5,1	; ah = (ah << 1) | (al >> 31); decrement loop count
	cmpuge	 f0,r4,r3	|| sra	 r2,r2,-1	; f0 = (ah >= b); al <<= 1
	sub/tx	 r4,r4,r3	|| or/tx r2,r2,1	; ah -= (tmpf) ? b : 0; al |= tmpf
	bratnz.s r5,.Lloop	|| nop			; loop back if not done
	jmp	 link           || or/xt r2,r0,r4	; if mod, update register, then return to user
.Ludiv:
	.size	 __udivsi3,.Ludiv-__udivsi3
	.stabs	"",36,0,0,.Ludiv-__udivsi3

#endif /* L_udivsi3 */


#ifdef L_divsi3

/* For division, we use the following algorithm:
 *
 *	unsigned
 *	__divsi3 (unsigned a, unsigned b)
 *	{
 *	  unsigned al = __builtin_abs (a);
 *	  unsigned b2 = __builtin_abs (b);
 *	  unsigned ah = 0;
 *	  unsigned tmpf;
 *	  int i;
 *
 *	  for (i = 32; i > 0; i--)
 *	    {
 *	      ah = (ah << 1) | (al >> 31);
 *	      tmpf = (ah >= b2) ? 1 : 0;
 *	      ah -= ((tmpf) ? b2 : 0);
 *	      al = (al << 1) | tmpf;
 *	    }
 *
 *	  if (a < 0)
 *	    ah = -ah, al = -al;
 *
 *	  if (b < 0)
 *	    al = -al;
 *
 *	  return al;	// for __divsi3
 *	  return ah;	// for __modsi3
 *	}
 */

	.file	"_divsi3"
	.text
	.globl	__modsi3
	.globl	__divsi3
	.type	__modsi3,@function
	.type	__divsi3,@function
	.stabs	"libgcc1.asm",100,0,0,__modsi3
	.stabs	"int:t(0,1)=r(0,1);-2147483648;2147483647;",128,0,0,0
	.stabs	"__modsi3:F(0,1)",36,0,1,__modsi3
	.stabs	"a:P(0,1)",64,0,1,2
	.stabs	"b:P(0,1)",64,0,1,3

__modsi3:
	bra.s .Lmerge		|| orfg	f1,f1,1		; indicate this is __modsi3
.Lmod:
	.size	__modsi3,.Lmod-__modsi3
	.stabs	"",36,0,0,.Lmod-__modsi3

	.stabs	"__divsi3:F(0,1)",36,0,1,__divsi3
	.stabs	"a:P(0,1)",64,0,1,2
	.stabs	"b:P(0,1)",64,0,1,3
__divsi3:
	andfg	f1,f1,0		|| nop			; indicate this is __divsi3

.Lmerge:
	; r2 = al
	; r3 = b2
	; r4 = ah
	; r5 = loop counter
	; r6 = a
	; r7 = b
	; f0 = tmpf
	; f1 = 1 if this is mod, 0 if this is div
	or	 r6,r0,r2	|| or	  r7,r0,r3	; copy original inputs
	abs	 r2,r2		|| abs	  r3,r3		; make both postive
	or	 r4,r0,0	|| sub	 r5,r0,-32	; ah = 0, loop = 32

.Lloop:
	src	 r4,r2,-1	|| sub	  r5,r5,1	; ah = (ah << 1) | (al >> 31); decrement loop count
	cmpuge	 f0,r4,r3	|| sra	  r2,r2,-1	; f0 = (ah >= b); al <<= 1
	sub/tx	 r4,r4,r3	|| or/tx  r2,r2,1	; ah -= (tmpf) ? b : 0; al |= tmpf
	bratnz.s r5,.Lloop	|| nop			; loop back if not done
	cmplt    f0,r6,0	|| nop			; f0 = (a < 0)

	sub/tx	 r2,r0,r2	|| sub/tx r4,r0,r4	; negate both al, ah if (a < 0)
	cmplt	 f0,r7,0	-> sub/tx r2,r0,r2	; negate al if (b < 0)
	jmp	 link		|| or/xt  r2,r0,r4	; update result if mod; return to user
.Ldiv:
	.size	 __divsi3,.Ldiv-__divsi3
	.stabs	"",36,0,0,.Ldiv-__divsi3

#endif /* L_divsi3 */