mirror of
				https://github.com/python/cpython.git
				synced 2025-10-31 05:31:20 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1419 lines
		
	
	
	
		
			29 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1419 lines
		
	
	
	
		
			29 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***********************************************************
 | |
| Copyright 1991-1995 by Stichting Mathematisch Centrum, Amsterdam,
 | |
| The Netherlands.
 | |
| 
 | |
|                         All Rights Reserved
 | |
| 
 | |
| Permission to use, copy, modify, and distribute this software and its 
 | |
| documentation for any purpose and without fee is hereby granted, 
 | |
| provided that the above copyright notice appear in all copies and that
 | |
| both that copyright notice and this permission notice appear in 
 | |
| supporting documentation, and that the names of Stichting Mathematisch
 | |
| Centrum or CWI not be used in advertising or publicity pertaining to
 | |
| distribution of the software without specific, written prior permission.
 | |
| 
 | |
| STICHTING MATHEMATISCH CENTRUM DISCLAIMS ALL WARRANTIES WITH REGARD TO
 | |
| THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
 | |
| FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH CENTRUM BE LIABLE
 | |
| FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 | |
| WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 | |
| ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
 | |
| OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 | |
| 
 | |
| ******************************************************************/
 | |
| 
 | |
| /* Long (arbitrary precision) integer object implementation */
 | |
| 
 | |
| /* XXX The functional organization of this file is terrible */
 | |
| 
 | |
| #include "allobjects.h"
 | |
| #include "longintrepr.h"
 | |
| #include <math.h>
 | |
| #include <assert.h>
 | |
| #include <ctype.h>
 | |
| 
 | |
| #define ABS(x) ((x) < 0 ? -(x) : (x))
 | |
| 
 | |
| /* Forward */
 | |
| static longobject *long_normalize PROTO((longobject *));
 | |
| static longobject *mul1 PROTO((longobject *, wdigit));
 | |
| static longobject *muladd1 PROTO((longobject *, wdigit, wdigit));
 | |
| static longobject *divrem1 PROTO((longobject *, wdigit, digit *));
 | |
| static object *long_format PROTO((object *aa, int base));
 | |
| 
 | |
| static int ticker;	/* XXX Could be shared with ceval? */
 | |
| 
 | |
| #define SIGCHECK(block) \
 | |
| 	if (--ticker < 0) { \
 | |
| 		ticker = 100; \
 | |
| 		if (sigcheck()) { block; } \
 | |
| 	}
 | |
| 
 | |
| /* Normalize (remove leading zeros from) a long int object.
 | |
|    Doesn't attempt to free the storage--in most cases, due to the nature
 | |
|    of the algorithms used, this could save at most be one word anyway. */
 | |
| 
 | |
| static longobject *
 | |
| long_normalize(v)
 | |
| 	register longobject *v;
 | |
| {
 | |
| 	int j = ABS(v->ob_size);
 | |
| 	register int i = j;
 | |
| 	
 | |
| 	while (i > 0 && v->ob_digit[i-1] == 0)
 | |
| 		--i;
 | |
| 	if (i != j)
 | |
| 		v->ob_size = (v->ob_size < 0) ? -(i) : i;
 | |
| 	return v;
 | |
| }
 | |
| 
 | |
| /* Allocate a new long int object with size digits.
 | |
|    Return NULL and set exception if we run out of memory. */
 | |
| 
 | |
| longobject *
 | |
| alloclongobject(size)
 | |
| 	int size;
 | |
| {
 | |
| 	return NEWVAROBJ(longobject, &Longtype, size);
 | |
| }
 | |
| 
 | |
| /* Create a new long int object from a C long int */
 | |
| 
 | |
| object *
 | |
| newlongobject(ival)
 | |
| 	long ival;
 | |
| {
 | |
| 	/* Assume a C long fits in at most 3 'digits' */
 | |
| 	/* XXX On 64 bit machines this isn't true!!! */
 | |
| 	longobject *v = alloclongobject(3);
 | |
| 	if (v != NULL) {
 | |
| 		if (ival < 0) {
 | |
| 			ival = -ival;
 | |
| 			v->ob_size = -(v->ob_size);
 | |
| 		}
 | |
| 		v->ob_digit[0] = ival & MASK;
 | |
| 		v->ob_digit[1] = (ival >> SHIFT) & MASK;
 | |
| 		v->ob_digit[2] = ((unsigned long)ival >> (2*SHIFT)) & MASK;
 | |
| 		v = long_normalize(v);
 | |
| 	}
 | |
| 	return (object *)v;
 | |
| }
 | |
| 
 | |
| /* Create a new long int object from a C double */
 | |
| 
 | |
| object *
 | |
| dnewlongobject(dval)
 | |
| 	double dval;
 | |
| {
 | |
| 	longobject *v;
 | |
| 	double frac;
 | |
| 	int i, ndig, expo, neg;
 | |
| 	neg = 0;
 | |
| 	if (dval < 0.0) {
 | |
| 		neg = 1;
 | |
| 		dval = -dval;
 | |
| 	}
 | |
| 	frac = frexp(dval, &expo); /* dval = frac*2**expo; 0.0 <= frac < 1.0 */
 | |
| 	if (expo <= 0)
 | |
| 		return newlongobject(0L);
 | |
| 	ndig = (expo-1) / SHIFT + 1; /* Number of 'digits' in result */
 | |
| 	v = alloclongobject(ndig);
 | |
| 	if (v == NULL)
 | |
| 		return NULL;
 | |
| 	frac = ldexp(frac, (expo-1) % SHIFT + 1);
 | |
| 	for (i = ndig; --i >= 0; ) {
 | |
| 		long bits = (long)frac;
 | |
| 		v->ob_digit[i] = bits;
 | |
| 		frac = frac - (double)bits;
 | |
| 		frac = ldexp(frac, SHIFT);
 | |
| 	}
 | |
| 	if (neg)
 | |
| 		v->ob_size = -(v->ob_size);
 | |
| 	return (object *)v;
 | |
| }
 | |
| 
 | |
| /* Get a C long int from a long int object.
 | |
|    Returns -1 and sets an error condition if overflow occurs. */
 | |
| 
 | |
| long
 | |
| getlongvalue(vv)
 | |
| 	object *vv;
 | |
| {
 | |
| 	register longobject *v;
 | |
| 	long x, prev;
 | |
| 	int i, sign;
 | |
| 	
 | |
| 	if (vv == NULL || !is_longobject(vv)) {
 | |
| 		err_badcall();
 | |
| 		return -1;
 | |
| 	}
 | |
| 	v = (longobject *)vv;
 | |
| 	i = v->ob_size;
 | |
| 	sign = 1;
 | |
| 	x = 0;
 | |
| 	if (i < 0) {
 | |
| 		sign = -1;
 | |
| 		i = -(i);
 | |
| 	}
 | |
| 	while (--i >= 0) {
 | |
| 		prev = x;
 | |
| 		x = (x << SHIFT) + v->ob_digit[i];
 | |
| 		if ((x >> SHIFT) != prev) {
 | |
| 			err_setstr(OverflowError,
 | |
| 				"long int too long to convert");
 | |
| 			return -1;
 | |
| 		}
 | |
| 	}
 | |
| 	return x * sign;
 | |
| }
 | |
| 
 | |
| /* Get a C double from a long int object.  No overflow check. */
 | |
| 
 | |
| double
 | |
| dgetlongvalue(vv)
 | |
| 	object *vv;
 | |
| {
 | |
| 	register longobject *v;
 | |
| 	double x;
 | |
| 	double multiplier = (double) (1L << SHIFT);
 | |
| 	int i, sign;
 | |
| 	
 | |
| 	if (vv == NULL || !is_longobject(vv)) {
 | |
| 		err_badcall();
 | |
| 		return -1;
 | |
| 	}
 | |
| 	v = (longobject *)vv;
 | |
| 	i = v->ob_size;
 | |
| 	sign = 1;
 | |
| 	x = 0.0;
 | |
| 	if (i < 0) {
 | |
| 		sign = -1;
 | |
| 		i = -(i);
 | |
| 	}
 | |
| 	while (--i >= 0) {
 | |
| 		x = x*multiplier + (double)v->ob_digit[i];
 | |
| 	}
 | |
| 	return x * sign;
 | |
| }
 | |
| 
 | |
| /* Multiply by a single digit, ignoring the sign. */
 | |
| 
 | |
| static longobject *
 | |
| mul1(a, n)
 | |
| 	longobject *a;
 | |
| 	wdigit n;
 | |
| {
 | |
| 	return muladd1(a, n, (digit)0);
 | |
| }
 | |
| 
 | |
| /* Multiply by a single digit and add a single digit, ignoring the sign. */
 | |
| 
 | |
| static longobject *
 | |
| muladd1(a, n, extra)
 | |
| 	longobject *a;
 | |
| 	wdigit n;
 | |
| 	wdigit extra;
 | |
| {
 | |
| 	int size_a = ABS(a->ob_size);
 | |
| 	longobject *z = alloclongobject(size_a+1);
 | |
| 	twodigits carry = extra;
 | |
| 	int i;
 | |
| 	
 | |
| 	if (z == NULL)
 | |
| 		return NULL;
 | |
| 	for (i = 0; i < size_a; ++i) {
 | |
| 		carry += (twodigits)a->ob_digit[i] * n;
 | |
| 		z->ob_digit[i] = carry & MASK;
 | |
| 		carry >>= SHIFT;
 | |
| 	}
 | |
| 	z->ob_digit[i] = carry;
 | |
| 	return long_normalize(z);
 | |
| }
 | |
| 
 | |
| /* Divide a long integer by a digit, returning both the quotient
 | |
|    (as function result) and the remainder (through *prem).
 | |
|    The sign of a is ignored; n should not be zero. */
 | |
| 
 | |
| static longobject *
 | |
| divrem1(a, n, prem)
 | |
| 	longobject *a;
 | |
| 	wdigit n;
 | |
| 	digit *prem;
 | |
| {
 | |
| 	int size = ABS(a->ob_size);
 | |
| 	longobject *z;
 | |
| 	int i;
 | |
| 	twodigits rem = 0;
 | |
| 	
 | |
| 	assert(n > 0 && n <= MASK);
 | |
| 	z = alloclongobject(size);
 | |
| 	if (z == NULL)
 | |
| 		return NULL;
 | |
| 	for (i = size; --i >= 0; ) {
 | |
| 		rem = (rem << SHIFT) + a->ob_digit[i];
 | |
| 		z->ob_digit[i] = rem/n;
 | |
| 		rem %= n;
 | |
| 	}
 | |
| 	*prem = rem;
 | |
| 	return long_normalize(z);
 | |
| }
 | |
| 
 | |
| /* Convert a long int object to a string, using a given conversion base.
 | |
|    Return a string object.
 | |
|    If base is 8 or 16, add the proper prefix '0' or '0x'.
 | |
|    External linkage: used in bltinmodule.c by hex() and oct(). */
 | |
| 
 | |
| static object *
 | |
| long_format(aa, base)
 | |
| 	object *aa;
 | |
| 	int base;
 | |
| {
 | |
| 	register longobject *a = (longobject *)aa;
 | |
| 	stringobject *str;
 | |
| 	int i;
 | |
| 	int size_a = ABS(a->ob_size);
 | |
| 	char *p;
 | |
| 	int bits;
 | |
| 	char sign = '\0';
 | |
| 
 | |
| 	if (a == NULL || !is_longobject(a)) {
 | |
| 		err_badcall();
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	assert(base >= 2 && base <= 36);
 | |
| 	
 | |
| 	/* Compute a rough upper bound for the length of the string */
 | |
| 	i = base;
 | |
| 	bits = 0;
 | |
| 	while (i > 1) {
 | |
| 		++bits;
 | |
| 		i >>= 1;
 | |
| 	}
 | |
| 	i = 6 + (size_a*SHIFT + bits-1) / bits;
 | |
| 	str = (stringobject *) newsizedstringobject((char *)0, i);
 | |
| 	if (str == NULL)
 | |
| 		return NULL;
 | |
| 	p = GETSTRINGVALUE(str) + i;
 | |
| 	*p = '\0';
 | |
| 	*--p = 'L';
 | |
| 	if (a->ob_size < 0)
 | |
| 		sign = '-';
 | |
| 	
 | |
| 	INCREF(a);
 | |
| 	do {
 | |
| 		digit rem;
 | |
| 		longobject *temp = divrem1(a, (digit)base, &rem);
 | |
| 		if (temp == NULL) {
 | |
| 			DECREF(a);
 | |
| 			DECREF(str);
 | |
| 			return NULL;
 | |
| 		}
 | |
| 		if (rem < 10)
 | |
| 			rem += '0';
 | |
| 		else
 | |
| 			rem += 'A'-10;
 | |
| 		assert(p > GETSTRINGVALUE(str));
 | |
| 		*--p = rem;
 | |
| 		DECREF(a);
 | |
| 		a = temp;
 | |
| 		SIGCHECK({
 | |
| 			DECREF(a);
 | |
| 			DECREF(str);
 | |
| 			return NULL;
 | |
| 		})
 | |
| 	} while (ABS(a->ob_size) != 0);
 | |
| 	DECREF(a);
 | |
| 	if (base == 8) {
 | |
| 		if (size_a != 0)
 | |
| 			*--p = '0';
 | |
| 	}
 | |
| 	else if (base == 16) {
 | |
| 		*--p = 'x';
 | |
| 		*--p = '0';
 | |
| 	}
 | |
| 	else if (base != 10) {
 | |
| 		*--p = '#';
 | |
| 		*--p = '0' + base%10;
 | |
| 		if (base > 10)
 | |
| 			*--p = '0' + base/10;
 | |
| 	}
 | |
| 	if (sign)
 | |
| 		*--p = sign;
 | |
| 	if (p != GETSTRINGVALUE(str)) {
 | |
| 		char *q = GETSTRINGVALUE(str);
 | |
| 		assert(p > q);
 | |
| 		do {
 | |
| 		} while ((*q++ = *p++) != '\0');
 | |
| 		q--;
 | |
| 		resizestring((object **)&str, (int) (q - GETSTRINGVALUE(str)));
 | |
| 	}
 | |
| 	return (object *)str;
 | |
| }
 | |
| 
 | |
| #if 0
 | |
| /* Convert a string to a long int object, in a given base.
 | |
|    Base zero implies a default depending on the number.
 | |
|    External linkage: used in compile.c and stropmodule.c. */
 | |
| 
 | |
| object *
 | |
| long_scan(str, base)
 | |
| 	char *str;
 | |
| 	int base;
 | |
| {
 | |
| 	return long_escan(str, (char **)NULL, base);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| object *
 | |
| long_escan(str, pend, base)
 | |
| 	char *str;
 | |
| 	char **pend;
 | |
| 	int base;
 | |
| {
 | |
| 	int sign = 1;
 | |
| 	longobject *z;
 | |
| 	
 | |
| 	if (base != 0 && base < 2 || base > 36) {
 | |
| 		err_setstr(ValueError, "invalid base for long literal");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	while (*str != '\0' && isspace(Py_CHARMASK(*str)))
 | |
| 		str++;
 | |
| 	if (*str == '+')
 | |
| 		++str;
 | |
| 	else if (*str == '-') {
 | |
| 		++str;
 | |
| 		sign = -1;
 | |
| 	}
 | |
| 	while (*str != '\0' && isspace(Py_CHARMASK(*str)))
 | |
| 		str++;
 | |
| 	if (base == 0) {
 | |
| 		if (str[0] != '0')
 | |
| 			base = 10;
 | |
| 		else if (str[1] == 'x' || str[1] == 'X')
 | |
| 			base = 16;
 | |
| 		else
 | |
| 			base = 8;
 | |
| 	}
 | |
| 	if (base == 16 && str[0] == '0' && (str[1] == 'x' || str[1] == 'X'))
 | |
| 		str += 2;
 | |
| 	z = alloclongobject(0);
 | |
| 	for ( ; z != NULL; ++str) {
 | |
| 		int k = -1;
 | |
| 		longobject *temp;
 | |
| 		
 | |
| 		if (*str <= '9')
 | |
| 			k = *str - '0';
 | |
| 		else if (*str >= 'a')
 | |
| 			k = *str - 'a' + 10;
 | |
| 		else if (*str >= 'A')
 | |
| 			k = *str - 'A' + 10;
 | |
| 		if (k < 0 || k >= base)
 | |
| 			break;
 | |
| 		temp = muladd1(z, (digit)base, (digit)k);
 | |
| 		DECREF(z);
 | |
| 		z = temp;
 | |
| 	}
 | |
| 	if (sign < 0 && z != NULL && z->ob_size != 0)
 | |
| 		z->ob_size = -(z->ob_size);
 | |
| 	if (pend)
 | |
| 		*pend = str;
 | |
| 	return (object *) z;
 | |
| }
 | |
| 
 | |
| static longobject *x_divrem PROTO((longobject *, longobject *, longobject **));
 | |
| static object *long_pos PROTO((longobject *));
 | |
| static long_divrem PROTO((longobject *, longobject *,
 | |
| 	longobject **, longobject **));
 | |
| 
 | |
| /* Long division with remainder, top-level routine */
 | |
| 
 | |
| static int
 | |
| long_divrem(a, b, pdiv, prem)
 | |
| 	longobject *a, *b;
 | |
| 	longobject **pdiv;
 | |
| 	longobject **prem;
 | |
| {
 | |
| 	int size_a = ABS(a->ob_size), size_b = ABS(b->ob_size);
 | |
| 	longobject *z;
 | |
| 	
 | |
| 	if (size_b == 0) {
 | |
| 		err_setstr(ZeroDivisionError, "long division or modulo");
 | |
| 		return -1;
 | |
| 	}
 | |
| 	if (size_a < size_b ||
 | |
| 			size_a == size_b &&
 | |
| 			a->ob_digit[size_a-1] < b->ob_digit[size_b-1]) {
 | |
| 		/* |a| < |b|. */
 | |
| 		*pdiv = alloclongobject(0);
 | |
| 		INCREF(a);
 | |
| 		*prem = (longobject *) a;
 | |
| 		return 0;
 | |
| 	}
 | |
| 	if (size_b == 1) {
 | |
| 		digit rem = 0;
 | |
| 		z = divrem1(a, b->ob_digit[0], &rem);
 | |
| 		if (z == NULL)
 | |
| 			return -1;
 | |
| 		*prem = (longobject *) newlongobject((long)rem);
 | |
| 	}
 | |
| 	else {
 | |
| 		z = x_divrem(a, b, prem);
 | |
| 		if (z == NULL)
 | |
| 			return -1;
 | |
| 	}
 | |
| 	/* Set the signs.
 | |
| 	   The quotient z has the sign of a*b;
 | |
| 	   the remainder r has the sign of a,
 | |
| 	   so a = b*z + r. */
 | |
| 	if ((a->ob_size < 0) != (b->ob_size < 0))
 | |
| 		z->ob_size = -(z->ob_size);
 | |
| 	if (a->ob_size < 0 && (*prem)->ob_size != 0)
 | |
| 		(*prem)->ob_size = -((*prem)->ob_size);
 | |
| 	*pdiv = z;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /* Unsigned long division with remainder -- the algorithm */
 | |
| 
 | |
| static longobject *
 | |
| x_divrem(v1, w1, prem)
 | |
| 	longobject *v1, *w1;
 | |
| 	longobject **prem;
 | |
| {
 | |
| 	int size_v = ABS(v1->ob_size), size_w = ABS(w1->ob_size);
 | |
| 	digit d = (twodigits)BASE / (w1->ob_digit[size_w-1] + 1);
 | |
| 	longobject *v = mul1(v1, d);
 | |
| 	longobject *w = mul1(w1, d);
 | |
| 	longobject *a;
 | |
| 	int j, k;
 | |
| 	
 | |
| 	if (v == NULL || w == NULL) {
 | |
| 		XDECREF(v);
 | |
| 		XDECREF(w);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	
 | |
| 	assert(size_v >= size_w && size_w > 1); /* Assert checks by div() */
 | |
| 	assert(v->ob_refcnt == 1); /* Since v will be used as accumulator! */
 | |
| 	assert(size_w == ABS(w->ob_size)); /* That's how d was calculated */
 | |
| 	
 | |
| 	size_v = ABS(v->ob_size);
 | |
| 	a = alloclongobject(size_v - size_w + 1);
 | |
| 	
 | |
| 	for (j = size_v, k = a->ob_size-1; a != NULL && k >= 0; --j, --k) {
 | |
| 		digit vj = (j >= size_v) ? 0 : v->ob_digit[j];
 | |
| 		twodigits q;
 | |
| 		stwodigits carry = 0;
 | |
| 		int i;
 | |
| 		
 | |
| 		SIGCHECK({
 | |
| 			DECREF(a);
 | |
| 			a = NULL;
 | |
| 			break;
 | |
| 		})
 | |
| 		if (vj == w->ob_digit[size_w-1])
 | |
| 			q = MASK;
 | |
| 		else
 | |
| 			q = (((twodigits)vj << SHIFT) + v->ob_digit[j-1]) /
 | |
| 				w->ob_digit[size_w-1];
 | |
| 		
 | |
| 		while (w->ob_digit[size_w-2]*q >
 | |
| 				((
 | |
| 					((twodigits)vj << SHIFT)
 | |
| 					+ v->ob_digit[j-1]
 | |
| 					- q*w->ob_digit[size_w-1]
 | |
| 								) << SHIFT)
 | |
| 				+ v->ob_digit[j-2])
 | |
| 			--q;
 | |
| 		
 | |
| 		for (i = 0; i < size_w && i+k < size_v; ++i) {
 | |
| 			twodigits z = w->ob_digit[i] * q;
 | |
| 			digit zz = z >> SHIFT;
 | |
| 			carry += v->ob_digit[i+k] - z + ((twodigits)zz << SHIFT);
 | |
| 			v->ob_digit[i+k] = carry & MASK;
 | |
| 			carry = (carry >> SHIFT) - zz;
 | |
| 		}
 | |
| 		
 | |
| 		if (i+k < size_v) {
 | |
| 			carry += v->ob_digit[i+k];
 | |
| 			v->ob_digit[i+k] = 0;
 | |
| 		}
 | |
| 		
 | |
| 		if (carry == 0)
 | |
| 			a->ob_digit[k] = q;
 | |
| 		else {
 | |
| 			assert(carry == -1);
 | |
| 			a->ob_digit[k] = q-1;
 | |
| 			carry = 0;
 | |
| 			for (i = 0; i < size_w && i+k < size_v; ++i) {
 | |
| 				carry += v->ob_digit[i+k] + w->ob_digit[i];
 | |
| 				v->ob_digit[i+k] = carry & MASK;
 | |
| 				carry >>= SHIFT;
 | |
| 			}
 | |
| 		}
 | |
| 	} /* for j, k */
 | |
| 	
 | |
| 	if (a == NULL)
 | |
| 		*prem = NULL;
 | |
| 	else {
 | |
| 		a = long_normalize(a);
 | |
| 		*prem = divrem1(v, d, &d);
 | |
| 		/* d receives the (unused) remainder */
 | |
| 		if (*prem == NULL) {
 | |
| 			DECREF(a);
 | |
| 			a = NULL;
 | |
| 		}
 | |
| 	}
 | |
| 	DECREF(v);
 | |
| 	DECREF(w);
 | |
| 	return a;
 | |
| }
 | |
| 
 | |
| /* Methods */
 | |
| 
 | |
| /* Forward */
 | |
| static void long_dealloc PROTO((object *));
 | |
| static object *long_repr PROTO((object *));
 | |
| static int long_compare PROTO((longobject *, longobject *));
 | |
| static long long_hash PROTO((longobject *));
 | |
| 
 | |
| static object *long_add PROTO((longobject *, longobject *));
 | |
| static object *long_sub PROTO((longobject *, longobject *));
 | |
| static object *long_mul PROTO((longobject *, longobject *));
 | |
| static object *long_div PROTO((longobject *, longobject *));
 | |
| static object *long_mod PROTO((longobject *, longobject *));
 | |
| static object *long_divmod PROTO((longobject *, longobject *));
 | |
| static object *long_pow PROTO((longobject *, longobject *, longobject *));
 | |
| static object *long_neg PROTO((longobject *));
 | |
| static object *long_pos PROTO((longobject *));
 | |
| static object *long_abs PROTO((longobject *));
 | |
| static int long_nonzero PROTO((longobject *));
 | |
| static object *long_invert PROTO((longobject *));
 | |
| static object *long_lshift PROTO((longobject *, longobject *));
 | |
| static object *long_rshift PROTO((longobject *, longobject *));
 | |
| static object *long_and PROTO((longobject *, longobject *));
 | |
| static object *long_xor PROTO((longobject *, longobject *));
 | |
| static object *long_or PROTO((longobject *, longobject *));
 | |
| 
 | |
| static void
 | |
| long_dealloc(v)
 | |
| 	object *v;
 | |
| {
 | |
| 	DEL(v);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_repr(v)
 | |
| 	object *v;
 | |
| {
 | |
| 	return long_format(v, 10);
 | |
| }
 | |
| 
 | |
| static int
 | |
| long_compare(a, b)
 | |
| 	longobject *a, *b;
 | |
| {
 | |
| 	int sign;
 | |
| 	
 | |
| 	if (a->ob_size != b->ob_size) {
 | |
| 		if (ABS(a->ob_size) == 0 && ABS(b->ob_size) == 0)
 | |
| 			sign = 0;
 | |
| 		else
 | |
| 			sign = a->ob_size - b->ob_size;
 | |
| 	}
 | |
| 	else {
 | |
| 		int i = ABS(a->ob_size);
 | |
| 		while (--i >= 0 && a->ob_digit[i] == b->ob_digit[i])
 | |
| 			;
 | |
| 		if (i < 0)
 | |
| 			sign = 0;
 | |
| 		else {
 | |
| 			sign = (int)a->ob_digit[i] - (int)b->ob_digit[i];
 | |
| 			if (a->ob_size < 0)
 | |
| 				sign = -sign;
 | |
| 		}
 | |
| 	}
 | |
| 	return sign < 0 ? -1 : sign > 0 ? 1 : 0;
 | |
| }
 | |
| 
 | |
| static long
 | |
| long_hash(v)
 | |
| 	longobject *v;
 | |
| {
 | |
| 	long x;
 | |
| 	int i, sign;
 | |
| 
 | |
| 	/* This is designed so that Python ints and longs with the
 | |
| 	   same value hash to the same value, otherwise comparisons
 | |
| 	   of mapping keys will turn out weird */
 | |
| 	i = v->ob_size;
 | |
| 	sign = 1;
 | |
| 	x = 0;
 | |
| 	if (i < 0) {
 | |
| 		sign = -1;
 | |
| 		i = -(i);
 | |
| 	}
 | |
| 	while (--i >= 0) {
 | |
| 		/* Force a 32-bit circular shift */
 | |
| 		x = ((x << SHIFT) & ~MASK) | ((x >> (32-SHIFT)) & MASK);
 | |
| 		x += v->ob_digit[i];
 | |
| 	}
 | |
| 	x = x * sign;
 | |
| 	if (x == -1)
 | |
| 		x = -2;
 | |
| 	return x;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* Add the absolute values of two long integers. */
 | |
| 
 | |
| static longobject *x_add PROTO((longobject *, longobject *));
 | |
| static longobject *
 | |
| x_add(a, b)
 | |
| 	longobject *a, *b;
 | |
| {
 | |
| 	int size_a = ABS(a->ob_size), size_b = ABS(b->ob_size);
 | |
| 	longobject *z;
 | |
| 	int i;
 | |
| 	digit carry = 0;
 | |
| 	
 | |
| 	/* Ensure a is the larger of the two: */
 | |
| 	if (size_a < size_b) {
 | |
| 		{ longobject *temp = a; a = b; b = temp; }
 | |
| 		{ int size_temp = size_a; size_a = size_b; size_b = size_temp; }
 | |
| 	}
 | |
| 	z = alloclongobject(size_a+1);
 | |
| 	if (z == NULL)
 | |
| 		return NULL;
 | |
| 	for (i = 0; i < size_b; ++i) {
 | |
| 		carry += a->ob_digit[i] + b->ob_digit[i];
 | |
| 		z->ob_digit[i] = carry & MASK;
 | |
| 		/* The following assumes unsigned shifts don't
 | |
| 		   propagate the sign bit. */
 | |
| 		carry >>= SHIFT;
 | |
| 	}
 | |
| 	for (; i < size_a; ++i) {
 | |
| 		carry += a->ob_digit[i];
 | |
| 		z->ob_digit[i] = carry & MASK;
 | |
| 		carry >>= SHIFT;
 | |
| 	}
 | |
| 	z->ob_digit[i] = carry;
 | |
| 	return long_normalize(z);
 | |
| }
 | |
| 
 | |
| /* Subtract the absolute values of two integers. */
 | |
| 
 | |
| static longobject *x_sub PROTO((longobject *, longobject *));
 | |
| static longobject *
 | |
| x_sub(a, b)
 | |
| 	longobject *a, *b;
 | |
| {
 | |
| 	int size_a = ABS(a->ob_size), size_b = ABS(b->ob_size);
 | |
| 	longobject *z;
 | |
| 	int i;
 | |
| 	int sign = 1;
 | |
| 	digit borrow = 0;
 | |
| 	
 | |
| 	/* Ensure a is the larger of the two: */
 | |
| 	if (size_a < size_b) {
 | |
| 		sign = -1;
 | |
| 		{ longobject *temp = a; a = b; b = temp; }
 | |
| 		{ int size_temp = size_a; size_a = size_b; size_b = size_temp; }
 | |
| 	}
 | |
| 	else if (size_a == size_b) {
 | |
| 		/* Find highest digit where a and b differ: */
 | |
| 		i = size_a;
 | |
| 		while (--i >= 0 && a->ob_digit[i] == b->ob_digit[i])
 | |
| 			;
 | |
| 		if (i < 0)
 | |
| 			return alloclongobject(0);
 | |
| 		if (a->ob_digit[i] < b->ob_digit[i]) {
 | |
| 			sign = -1;
 | |
| 			{ longobject *temp = a; a = b; b = temp; }
 | |
| 		}
 | |
| 		size_a = size_b = i+1;
 | |
| 	}
 | |
| 	z = alloclongobject(size_a);
 | |
| 	if (z == NULL)
 | |
| 		return NULL;
 | |
| 	for (i = 0; i < size_b; ++i) {
 | |
| 		/* The following assumes unsigned arithmetic
 | |
| 		   works module 2**N for some N>SHIFT. */
 | |
| 		borrow = a->ob_digit[i] - b->ob_digit[i] - borrow;
 | |
| 		z->ob_digit[i] = borrow & MASK;
 | |
| 		borrow >>= SHIFT;
 | |
| 		borrow &= 1; /* Keep only one sign bit */
 | |
| 	}
 | |
| 	for (; i < size_a; ++i) {
 | |
| 		borrow = a->ob_digit[i] - borrow;
 | |
| 		z->ob_digit[i] = borrow & MASK;
 | |
| 		borrow >>= SHIFT;
 | |
| 	}
 | |
| 	assert(borrow == 0);
 | |
| 	if (sign < 0)
 | |
| 		z->ob_size = -(z->ob_size);
 | |
| 	return long_normalize(z);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_add(a, b)
 | |
| 	longobject *a;
 | |
| 	longobject *b;
 | |
| {
 | |
| 	longobject *z;
 | |
| 	
 | |
| 	if (a->ob_size < 0) {
 | |
| 		if (b->ob_size < 0) {
 | |
| 			z = x_add(a, b);
 | |
| 			if (z != NULL && z->ob_size != 0)
 | |
| 				z->ob_size = -(z->ob_size);
 | |
| 		}
 | |
| 		else
 | |
| 			z = x_sub(b, a);
 | |
| 	}
 | |
| 	else {
 | |
| 		if (b->ob_size < 0)
 | |
| 			z = x_sub(a, b);
 | |
| 		else
 | |
| 			z = x_add(a, b);
 | |
| 	}
 | |
| 	return (object *)z;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_sub(a, b)
 | |
| 	longobject *a;
 | |
| 	longobject *b;
 | |
| {
 | |
| 	longobject *z;
 | |
| 	
 | |
| 	if (a->ob_size < 0) {
 | |
| 		if (b->ob_size < 0)
 | |
| 			z = x_sub(a, b);
 | |
| 		else
 | |
| 			z = x_add(a, b);
 | |
| 		if (z != NULL && z->ob_size != 0)
 | |
| 			z->ob_size = -(z->ob_size);
 | |
| 	}
 | |
| 	else {
 | |
| 		if (b->ob_size < 0)
 | |
| 			z = x_add(a, b);
 | |
| 		else
 | |
| 			z = x_sub(a, b);
 | |
| 	}
 | |
| 	return (object *)z;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_mul(a, b)
 | |
| 	longobject *a;
 | |
| 	longobject *b;
 | |
| {
 | |
| 	int size_a;
 | |
| 	int size_b;
 | |
| 	longobject *z;
 | |
| 	int i;
 | |
| 	
 | |
| 	size_a = ABS(a->ob_size);
 | |
| 	size_b = ABS(b->ob_size);
 | |
| 	z = alloclongobject(size_a + size_b);
 | |
| 	if (z == NULL)
 | |
| 		return NULL;
 | |
| 	for (i = 0; i < z->ob_size; ++i)
 | |
| 		z->ob_digit[i] = 0;
 | |
| 	for (i = 0; i < size_a; ++i) {
 | |
| 		twodigits carry = 0;
 | |
| 		twodigits f = a->ob_digit[i];
 | |
| 		int j;
 | |
| 		
 | |
| 		SIGCHECK({
 | |
| 			DECREF(z);
 | |
| 			return NULL;
 | |
| 		})
 | |
| 		for (j = 0; j < size_b; ++j) {
 | |
| 			carry += z->ob_digit[i+j] + b->ob_digit[j] * f;
 | |
| 			z->ob_digit[i+j] = carry & MASK;
 | |
| 			carry >>= SHIFT;
 | |
| 		}
 | |
| 		for (; carry != 0; ++j) {
 | |
| 			assert(i+j < z->ob_size);
 | |
| 			carry += z->ob_digit[i+j];
 | |
| 			z->ob_digit[i+j] = carry & MASK;
 | |
| 			carry >>= SHIFT;
 | |
| 		}
 | |
| 	}
 | |
| 	if (a->ob_size < 0)
 | |
| 		z->ob_size = -(z->ob_size);
 | |
| 	if (b->ob_size < 0)
 | |
| 		z->ob_size = -(z->ob_size);
 | |
| 	return (object *) long_normalize(z);
 | |
| }
 | |
| 
 | |
| /* The / and % operators are now defined in terms of divmod().
 | |
|    The expression a mod b has the value a - b*floor(a/b).
 | |
|    The long_divrem function gives the remainder after division of
 | |
|    |a| by |b|, with the sign of a.  This is also expressed
 | |
|    as a - b*trunc(a/b), if trunc truncates towards zero.
 | |
|    Some examples:
 | |
|    	 a	 b	a rem b		a mod b
 | |
|    	 13	 10	 3		 3
 | |
|    	-13	 10	-3		 7
 | |
|    	 13	-10	 3		-7
 | |
|    	-13	-10	-3		-3
 | |
|    So, to get from rem to mod, we have to add b if a and b
 | |
|    have different signs.  We then subtract one from the 'div'
 | |
|    part of the outcome to keep the invariant intact. */
 | |
| 
 | |
| static int l_divmod PROTO((longobject *, longobject *,
 | |
| 	longobject **, longobject **));
 | |
| static int
 | |
| l_divmod(v, w, pdiv, pmod)
 | |
| 	longobject *v;
 | |
| 	longobject *w;
 | |
| 	longobject **pdiv;
 | |
| 	longobject **pmod;
 | |
| {
 | |
| 	longobject *div, *mod;
 | |
| 	
 | |
| 	if (long_divrem(v, w, &div, &mod) < 0)
 | |
| 		return -1;
 | |
| 	if (mod->ob_size < 0 && w->ob_size > 0 ||
 | |
| 				mod->ob_size > 0 && w->ob_size < 0) {
 | |
| 		longobject *temp;
 | |
| 		longobject *one;
 | |
| 		temp = (longobject *) long_add(mod, w);
 | |
| 		DECREF(mod);
 | |
| 		mod = temp;
 | |
| 		if (mod == NULL) {
 | |
| 			DECREF(div);
 | |
| 			return -1;
 | |
| 		}
 | |
| 		one = (longobject *) newlongobject(1L);
 | |
| 		if (one == NULL ||
 | |
| 		    (temp = (longobject *) long_sub(div, one)) == NULL) {
 | |
| 			DECREF(mod);
 | |
| 			DECREF(div);
 | |
| 			XDECREF(one);
 | |
| 			return -1;
 | |
| 		}
 | |
| 		DECREF(one);
 | |
| 		DECREF(div);
 | |
| 		div = temp;
 | |
| 	}
 | |
| 	*pdiv = div;
 | |
| 	*pmod = mod;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_div(v, w)
 | |
| 	longobject *v;
 | |
| 	longobject *w;
 | |
| {
 | |
| 	longobject *div, *mod;
 | |
| 	if (l_divmod(v, w, &div, &mod) < 0)
 | |
| 		return NULL;
 | |
| 	DECREF(mod);
 | |
| 	return (object *)div;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_mod(v, w)
 | |
| 	longobject *v;
 | |
| 	longobject *w;
 | |
| {
 | |
| 	longobject *div, *mod;
 | |
| 	if (l_divmod(v, w, &div, &mod) < 0)
 | |
| 		return NULL;
 | |
| 	DECREF(div);
 | |
| 	return (object *)mod;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_divmod(v, w)
 | |
| 	longobject *v;
 | |
| 	longobject *w;
 | |
| {
 | |
| 	object *z;
 | |
| 	longobject *div, *mod;
 | |
| 	if (l_divmod(v, w, &div, &mod) < 0)
 | |
| 		return NULL;
 | |
| 	z = newtupleobject(2);
 | |
| 	if (z != NULL) {
 | |
| 		settupleitem(z, 0, (object *) div);
 | |
| 		settupleitem(z, 1, (object *) mod);
 | |
| 	}
 | |
| 	else {
 | |
| 		DECREF(div);
 | |
| 		DECREF(mod);
 | |
| 	}
 | |
| 	return z;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_pow(a, b, c)
 | |
| 	longobject *a;
 | |
| 	longobject *b;
 | |
| 	longobject *c;
 | |
| {
 | |
| 	longobject *z, *div, *mod;
 | |
| 	int size_b, i;
 | |
| 	
 | |
| 	size_b = b->ob_size;
 | |
| 	if (size_b < 0) {
 | |
| 		err_setstr(ValueError, "long integer to the negative power");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	z = (longobject *)newlongobject(1L);
 | |
| 	INCREF(a);
 | |
| 	for (i = 0; i < size_b; ++i) {
 | |
| 		digit bi = b->ob_digit[i];
 | |
| 		int j;
 | |
| 	
 | |
| 		for (j = 0; j < SHIFT; ++j) {
 | |
| 			longobject *temp;
 | |
| 		
 | |
| 			if (bi & 1) {
 | |
| 				temp = (longobject *)long_mul(z, a);
 | |
| 				DECREF(z);
 | |
| 			 	if ((object*)c!=None && temp!=NULL) {
 | |
| 			 		l_divmod(temp, c, &div, &mod);
 | |
| 				 	XDECREF(div);
 | |
| 				 	DECREF(temp);
 | |
| 				 	temp = mod;
 | |
| 				}
 | |
| 			 	z = temp;
 | |
| 				if (z == NULL)
 | |
| 					break;
 | |
| 			}
 | |
| 			bi >>= 1;
 | |
| 			if (bi == 0 && i+1 == size_b)
 | |
| 				break;
 | |
| 			temp = (longobject *)long_mul(a, a);
 | |
| 			DECREF(a);
 | |
| 		 	if ((object*)c!=None && temp!=NULL) {
 | |
| 			 	l_divmod(temp, c, &div, &mod);
 | |
| 			 	XDECREF(div);
 | |
| 			 	DECREF(temp);
 | |
| 			 	temp = mod;
 | |
| 			}
 | |
| 			a = temp;
 | |
| 			if (a == NULL) {
 | |
| 				DECREF(z);
 | |
| 				z = NULL;
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 		if (a == NULL || z == NULL)
 | |
| 			break;
 | |
| 	}
 | |
| 	XDECREF(a);
 | |
| 	if ((object*)c!=None && z!=NULL) {
 | |
| 			l_divmod(z, c, &div, &mod);
 | |
| 			XDECREF(div);
 | |
| 			DECREF(z);
 | |
| 			z=mod;
 | |
| 	}
 | |
| 	return (object *)z;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_invert(v)
 | |
| 	longobject *v;
 | |
| {
 | |
| 	/* Implement ~x as -(x+1) */
 | |
| 	longobject *x;
 | |
| 	longobject *w;
 | |
| 	w = (longobject *)newlongobject(1L);
 | |
| 	if (w == NULL)
 | |
| 		return NULL;
 | |
| 	x = (longobject *) long_add(v, w);
 | |
| 	DECREF(w);
 | |
| 	if (x == NULL)
 | |
| 		return NULL;
 | |
| 	if (x->ob_size != 0)
 | |
| 		x->ob_size = -(x->ob_size);
 | |
| 	return (object *)x;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_pos(v)
 | |
| 	longobject *v;
 | |
| {
 | |
| 	INCREF(v);
 | |
| 	return (object *)v;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_neg(v)
 | |
| 	longobject *v;
 | |
| {
 | |
| 	longobject *z;
 | |
| 	int i, n;
 | |
| 	n = ABS(v->ob_size);
 | |
| 	if (n == 0) {
 | |
| 		/* -0 == 0 */
 | |
| 		INCREF(v);
 | |
| 		return (object *) v;
 | |
| 	}
 | |
| 	z = alloclongobject(ABS(n));
 | |
| 	if (z == NULL)
 | |
| 		return NULL;
 | |
| 	for (i = 0; i < n; i++)
 | |
| 		z->ob_digit[i] = v->ob_digit[i];
 | |
| 	z->ob_size = -(v->ob_size);
 | |
| 	return (object *)z;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_abs(v)
 | |
| 	longobject *v;
 | |
| {
 | |
| 	if (v->ob_size < 0)
 | |
| 		return long_neg(v);
 | |
| 	else {
 | |
| 		INCREF(v);
 | |
| 		return (object *)v;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static int
 | |
| long_nonzero(v)
 | |
| 	longobject *v;
 | |
| {
 | |
| 	return ABS(v->ob_size) != 0;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_rshift(a, b)
 | |
| 	longobject *a;
 | |
| 	longobject *b;
 | |
| {
 | |
| 	longobject *z;
 | |
| 	long shiftby;
 | |
| 	int newsize, wordshift, loshift, hishift, i, j;
 | |
| 	digit lomask, himask;
 | |
| 	
 | |
| 	if (a->ob_size < 0) {
 | |
| 		/* Right shifting negative numbers is harder */
 | |
| 		longobject *a1, *a2, *a3;
 | |
| 		a1 = (longobject *) long_invert(a);
 | |
| 		if (a1 == NULL) return NULL;
 | |
| 		a2 = (longobject *) long_rshift(a1, b);
 | |
| 		DECREF(a1);
 | |
| 		if (a2 == NULL) return NULL;
 | |
| 		a3 = (longobject *) long_invert(a2);
 | |
| 		DECREF(a2);
 | |
| 		return (object *) a3;
 | |
| 	}
 | |
| 	
 | |
| 	shiftby = getlongvalue((object *)b);
 | |
| 	if (shiftby == -1L && err_occurred())
 | |
| 		return NULL;
 | |
| 	if (shiftby < 0) {
 | |
| 		err_setstr(ValueError, "negative shift count");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	wordshift = shiftby / SHIFT;
 | |
| 	newsize = ABS(a->ob_size) - wordshift;
 | |
| 	if (newsize <= 0) {
 | |
| 		z = alloclongobject(0);
 | |
| 		return (object *)z;
 | |
| 	}
 | |
| 	loshift = shiftby % SHIFT;
 | |
| 	hishift = SHIFT - loshift;
 | |
| 	lomask = ((digit)1 << hishift) - 1;
 | |
| 	himask = MASK ^ lomask;
 | |
| 	z = alloclongobject(newsize);
 | |
| 	if (z == NULL)
 | |
| 		return NULL;
 | |
| 	if (a->ob_size < 0)
 | |
| 		z->ob_size = -(z->ob_size);
 | |
| 	for (i = 0, j = wordshift; i < newsize; i++, j++) {
 | |
| 		z->ob_digit[i] = (a->ob_digit[j] >> loshift) & lomask;
 | |
| 		if (i+1 < newsize)
 | |
| 			z->ob_digit[i] |=
 | |
| 			  (a->ob_digit[j+1] << hishift) & himask;
 | |
| 	}
 | |
| 	return (object *) long_normalize(z);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_lshift(a, b)
 | |
| 	longobject *a;
 | |
| 	longobject *b;
 | |
| {
 | |
| 	longobject *z;
 | |
| 	long shiftby;
 | |
| 	int newsize, wordshift, loshift, hishift, i, j;
 | |
| 	digit lomask, himask;
 | |
| 	
 | |
| 	shiftby = getlongvalue((object *)b);
 | |
| 	if (shiftby == -1L && err_occurred())
 | |
| 		return NULL;
 | |
| 	if (shiftby < 0) {
 | |
| 		err_setstr(ValueError, "negative shift count");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	if (shiftby > MASK) {
 | |
| 		err_setstr(ValueError, "outrageous left shift count");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	if (shiftby % SHIFT == 0) {
 | |
| 		wordshift = shiftby / SHIFT;
 | |
| 		loshift = 0;
 | |
| 		hishift = SHIFT;
 | |
| 		newsize = ABS(a->ob_size) + wordshift;
 | |
| 		lomask = MASK;
 | |
| 		himask = 0;
 | |
| 	}
 | |
| 	else {
 | |
| 		wordshift = shiftby / SHIFT + 1;
 | |
| 		loshift = SHIFT - shiftby%SHIFT;
 | |
| 		hishift = shiftby % SHIFT;
 | |
| 		newsize = ABS(a->ob_size) + wordshift;
 | |
| 		lomask = ((digit)1 << hishift) - 1;
 | |
| 		himask = MASK ^ lomask;
 | |
| 	}
 | |
| 	z = alloclongobject(newsize);
 | |
| 	if (z == NULL)
 | |
| 		return NULL;
 | |
| 	if (a->ob_size < 0)
 | |
| 		z->ob_size = -(z->ob_size);
 | |
| 	for (i = 0; i < wordshift; i++)
 | |
| 		z->ob_digit[i] = 0;
 | |
| 	for (i = wordshift, j = 0; i < newsize; i++, j++) {
 | |
| 		if (i > 0)
 | |
| 			z->ob_digit[i-1] |=
 | |
| 				(a->ob_digit[j] << hishift) & himask;
 | |
| 		z->ob_digit[i] =
 | |
| 			(a->ob_digit[j] >> loshift) & lomask;
 | |
| 	}
 | |
| 	return (object *) long_normalize(z);
 | |
| }
 | |
| 
 | |
| 
 | |
| /* Bitwise and/xor/or operations */
 | |
| 
 | |
| #define MAX(x, y) ((x) < (y) ? (y) : (x))
 | |
| #define MIN(x, y) ((x) > (y) ? (y) : (x))
 | |
| 
 | |
| static object *long_bitwise PROTO((longobject *, int, longobject *));
 | |
| static object *
 | |
| long_bitwise(a, op, b)
 | |
| 	longobject *a;
 | |
| 	int op; /* '&', '|', '^' */
 | |
| 	longobject *b;
 | |
| {
 | |
| 	digit maska, maskb; /* 0 or MASK */
 | |
| 	int negz;
 | |
| 	int size_a, size_b, size_z;
 | |
| 	longobject *z;
 | |
| 	int i;
 | |
| 	digit diga, digb;
 | |
| 	object *v;
 | |
| 	
 | |
| 	if (a->ob_size < 0) {
 | |
| 		a = (longobject *) long_invert(a);
 | |
| 		maska = MASK;
 | |
| 	}
 | |
| 	else {
 | |
| 		INCREF(a);
 | |
| 		maska = 0;
 | |
| 	}
 | |
| 	if (b->ob_size < 0) {
 | |
| 		b = (longobject *) long_invert(b);
 | |
| 		maskb = MASK;
 | |
| 	}
 | |
| 	else {
 | |
| 		INCREF(b);
 | |
| 		maskb = 0;
 | |
| 	}
 | |
| 	
 | |
| 	size_a = a->ob_size;
 | |
| 	size_b = b->ob_size;
 | |
| 	size_z = MAX(size_a, size_b);
 | |
| 	z = alloclongobject(size_z);
 | |
| 	if (a == NULL || b == NULL || z == NULL) {
 | |
| 		XDECREF(a);
 | |
| 		XDECREF(b);
 | |
| 		XDECREF(z);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	
 | |
| 	negz = 0;
 | |
| 	switch (op) {
 | |
| 	case '^':
 | |
| 		if (maska != maskb) {
 | |
| 			maska ^= MASK;
 | |
| 			negz = -1;
 | |
| 		}
 | |
| 		break;
 | |
| 	case '&':
 | |
| 		if (maska && maskb) {
 | |
| 			op = '|';
 | |
| 			maska ^= MASK;
 | |
| 			maskb ^= MASK;
 | |
| 			negz = -1;
 | |
| 		}
 | |
| 		break;
 | |
| 	case '|':
 | |
| 		if (maska || maskb) {
 | |
| 			op = '&';
 | |
| 			maska ^= MASK;
 | |
| 			maskb ^= MASK;
 | |
| 			negz = -1;
 | |
| 		}
 | |
| 		break;
 | |
| 	}
 | |
| 	
 | |
| 	for (i = 0; i < size_z; ++i) {
 | |
| 		diga = (i < size_a ? a->ob_digit[i] : 0) ^ maska;
 | |
| 		digb = (i < size_b ? b->ob_digit[i] : 0) ^ maskb;
 | |
| 		switch (op) {
 | |
| 		case '&': z->ob_digit[i] = diga & digb; break;
 | |
| 		case '|': z->ob_digit[i] = diga | digb; break;
 | |
| 		case '^': z->ob_digit[i] = diga ^ digb; break;
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	DECREF(a);
 | |
| 	DECREF(b);
 | |
| 	z = long_normalize(z);
 | |
| 	if (negz == 0)
 | |
| 		return (object *) z;
 | |
| 	v = long_invert(z);
 | |
| 	DECREF(z);
 | |
| 	return v;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_and(a, b)
 | |
| 	longobject *a;
 | |
| 	longobject *b;
 | |
| {
 | |
| 	return long_bitwise(a, '&', b);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_xor(a, b)
 | |
| 	longobject *a;
 | |
| 	longobject *b;
 | |
| {
 | |
| 	return long_bitwise(a, '^', b);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_or(a, b)
 | |
| 	longobject *a;
 | |
| 	longobject *b;
 | |
| {
 | |
| 	return long_bitwise(a, '|', b);
 | |
| }
 | |
| 
 | |
| static int
 | |
| long_coerce(pv, pw)
 | |
| 	object **pv;
 | |
| 	object **pw;
 | |
| {
 | |
| 	if (is_intobject(*pw)) {
 | |
| 		*pw = newlongobject(getintvalue(*pw));
 | |
| 		INCREF(*pv);
 | |
| 		return 0;
 | |
| 	}
 | |
| 	return 1; /* Can't do it */
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_int(v)
 | |
| 	object *v;
 | |
| {
 | |
| 	long x;
 | |
| 	x = getlongvalue(v);
 | |
| 	if (err_occurred())
 | |
| 		return NULL;
 | |
| 	return newintobject(x);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_long(v)
 | |
| 	object *v;
 | |
| {
 | |
| 	INCREF(v);
 | |
| 	return v;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_float(v)
 | |
| 	object *v;
 | |
| {
 | |
| 	return newfloatobject(dgetlongvalue(v));
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_oct(v)
 | |
| 	object *v;
 | |
| {
 | |
| 	return long_format(v, 8);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| long_hex(v)
 | |
| 	object *v;
 | |
| {
 | |
| 	return long_format(v, 16);
 | |
| }
 | |
| 
 | |
| 
 | |
| #define UF (unaryfunc)
 | |
| #define BF (binaryfunc)
 | |
| #define TF (ternaryfunc)
 | |
| #define IF (inquiry)
 | |
| 
 | |
| static number_methods long_as_number = {
 | |
| 	BF long_add,	/*nb_add*/
 | |
| 	BF long_sub,	/*nb_subtract*/
 | |
| 	BF long_mul,	/*nb_multiply*/
 | |
| 	BF long_div,	/*nb_divide*/
 | |
| 	BF long_mod,	/*nb_remainder*/
 | |
| 	BF long_divmod,	/*nb_divmod*/
 | |
| 	TF long_pow,	/*nb_power*/
 | |
| 	UF long_neg,	/*nb_negative*/
 | |
| 	UF long_pos,	/*tp_positive*/
 | |
| 	UF long_abs,	/*tp_absolute*/
 | |
| 	IF long_nonzero,/*tp_nonzero*/
 | |
| 	UF long_invert,	/*nb_invert*/
 | |
| 	BF long_lshift,	/*nb_lshift*/
 | |
| 	BF long_rshift,	/*nb_rshift*/
 | |
| 	BF long_and,	/*nb_and*/
 | |
| 	BF long_xor,	/*nb_xor*/
 | |
| 	BF long_or,	/*nb_or*/
 | |
| 	(int (*) FPROTO((object **, object **)))
 | |
| 	(coercion)long_coerce, /*nb_coerce*/
 | |
| 	UF long_int,	/*nb_int*/
 | |
| 	UF long_long,	/*nb_long*/
 | |
| 	UF long_float,	/*nb_float*/
 | |
| 	UF long_oct,	/*nb_oct*/
 | |
| 	UF long_hex,	/*nb_hex*/
 | |
| };
 | |
| 
 | |
| typeobject Longtype = {
 | |
| 	OB_HEAD_INIT(&Typetype)
 | |
| 	0,
 | |
| 	"long int",
 | |
| 	sizeof(longobject) - sizeof(digit),
 | |
| 	sizeof(digit),
 | |
| 	(destructor)long_dealloc, /*tp_dealloc*/
 | |
| 	0,		/*tp_print*/
 | |
| 	0,		/*tp_getattr*/
 | |
| 	0,		/*tp_setattr*/
 | |
| 	(int (*) FPROTO((object *, object *)))
 | |
| 	(cmpfunc)long_compare, /*tp_compare*/
 | |
| 	(reprfunc)long_repr, /*tp_repr*/
 | |
| 	&long_as_number,/*tp_as_number*/
 | |
| 	0,		/*tp_as_sequence*/
 | |
| 	0,		/*tp_as_mapping*/
 | |
| 	(long (*) FPROTO((object *)))
 | |
| 	(hashfunc)long_hash, /*tp_hash*/
 | |
| };
 | 
