Added stix_mulints()

This commit is contained in:
hyunghwan.chung 2015-11-11 13:31:05 +00:00
parent e519c5078e
commit 0daf143cce
8 changed files with 425 additions and 101 deletions

View File

@ -115,15 +115,18 @@
{ {
} }
#class(#halfword) LargeInteger(Number) ## #class(#halfword) LargeInteger(Number)
#class(#word) LargeInteger(Number)
{ {
} }
#class(#halfword) LargePositiveInteger(LargeInteger) ## #class(#halfword) LargePositiveInteger(LargeInteger)
#class(#word) LargePositiveInteger(LargeInteger)
{ {
} }
#class(#halfword) LargeNegativeInteger(LargeInteger) ## #class(#halfword) LargeNegativeInteger(LargeInteger)
#class(#word) LargeNegativeInteger(LargeInteger)
{ {
} }

View File

@ -257,6 +257,10 @@ PROCESS TESTING
'444444444444444444' dump. '444444444444444444' dump.
" "
(-2305843009213693952 - 1) dump. (-2305843009213693952 - 1) dump.
##(-2305843009213693952 * 2305843009213693952) dump.
(((-2305843009213693952 - 10) * (-2305843009213693952 - 10) *(-2305843009213693952 - 10) * (-2305843009213693952 - 10) * 255) * ((-2305843009213693952 - 10) * (-2305843009213693952 - 10) *(-2305843009213693952 - 10) * (-2305843009213693952 - 10) * 255)) dump.
##((-2305843009213693952 - 10) * (-2305843009213693952 - 10) *(-2305843009213693952 - 10) * (-2305843009213693952 - 10) * 255) dump.
" "
FFI isNil dump. FFI isNil dump.
FFI notNil dump. FFI notNil dump.

View File

@ -26,7 +26,6 @@
#include "stix-prv.h" #include "stix-prv.h"
#define MAKE_WORD(hw1,hw2) ((stix_oow_t)(hw1) | (stix_oow_t)(hw2) << STIX_OOHW_BITS)
/*#define IS_POWER_OF_2(ui) (((ui) > 0) && (((ui) & (~(ui)+ 1)) == (ui)))*/ /*#define IS_POWER_OF_2(ui) (((ui) > 0) && (((ui) & (~(ui)+ 1)) == (ui)))*/
#define IS_POWER_OF_2(ui) (((ui) > 0) && ((ui) & ((ui) - 1)) == 0) /* unsigned integer only */ #define IS_POWER_OF_2(ui) (((ui) > 0) && ((ui) & ((ui) - 1)) == 0) /* unsigned integer only */
@ -45,19 +44,6 @@ static STIX_INLINE int oow_add_overflow (stix_oow_t a, stix_oow_t b, stix_oow_t*
} }
#endif #endif
#if (STIX_SIZEOF_OOHW_T == STIX_SIZEOF_INT) && defined(STIX_HAVE_BUILTIN_UADD_OVERFLOW)
# define oohw_add_overflow(a,b,c) __builtin_uadd_overflow(a,b,c)
#elif (STIX_SIZEOF_OOHW_T == STIX_SIZEOF_LONG) && defined(STIX_HAVE_BUILTIN_UADDL_OVERFLOW)
# define oohw_add_overflow(a,b,c) __builtin_uaddl_overflow(a,b,c)
#elif (STIX_SIZEOF_OOHW_T == STIX_SIZEOF_LONG_LONG) && defined(STIX_HAVE_BUILTIN_UADDLL_OVERFLOW)
# define oohw_add_overflow(a,b,c) __builtin_uaddll_overflow(a,b,c)
#else
static STIX_INLINE int oohw_add_overflow (stix_oohw_t a, stix_oohw_t b, stix_oohw_t* c)
{
*c = a + b;
return b > STIX_TYPE_MAX(stix_oohw_t) - a;
}
#endif
#if (STIX_SIZEOF_OOW_T == STIX_SIZEOF_INT) && defined(STIX_HAVE_BUILTIN_UMUL_OVERFLOW) #if (STIX_SIZEOF_OOW_T == STIX_SIZEOF_INT) && defined(STIX_HAVE_BUILTIN_UMUL_OVERFLOW)
# define oow_mul_overflow(a,b,c) __builtin_umul_overflow(a,b,c) # define oow_mul_overflow(a,b,c) __builtin_umul_overflow(a,b,c)
@ -81,6 +67,42 @@ static STIX_INLINE int oow_mul_overflow (stix_oow_t a, stix_oow_t b, stix_oow_t*
} }
#endif #endif
#if (SIZEOF_ATOM_T == STIX_SIZEOF_INT) && defined(STIX_HAVE_BUILTIN_UADD_OVERFLOW)
# define atom_add_overflow(a,b,c) __builtin_uadd_overflow(a,b,c)
#elif (SIZEOF_ATOM_T == STIX_SIZEOF_LONG) && defined(STIX_HAVE_BUILTIN_UADDL_OVERFLOW)
# define atom_add_overflow(a,b,c) __builtin_uaddl_overflow(a,b,c)
#elif (SIZEOF_ATOM_T == STIX_SIZEOF_LONG_LONG) && defined(STIX_HAVE_BUILTIN_UADDLL_OVERFLOW)
# define atom_add_overflow(a,b,c) __builtin_uaddll_overflow(a,b,c)
#else
static STIX_INLINE int atom_add_overflow (atom_t a, atom_t b, atom_t* c)
{
*c = a + b;
return b > STIX_TYPE_MAX(atom_t) - a;
}
#endif
#if (SIZEOF_ATOM_T == STIX_SIZEOF_INT) && defined(STIX_HAVE_BUILTIN_UMUL_OVERFLOW)
# define atom_mul_overflow(a,b,c) __builtin_umul_overflow(a,b,c)
#elif (SIZEOF_ATOM_T == STIX_SIZEOF_LONG) && defined(STIX_HAVE_BUILTIN_UMULL_OVERFLOW)
# define atom_mul_overflow(a,b,c) __builtin_uaddl_overflow(a,b,c)
#elif (SIZEOF_ATOM_T == STIX_SIZEOF_LONG_LONG) && defined(STIX_HAVE_BUILTIN_UMULLL_OVERFLOW)
# define atom_mul_overflow(a,b,c) __builtin_uaddll_overflow(a,b,c)
#else
static STIX_INLINE int atom_mul_overflow (atom_t a, atom_t b, atom_t* c)
{
#if (STIX_SIZEOF_UINTMAX_T > SIZEOF_ATOM_T)
stix_uintmax_t k;
k = (stix_uintmax_t)a * (stix_uintmax_t)b;
*c = (atom_t)k;
return (k >> ATOM_BITS) > 0;
/*return k > STIX_TYPE_MAX(atom_t);*/
#else
*c = a * b;
return b > 0 && a > STIX_TYPE_MAX(atom_t) / b; /* works for unsigned types only */
#endif
}
#endif
static STIX_INLINE int is_integer (stix_t* stix, stix_oop_t oop) static STIX_INLINE int is_integer (stix_t* stix, stix_oop_t oop)
{ {
stix_oop_t c; stix_oop_t c;
@ -95,23 +117,43 @@ static STIX_INLINE int is_integer (stix_t* stix, stix_oop_t oop)
static STIX_INLINE stix_oop_t make_bigint_with_ooi (stix_t* stix, stix_ooi_t i) static STIX_INLINE stix_oop_t make_bigint_with_ooi (stix_t* stix, stix_ooi_t i)
{ {
stix_oohw_t hw[2]; #if defined(USE_FULL_WORD)
stix_oow_t w; stix_oow_t w;
if (i >= 0) if (i >= 0)
{ {
w = i; w = i;
hw[0] = w & STIX_LBMASK(stix_oow_t,STIX_OOHW_BITS); return stix_instantiate (stix, stix->_large_positive_integer, &w, 1);
hw[1] = w >> STIX_OOHW_BITS; }
else
{
/* The caller must ensure that i is grater than the smallest value
* that stix_ooi_t can represent. otherwise, the absolute value
* cannot be held in stix_ooi_t. */
STIX_ASSERT (i > STIX_TYPE_MIN(stix_ooi_t));
w = -i;
return stix_instantiate (stix, stix->_large_negative_integer, &w, 1);
}
#else
atom_t hw[2];
stix_oow_t w;
if (i >= 0)
{
w = i;
hw[0] = w & STIX_LBMASK(stix_oow_t,ATOM_BITS);
hw[1] = w >> ATOM_BITS;
return stix_instantiate (stix, stix->_large_positive_integer, &hw, (hw[1] > 0? 2: 1)); return stix_instantiate (stix, stix->_large_positive_integer, &hw, (hw[1] > 0? 2: 1));
} }
else else
{ {
STIX_ASSERT (i > STIX_TYPE_MIN(stix_ooi_t));
w = -i; w = -i;
hw[0] = w & STIX_LBMASK(stix_oow_t,STIX_OOHW_BITS); hw[0] = w & STIX_LBMASK(stix_oow_t,ATOM_BITS);
hw[1] = w >> STIX_OOHW_BITS; hw[1] = w >> ATOM_BITS;
return stix_instantiate (stix, stix->_large_negative_integer, &hw, (hw[1] > 0? 2: 1)); return stix_instantiate (stix, stix->_large_negative_integer, &hw, (hw[1] > 0? 2: 1));
} }
#endif
} }
static STIX_INLINE stix_oop_t clone_bigint (stix_t* stix, stix_oop_t oop, stix_oow_t count) static STIX_INLINE stix_oop_t clone_bigint (stix_t* stix, stix_oop_t oop, stix_oow_t count)
@ -129,7 +171,7 @@ static STIX_INLINE stix_oop_t clone_bigint (stix_t* stix, stix_oop_t oop, stix_o
for (i = 0; i < count; i++) for (i = 0; i < count; i++)
{ {
((stix_oop_halfword_t)z)->slot[i] = ((stix_oop_halfword_t)oop)->slot[i]; ((oop_atom_t)z)->slot[i] = ((oop_atom_t)oop)->slot[i];
} }
return z; return z;
} }
@ -141,7 +183,7 @@ static STIX_INLINE stix_oow_t count_effective_digits (stix_oop_t oop)
for (i = STIX_OBJ_GET_SIZE(oop); i > 1; ) for (i = STIX_OBJ_GET_SIZE(oop); i > 1; )
{ {
--i; --i;
if (((stix_oop_halfword_t)oop)->slot[i] != 0) return i + 1; if (((oop_atom_t)oop)->slot[i] != 0) return i + 1;
} }
return 1; return 1;
@ -153,23 +195,13 @@ static stix_oop_t normalize_bigint (stix_t* stix, stix_oop_t oop)
STIX_ASSERT (STIX_OOP_IS_POINTER(oop)); STIX_ASSERT (STIX_OOP_IS_POINTER(oop));
count = count_effective_digits (oop); count = count_effective_digits (oop);
#if defined(USE_FULL_WORD)
if (count == 1) if (count == 1)
{
if (STIX_OBJ_GET_CLASS(oop) == stix->_large_positive_integer)
{
return STIX_OOP_FROM_SMINT(((stix_oop_halfword_t)oop)->slot[0]);
}
else
{
STIX_ASSERT (STIX_OBJ_GET_CLASS(oop) == stix->_large_negative_integer);
return STIX_OOP_FROM_SMINT(-(stix_oow_t)((stix_oop_halfword_t)oop)->slot[0]);
}
}
else if (count == 2)
{ {
stix_oow_t w; stix_oow_t w;
w = MAKE_WORD (((stix_oop_halfword_t)oop)->slot[0], ((stix_oop_halfword_t)oop)->slot[1]); w = ((oop_atom_t)oop)->slot[0];
if (STIX_OBJ_GET_CLASS(oop) == stix->_large_positive_integer) if (STIX_OBJ_GET_CLASS(oop) == stix->_large_positive_integer)
{ {
if (w <= STIX_SMINT_MAX) return STIX_OOP_FROM_SMINT(w); if (w <= STIX_SMINT_MAX) return STIX_OOP_FROM_SMINT(w);
@ -181,7 +213,36 @@ static stix_oop_t normalize_bigint (stix_t* stix, stix_oop_t oop)
if (w <= ((stix_oow_t)STIX_SMINT_MAX + 1)) return STIX_OOP_FROM_SMINT(-(stix_ooi_t)w); if (w <= ((stix_oow_t)STIX_SMINT_MAX + 1)) return STIX_OOP_FROM_SMINT(-(stix_ooi_t)w);
} }
} }
#else
if (count == 1)
{
if (STIX_OBJ_GET_CLASS(oop) == stix->_large_positive_integer)
{
return STIX_OOP_FROM_SMINT(((oop_atom_t)oop)->slot[0]);
}
else
{
STIX_ASSERT (STIX_OBJ_GET_CLASS(oop) == stix->_large_negative_integer);
return STIX_OOP_FROM_SMINT(-(stix_oow_t)((oop_atom_t)oop)->slot[0]);
}
}
else if (count == 2)
{
stix_oow_t w;
w = MAKE_WORD (((oop_atom_t)oop)->slot[0], ((oop_atom_t)oop)->slot[1]);
if (STIX_OBJ_GET_CLASS(oop) == stix->_large_positive_integer)
{
if (w <= STIX_SMINT_MAX) return STIX_OOP_FROM_SMINT(w);
}
else
{
STIX_ASSERT (STIX_OBJ_GET_CLASS(oop) == stix->_large_negative_integer);
/*if (w <= -STIX_SMINT_MIN) */
if (w <= ((stix_oow_t)STIX_SMINT_MAX + 1)) return STIX_OOP_FROM_SMINT(-(stix_ooi_t)w);
}
}
#endif
if (STIX_OBJ_GET_SIZE(oop) == count) if (STIX_OBJ_GET_SIZE(oop) == count)
{ {
/* no compaction is needed. return it as it is */ /* no compaction is needed. return it as it is */
@ -192,7 +253,7 @@ static stix_oop_t normalize_bigint (stix_t* stix, stix_oop_t oop)
} }
static STIX_INLINE int is_less_unsigned_array (const stix_oohw_t* x, stix_oow_t xs, const stix_oohw_t* y, stix_oow_t ys) static STIX_INLINE int is_less_unsigned_array (const atom_t* x, stix_oow_t xs, const atom_t* y, stix_oow_t ys)
{ {
stix_oow_t i; stix_oow_t i;
@ -209,8 +270,8 @@ static STIX_INLINE int is_less_unsigned_array (const stix_oohw_t* x, stix_oow_t
static STIX_INLINE int is_less_unsigned (stix_oop_t x, stix_oop_t y) static STIX_INLINE int is_less_unsigned (stix_oop_t x, stix_oop_t y)
{ {
return is_less_unsigned_array ( return is_less_unsigned_array (
((stix_oop_halfword_t)x)->slot, STIX_OBJ_GET_SIZE(x), ((oop_atom_t)x)->slot, STIX_OBJ_GET_SIZE(x),
((stix_oop_halfword_t)y)->slot, STIX_OBJ_GET_SIZE(y)); ((oop_atom_t)y)->slot, STIX_OBJ_GET_SIZE(y));
} }
static STIX_INLINE int is_less (stix_t* stix, stix_oop_t x, stix_oop_t y) static STIX_INLINE int is_less (stix_t* stix, stix_oop_t x, stix_oop_t y)
@ -227,55 +288,58 @@ static STIX_INLINE int is_equal (stix_t* stix, stix_oop_t x, stix_oop_t y)
{ {
/* check if two large integers are equal to each other */ /* check if two large integers are equal to each other */
return STIX_OBJ_GET_CLASS(x) == STIX_OBJ_GET_CLASS(y) && STIX_OBJ_GET_SIZE(x) == STIX_OBJ_GET_SIZE(y) && return STIX_OBJ_GET_CLASS(x) == STIX_OBJ_GET_CLASS(y) && STIX_OBJ_GET_SIZE(x) == STIX_OBJ_GET_SIZE(y) &&
STIX_MEMCMP(((stix_oop_halfword_t)x)->slot, ((stix_oop_halfword_t)y)->slot, STIX_OBJ_GET_SIZE(x) * STIX_SIZEOF(stix_oohw_t)) == 0; STIX_MEMCMP(((oop_atom_t)x)->slot, ((oop_atom_t)y)->slot, STIX_OBJ_GET_SIZE(x) * STIX_SIZEOF(atom_t)) == 0;
} }
static STIX_INLINE stix_oow_t add_unsigned_array (const stix_oohw_t* x, stix_oow_t xs, const stix_oohw_t* y, stix_oow_t ys, stix_oohw_t* z) static STIX_INLINE stix_oow_t add_unsigned_array (const atom_t* x, stix_oow_t xs, const atom_t* y, stix_oow_t ys, atom_t* z)
{ {
stix_oow_t i, w; stix_oow_t i;
stix_oow_t carry = 0; bigatom_t w;
bigatom_t carry = 0;
STIX_ASSERT (xs >= ys); STIX_ASSERT (xs >= ys);
for (i = 0; i < ys; i++) for (i = 0; i < ys; i++)
{ {
w = (stix_oow_t)x[i] + (stix_oow_t)y[i] + carry; w = (bigatom_t)x[i] + (bigatom_t)y[i] + carry;
carry = w >> STIX_OOHW_BITS; carry = w >> ATOM_BITS;
z[i] = w & STIX_LBMASK(stix_oow_t, STIX_OOHW_BITS); z[i] = w /*& STIX_LBMASK(bigatom_t, ATOM_BITS) */;
} }
for (; i < xs; i++) for (; i < xs; i++)
{ {
w = (stix_oow_t)x[i] + carry; w = (bigatom_t)x[i] + carry;
carry = w >> STIX_OOHW_BITS; carry = w >> ATOM_BITS;
z[i] = w & STIX_LBMASK(stix_oow_t, STIX_OOHW_BITS); z[i] = w /*& STIX_LBMASK(bigatom_t, ATOM_BITS)*/;
} }
if (i > 1 && carry == 0) return i - 1; if (i > 1 && carry == 0) return i - 1;
z[i] = carry; z[i] = carry;
return i; return i;
} }
static STIX_INLINE stix_oow_t subtract_unsigned_array (const stix_oohw_t* x, stix_oow_t xs, const stix_oohw_t* y, stix_oow_t ys, stix_oohw_t* z) static STIX_INLINE stix_oow_t subtract_unsigned_array (const atom_t* x, stix_oow_t xs, const atom_t* y, stix_oow_t ys, atom_t* z)
{ {
stix_oow_t i, w; stix_oow_t i;
stix_oow_t borrow = 0; bigatom_t w;
stix_oow_t borrowed_word; bigatom_t borrow = 0;
bigatom_t borrowed_word;
STIX_ASSERT (!is_less_unsigned_array(x, xs, y, ys)); STIX_ASSERT (!is_less_unsigned_array(x, xs, y, ys));
borrowed_word = (stix_oow_t)1 << STIX_OOHW_BITS; borrowed_word = (bigatom_t)1 << ATOM_BITS;
for (i = 0; i < ys; i++) for (i = 0; i < ys; i++)
{ {
w = (stix_oow_t)y[i] + borrow; w = (bigatom_t)y[i] + borrow;
if ((stix_oow_t)x[i] >= w) if ((bigatom_t)x[i] >= w)
{ {
z[i] = x[i] - w; z[i] = x[i] - w;
borrow = 0; borrow = 0;
} }
else else
{ {
z[i] = (borrowed_word + (stix_oow_t)x[i]) - w; z[i] = (borrowed_word + (bigatom_t)x[i]) - w;
borrow = 1; borrow = 1;
} }
} }
@ -284,12 +348,12 @@ static STIX_INLINE stix_oow_t subtract_unsigned_array (const stix_oohw_t* x, sti
{ {
if (x[i] >= borrow) if (x[i] >= borrow)
{ {
z[i] = x[i] - (stix_oohw_t)borrow; z[i] = x[i] - (atom_t)borrow;
borrow = 0; borrow = 0;
} }
else else
{ {
z[i] = (borrowed_word + (stix_oow_t)x[i]) - borrow; z[i] = (borrowed_word + (bigatom_t)x[i]) - borrow;
borrow = 1; borrow = 1;
} }
} }
@ -298,9 +362,101 @@ static STIX_INLINE stix_oow_t subtract_unsigned_array (const stix_oohw_t* x, sti
return i; return i;
} }
static STIX_INLINE void multiply_unsigned_array (const atom_t* x, stix_oow_t xs, const atom_t* y, stix_oow_t ys, atom_t* z)
{
#if 0
/* Comba multiplication */ /* TODO: implement Karatsuba algorithm when the input length is long */
/* TODO: I don't know why this doesn't work
0FF0000000000013EC00000000000956A00000000001F20C000000000026E8F0 * 0FF0000000000013EC00000000000956A00000000001F20C000000000026E8F0 must
produce
00FE010000000002 7B028000000002B6 8ABC00000001B216 B580000000A990DE E60000002A6437B9 800000069FA8B4FC 0000009765EB9680 000005E9FB33E100
but this produce
00FE010000000002 7B028000000002B6 8ABC00000001B215 B580000000A990DE E60000002A6437B9 800000069FA8B4FC 0000009765EB9680 000005E9FB33E100*
* */
bigatom_t v;
stix_oow_t s1, s2;
stix_oow_t i, j;
stix_oow_t a, b;
s1 = xs + ys;
v = 0;
for (i = 0; i < s1; i++)
{
b = (i < ys - 1)? i: ys - 1;
a = i - b;
s2 = (xs - a < b + 1)? xs - a: b + 1;
for (j = 0; j < s2; j++)
{
v += (bigatom_t)x[a + j] * (bigatom_t)y[b - j];
}
z[i] = (atom_t)v;
v >>= ATOM_BITS;
}
#elif 1
stix_oow_t i, j;
bigatom_t v;
atom_t carry;
for (i = 0; i < ys; i++)
{
if (y[i] == 0)
{
z[xs + i] = 0;
}
else
{
carry = 0;
for (j = 0; j < xs; j++)
{
v = (bigatom_t)x[j] * (bigatom_t)y[i] + (bigatom_t)carry + (bigatom_t)z[j + i];
z[j + i] = (atom_t)v;
carry = (atom_t)(v >> ATOM_BITS);
}
z[xs + i] = carry;
}
}
#else
stix_oow_t i, j, idx;
bigatom_t v;
atom_t carry;
for (i = 0; i < ys; i++)
{
idx = i;
for (j = 0; j < xs; j++)
{
v = (bigatom_t)x[j] * (bigatom_t)y[i] + (bigatom_t)carry + (bigatom_t)z[idx];
z[idx] = (atom_t)v;
carry = (atom_t)(v >> ATOM_BITS);
idx++;
}
while (carry > 0)
{
v = (bigatom_t)z[idx] + (bigatom_t)carry;
z[idx] = (atom_t)v;
carry = (atom_t)(v >> ATOM_BITS);
idx++;
}
}
#endif
}
static stix_oop_t add_unsigned_integers (stix_t* stix, stix_oop_t x, stix_oop_t y) static stix_oop_t add_unsigned_integers (stix_t* stix, stix_oop_t x, stix_oop_t y)
{ {
stix_oohw_t* a, * b; atom_t* a, * b;
stix_oow_t as, bs, zs; stix_oow_t as, bs, zs;
stix_oop_t z; stix_oop_t z;
@ -315,16 +471,16 @@ static stix_oop_t add_unsigned_integers (stix_t* stix, stix_oop_t x, stix_oop_t
if (as >= bs) if (as >= bs)
{ {
a = ((stix_oop_halfword_t)x)->slot; a = ((oop_atom_t)x)->slot;
b = ((stix_oop_halfword_t)y)->slot; b = ((oop_atom_t)y)->slot;
} }
else else
{ {
a = ((stix_oop_halfword_t)y)->slot; a = ((oop_atom_t)y)->slot;
b = ((stix_oop_halfword_t)x)->slot; b = ((oop_atom_t)x)->slot;
} }
add_unsigned_array (a, as, b, bs, ((stix_oop_halfword_t)z)->slot); add_unsigned_array (a, as, b, bs, ((oop_atom_t)z)->slot);
return z; return z;
} }
@ -340,9 +496,25 @@ static stix_oop_t subtract_unsigned_integers (stix_t* stix, stix_oop_t x, stix_o
stix_poptmps (stix, 2); stix_poptmps (stix, 2);
subtract_unsigned_array ( subtract_unsigned_array (
((stix_oop_halfword_t)x)->slot, STIX_OBJ_GET_SIZE(x), ((oop_atom_t)x)->slot, STIX_OBJ_GET_SIZE(x),
((stix_oop_halfword_t)y)->slot, STIX_OBJ_GET_SIZE(y), ((oop_atom_t)y)->slot, STIX_OBJ_GET_SIZE(y),
((stix_oop_halfword_t)z)->slot); ((oop_atom_t)z)->slot);
return z;
}
static stix_oop_t multiply_unsigned_integers (stix_t* stix, stix_oop_t x, stix_oop_t y)
{
stix_oop_t z;
stix_pushtmp (stix, &x);
stix_pushtmp (stix, &y);
z = stix_instantiate (stix, stix->_large_positive_integer, STIX_NULL, STIX_OBJ_GET_SIZE(x) + STIX_OBJ_GET_SIZE(y));
stix_poptmps (stix, 2);
multiply_unsigned_array (
((oop_atom_t)x)->slot, STIX_OBJ_GET_SIZE(x),
((oop_atom_t)y)->slot, STIX_OBJ_GET_SIZE(y),
((oop_atom_t)z)->slot);
return z; return z;
} }
@ -358,6 +530,8 @@ stix_oop_t stix_addints (stix_t* stix, stix_oop_t x, stix_oop_t y)
i = STIX_OOP_TO_SMINT(x) + STIX_OOP_TO_SMINT(y); i = STIX_OOP_TO_SMINT(x) + STIX_OOP_TO_SMINT(y);
if (STIX_OOI_IN_SMINT_RANGE(i)) return STIX_OOP_FROM_SMINT(i); if (STIX_OOI_IN_SMINT_RANGE(i)) return STIX_OOP_FROM_SMINT(i);
STIX_ASSERT (STIX_SMINT_MAX + STIX_SMINT_MAX < STIX_TYPE_MAX(stix_ooi_t));
STIX_ASSERT (STIX_SMINT_MIN + STIX_SMINT_MIN > STIX_TYPE_MAX(stix_ooi_t));
return make_bigint_with_ooi (stix, i); return make_bigint_with_ooi (stix, i);
} }
else else
@ -518,6 +692,82 @@ oops_einval:
return STIX_NULL; return STIX_NULL;
} }
stix_oop_t stix_mulints (stix_t* stix, stix_oop_t x, stix_oop_t y)
{
stix_oop_t z;
if (STIX_OOP_IS_SMINT(x) && STIX_OOP_IS_SMINT(y))
{
/* TODO: XXXXXXXXXXXXXXXXXXXXXXXxx */
stix_ooi_t i;
/* no integer overflow/underflow must occur as the possible integer
* range is narrowed by the tag bits used */
i = STIX_OOP_TO_SMINT(x) * STIX_OOP_TO_SMINT(y);
if (STIX_OOI_IN_SMINT_RANGE(i)) return STIX_OOP_FROM_SMINT(i);
STIX_ASSERT (STIX_SMINT_MAX + STIX_SMINT_MAX < STIX_TYPE_MAX(stix_ooi_t));
STIX_ASSERT (STIX_SMINT_MIN + STIX_SMINT_MIN > STIX_TYPE_MAX(stix_ooi_t));
return make_bigint_with_ooi (stix, i);
}
else
{
stix_ooi_t v;
if (STIX_OOP_IS_SMINT(x))
{
if (!is_integer(stix,y)) goto oops_einval;
v = STIX_OOP_TO_SMINT(x);
if (v == 0) return STIX_OOP_FROM_SMINT(0);
stix_pushtmp (stix, &y);
x = make_bigint_with_ooi (stix, v);
stix_poptmp (stix);
if (!x) return STIX_NULL;
}
else if (STIX_OOP_IS_SMINT(y))
{
if (!is_integer(stix,x)) goto oops_einval;
v = STIX_OOP_TO_SMINT(y);
if (v == 0) return STIX_OOP_FROM_SMINT(0);
stix_pushtmp (stix, &x);
y = make_bigint_with_ooi (stix, v);
stix_poptmp (stix);
if (!y) return STIX_NULL;
}
else
{
if (!is_integer(stix,x)) goto oops_einval;
if (!is_integer(stix,y)) goto oops_einval;
}
z = multiply_unsigned_integers (stix, x, y);
if (!z) return STIX_NULL;
if (STIX_OBJ_GET_CLASS(x) != STIX_OBJ_GET_CLASS(y))
STIX_OBJ_SET_CLASS(z, stix->_large_negative_integer);
}
{ int i;
printf ("MUL=>");
for (i = STIX_OBJ_GET_SIZE(z); i > 0;)
{
printf ("%016lX ", (unsigned long)((oop_atom_t)z)->slot[--i]);
}
printf ("\n");
}
return normalize_bigint (stix, z);
oops_einval:
stix->errnum = STIX_EINVAL;
return STIX_NULL;
}
static stix_uint8_t ooch_val_tab[] = static stix_uint8_t ooch_val_tab[] =
{ {
99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99,
@ -534,8 +784,8 @@ stix_oop_t stix_strtoint (stix_t* stix, const stix_ooch_t* str, stix_oow_t len,
{ {
int neg = 0; int neg = 0;
const stix_ooch_t* ptr, * start, * end; const stix_ooch_t* ptr, * start, * end;
stix_oow_t w, v; bigatom_t w, v;
stix_oohw_t hw[64], * hwp = STIX_NULL; atom_t hw[16], * hwp = STIX_NULL;
stix_oow_t hwlen, outlen; stix_oow_t hwlen, outlen;
stix_oop_t res; stix_oop_t res;
@ -614,7 +864,7 @@ stix_oop_t stix_strtoint (stix_t* stix, const stix_ooch_t* str, stix_oow_t len,
/* bytes */ /* bytes */
outlen = ((stix_oow_t)(end - str) * exp + 7) / 8; outlen = ((stix_oow_t)(end - str) * exp + 7) / 8;
/* number of stix_oohw_t */ /* number of atom_t */
outlen = (outlen + STIX_SIZEOF(hw[0]) - 1) / STIX_SIZEOF(hw[0]); outlen = (outlen + STIX_SIZEOF(hw[0]) - 1) / STIX_SIZEOF(hw[0]);
if (outlen > STIX_COUNTOF(hw)) if (outlen > STIX_COUNTOF(hw))
@ -639,35 +889,35 @@ stix_oop_t stix_strtoint (stix_t* stix, const stix_ooch_t* str, stix_oow_t len,
w |= (v << bitcnt); w |= (v << bitcnt);
bitcnt += exp; bitcnt += exp;
if (bitcnt >= STIX_OOHW_BITS) if (bitcnt >= ATOM_BITS)
{ {
bitcnt -= STIX_OOHW_BITS; bitcnt -= ATOM_BITS;
hwp[hwlen++] = (stix_oohw_t)(w & STIX_LBMASK(stix_oow_t, STIX_OOHW_BITS)); hwp[hwlen++] = w; /*(atom_t)(w & STIX_LBMASK(bigatom_t, ATOM_BITS));*/
w >>= STIX_OOHW_BITS; w >>= ATOM_BITS;
} }
ptr--; ptr--;
} }
STIX_ASSERT (w <= STIX_TYPE_MAX(stix_oohw_t)); STIX_ASSERT (w <= STIX_TYPE_MAX(atom_t));
if (hwlen == 0 || w > 0) hwp[hwlen++] = w; if (hwlen == 0 || w > 0) hwp[hwlen++] = w;
} }
else else
{ {
stix_oow_t r1, r2; bigatom_t r1, r2;
stix_oohw_t multiplier; atom_t multiplier;
int dg, i, safe_ndigits; int dg, i, safe_ndigits;
w = 0; w = 0;
ptr = start; ptr = start;
safe_ndigits = stix->bigint[radix].safe_ndigits; safe_ndigits = stix->bigint[radix].safe_ndigits;
multiplier = stix->bigint[radix].multiplier; multiplier = (atom_t)stix->bigint[radix].multiplier;
outlen = (end - str) / safe_ndigits + 1; outlen = (end - str) / safe_ndigits + 1;
if (outlen > STIX_COUNTOF(hw)) if (outlen > STIX_COUNTOF(hw))
{ {
hwp = stix_allocmem (stix, outlen * STIX_SIZEOF(stix_oohw_t)); hwp = stix_allocmem (stix, outlen * STIX_SIZEOF(atom_t));
if (!hwp) return STIX_NULL; if (!hwp) return STIX_NULL;
} }
else else
@ -692,31 +942,32 @@ stix_oop_t stix_strtoint (stix_t* stix, const stix_ooch_t* str, stix_oow_t len,
v = ooch_val_tab[*ptr]; v = ooch_val_tab[*ptr];
if (v >= radix) goto oops_einval; if (v >= radix) goto oops_einval;
r1 = r1 * radix + (stix_oohw_t)v; r1 = r1 * radix + (atom_t)v;
ptr++; ptr++;
} }
r2 = r1; r2 = r1;
for (i = 0; i < hwlen; i++) for (i = 0; i < hwlen; i++)
{ {
stix_oohw_t high, low; atom_t high, low;
v = (stix_oow_t)hwp[i] * multiplier; v = (bigatom_t)hwp[i] * multiplier;
high = (stix_oohw_t)(v >> STIX_OOHW_BITS); high = (atom_t)(v >> ATOM_BITS);
low = (stix_oohw_t)(v /*& STIX_LBMASK(stix_oow_t, STIX_OOHW_BITS)*/); low = (atom_t)(v /*& STIX_LBMASK(stix_oow_t, ATOM_BITS)*/);
#if defined(oohw_add_overflow) #if defined(atom_add_overflow)
/* use oohw_add_overflow() only if it's compiler-builtin. */ /* use atom_add_overflow() only if it's compiler-builtin. */
r2 = high + oohw_add_overflow(low, r2, &low); r2 = high + atom_add_overflow(low, r2, &low);
#else #else
/* don't use the fall-back version of oohw_add_overflow() */ /* don't use the fall-back version of atom_add_overflow() */
low += r2; low += r2;
r2 = high + (low < r2); r2 = (bigatom_t)high + (low < r2);
#endif #endif
hwp[i] = low; hwp[i] = low;
} }
if (r2) hwp[hwlen++] = r2; if (r2) hwp[hwlen++] = (atom_t)r2;
} }
while (ptr < end); while (ptr < end);
} }
@ -724,12 +975,26 @@ stix_oop_t stix_strtoint (stix_t* stix, const stix_ooch_t* str, stix_oow_t len,
{ int i; { int i;
for (i = hwlen; i > 0;) for (i = hwlen; i > 0;)
{ {
printf ("%08x ", hwp[--i]); printf ("%08lx ", (unsigned long)hwp[--i]);
} }
printf ("\n"); printf ("\n");
} }
STIX_ASSERT (hwlen > 1); STIX_ASSERT (hwlen > 1);
#if defined(USE_FULL_WORD)
if (hwlen == 1)
{
w = hwp[0];
if (neg)
{
if (w <= STIX_SMINT_MAX + 1) return STIX_OOP_FROM_SMINT(-(stix_ooi_t)w);
}
else
{
if (w <= STIX_SMINT_MAX) return STIX_OOP_FROM_SMINT(w);
}
}
#else
if (hwlen == 1) return STIX_OOP_FROM_SMINT((stix_ooi_t)hwp[0] * -neg); if (hwlen == 1) return STIX_OOP_FROM_SMINT((stix_ooi_t)hwp[0] * -neg);
else if (hwlen == 2) else if (hwlen == 2)
{ {
@ -743,6 +1008,7 @@ printf ("\n");
if (w <= STIX_SMINT_MAX) return STIX_OOP_FROM_SMINT(w); if (w <= STIX_SMINT_MAX) return STIX_OOP_FROM_SMINT(w);
} }
} }
#endif
res = stix_instantiate (stix, (neg? stix->_large_negative_integer: stix->_large_positive_integer), hwp, hwlen); res = stix_instantiate (stix, (neg? stix->_large_negative_integer: stix->_large_positive_integer), hwp, hwlen);
if (hwp && hw != hwp) stix_freemem (stix, hwp); if (hwp && hw != hwp) stix_freemem (stix, hwp);

View File

@ -1036,7 +1036,7 @@ static int prim_integer_add (stix_t* stix, stix_ooi_t nargs)
#else #else
res = stix_addints (stix, rcv, arg); res = stix_addints (stix, rcv, arg);
if (!res) return -1; /* hard failure */ if (!res) return (stix->errnum == STIX_EINVAL)? 0: -1; /* soft or hard failure */
ACTIVE_STACK_POP (stix); ACTIVE_STACK_POP (stix);
ACTIVE_STACK_SETTOP (stix, res); ACTIVE_STACK_SETTOP (stix, res);
@ -1070,7 +1070,7 @@ static int prim_integer_sub (stix_t* stix, stix_ooi_t nargs)
return 0; return 0;
#else #else
res = stix_subints (stix, rcv, arg); res = stix_subints (stix, rcv, arg);
if (!res) return -1; /* hard failure */ if (!res) return (stix->errnum == STIX_EINVAL)? 0: -1; /* soft or hard failure */
ACTIVE_STACK_POP (stix); ACTIVE_STACK_POP (stix);
ACTIVE_STACK_SETTOP (stix, res); ACTIVE_STACK_SETTOP (stix, res);
@ -1080,16 +1080,18 @@ static int prim_integer_sub (stix_t* stix, stix_ooi_t nargs)
static int prim_integer_mul (stix_t* stix, stix_ooi_t nargs) static int prim_integer_mul (stix_t* stix, stix_ooi_t nargs)
{ {
stix_ooi_t tmp; stix_oop_t rcv, arg, res;
stix_oop_t rcv, arg;
STIX_ASSERT (nargs == 1); STIX_ASSERT (nargs == 1);
rcv = ACTIVE_STACK_GET(stix, stix->sp - 1); rcv = ACTIVE_STACK_GET(stix, stix->sp - 1);
arg = ACTIVE_STACK_GET(stix, stix->sp); arg = ACTIVE_STACK_GET(stix, stix->sp);
#if 0
if (STIX_OOP_IS_SMINT(rcv) && STIX_OOP_IS_SMINT(arg)) if (STIX_OOP_IS_SMINT(rcv) && STIX_OOP_IS_SMINT(arg))
{ {
stix_ooi_t tmp;
tmp = STIX_OOP_TO_SMINT(rcv) * STIX_OOP_TO_SMINT(arg); tmp = STIX_OOP_TO_SMINT(rcv) * STIX_OOP_TO_SMINT(arg);
/* TODO: check overflow. if so convert it to LargeInteger */ /* TODO: check overflow. if so convert it to LargeInteger */
@ -1100,6 +1102,14 @@ static int prim_integer_mul (stix_t* stix, stix_ooi_t nargs)
/* TODO: handle LargeInteger */ /* TODO: handle LargeInteger */
return 0; return 0;
#else
res = stix_mulints (stix, rcv, arg);
if (!res) return (stix->errnum == STIX_EINVAL)? 0: -1; /* soft or hard failure */
ACTIVE_STACK_POP (stix);
ACTIVE_STACK_SETTOP (stix, res);
return 1;
#endif
} }
static int prim_integer_eq (stix_t* stix, stix_ooi_t nargs) static int prim_integer_eq (stix_t* stix, stix_ooi_t nargs)

View File

@ -145,8 +145,13 @@ static int ignite_1 (stix_t* stix)
* Does this make sense? */ * Does this make sense? */
stix->_character = alloc_kernel_class (stix, 0, STIX_CLASS_SPEC_MAKE(0, 0, STIX_OBJ_TYPE_OOP)); stix->_character = alloc_kernel_class (stix, 0, STIX_CLASS_SPEC_MAKE(0, 0, STIX_OBJ_TYPE_OOP));
stix->_small_integer = alloc_kernel_class (stix, 0, STIX_CLASS_SPEC_MAKE(0, 0, STIX_OBJ_TYPE_OOP)); stix->_small_integer = alloc_kernel_class (stix, 0, STIX_CLASS_SPEC_MAKE(0, 0, STIX_OBJ_TYPE_OOP));
#if defined(USE_FULL_WORD)
stix->_large_positive_integer = alloc_kernel_class (stix, 0, STIX_CLASS_SPEC_MAKE(0, 1, STIX_OBJ_TYPE_WORD));
stix->_large_negative_integer = alloc_kernel_class (stix, 0, STIX_CLASS_SPEC_MAKE(0, 1, STIX_OBJ_TYPE_WORD));
#else
stix->_large_positive_integer = alloc_kernel_class (stix, 0, STIX_CLASS_SPEC_MAKE(0, 1, STIX_OBJ_TYPE_HALFWORD)); stix->_large_positive_integer = alloc_kernel_class (stix, 0, STIX_CLASS_SPEC_MAKE(0, 1, STIX_OBJ_TYPE_HALFWORD));
stix->_large_negative_integer = alloc_kernel_class (stix, 0, STIX_CLASS_SPEC_MAKE(0, 1, STIX_OBJ_TYPE_HALFWORD)); stix->_large_negative_integer = alloc_kernel_class (stix, 0, STIX_CLASS_SPEC_MAKE(0, 1, STIX_OBJ_TYPE_HALFWORD));
#endif
if (!stix->_apex || !stix->_undefined_object || if (!stix->_apex || !stix->_undefined_object ||
!stix->_object || !stix->_string || !stix->_object || !stix->_string ||

View File

@ -382,13 +382,16 @@ int main (int argc, char* argv[])
stix_vmprim_t vmprim; stix_vmprim_t vmprim;
int i; int i;
printf ("Stix 1.0.0 - max named %lu max indexed %lu max class %lu max classinst %lu smintmax %ld smintmax %ld\n", printf ("Stix 1.0.0 - max named %lu max indexed %lu max class %lu max classinst %lu oowmax %lu, smintmax %ld smintmax %ld\n",
(unsigned long int)STIX_MAX_NAMED_INSTVARS, (unsigned long int)STIX_MAX_NAMED_INSTVARS,
(unsigned long int)STIX_MAX_INDEXED_INSTVARS(STIX_MAX_NAMED_INSTVARS), (unsigned long int)STIX_MAX_INDEXED_INSTVARS(STIX_MAX_NAMED_INSTVARS),
(unsigned long int)STIX_MAX_CLASSVARS, (unsigned long int)STIX_MAX_CLASSVARS,
(unsigned long int)STIX_MAX_CLASSINSTVARS, (unsigned long int)STIX_MAX_CLASSINSTVARS,
(unsigned long int)STIX_TYPE_MAX(stix_oow_t),
(long)STIX_SMINT_MAX, (long)STIX_SMINT_MIN); (long)STIX_SMINT_MAX, (long)STIX_SMINT_MIN);
printf ("STIX_SMINT_MIN + STIX_SMINT_MIN => %ld\n", (long)(STIX_SMINT_MIN + STIX_SMINT_MIN));
printf ("STIX_SMINT_MIN - STIX_SMINT_MAX => %ld\n", (long)(STIX_SMINT_MIN - STIX_SMINT_MAX));
#if !defined(macintosh) #if !defined(macintosh)
if (argc < 2) if (argc < 2)

View File

@ -54,6 +54,10 @@
/*#define STIX_DEBUG_EXEC*/ /*#define STIX_DEBUG_EXEC*/
#define STIX_PROFILE_EXEC #define STIX_PROFILE_EXEC
#if STIX_SIZEOF_UINTMAX_T > STIX_SIZEOF_OOW_T
# define USE_FULL_WORD
#endif
#include <stdio.h> /* TODO: delete these header inclusion lines */ #include <stdio.h> /* TODO: delete these header inclusion lines */
#include <string.h> #include <string.h>
#include <assert.h> #include <assert.h>
@ -98,6 +102,25 @@
#define STIX_ALIGN(x,y) ((((x) + (y) - 1) / (y)) * (y)) #define STIX_ALIGN(x,y) ((((x) + (y) - 1) / (y)) * (y))
/* ========================================================================= */
/* BIGINT TYPES AND MACROS */
/* ========================================================================= */
#if defined(USE_FULL_WORD)
typedef stix_uintmax_t bigatom_t;
typedef stix_oow_t atom_t;
typedef stix_oop_word_t oop_atom_t;
# define ATOM_BITS STIX_OOW_BITS
# define SIZEOF_ATOM_T STIX_SIZEOF_OOW_T
#else
typedef stix_oow_t bigatom_t;
typedef stix_oohw_t atom_t;
# define ATOM_BITS STIX_OOHW_BITS
typedef stix_oop_halfword_t oop_atom_t;
# define SIZEOF_ATOM_T STIX_SIZEOF_OOHW_T
# define MAKE_WORD(hw1,hw2) ((stix_oow_t)(hw1) | (stix_oow_t)(hw2) << ATOM_BITS)
#endif
/* ========================================================================= */ /* ========================================================================= */
/* CLASS SPEC ENCODING */ /* CLASS SPEC ENCODING */
/* ========================================================================= */ /* ========================================================================= */
@ -1082,6 +1105,12 @@ stix_oop_t stix_subints (
stix_oop_t y stix_oop_t y
); );
stix_oop_t stix_mulints (
stix_t* stix,
stix_oop_t x,
stix_oop_t y
);
stix_oop_t stix_strtoint ( stix_oop_t stix_strtoint (
stix_t* stix, stix_t* stix,
const stix_ooch_t* str, const stix_ooch_t* str,

View File

@ -59,17 +59,21 @@ void stix_close (stix_t* stix)
static void fill_bigint_tables (stix_t* stix) static void fill_bigint_tables (stix_t* stix)
{ {
int radix, safe_ndigits; int radix, safe_ndigits;
stix_oow_t multiplier, ub, w; stix_oow_t ub, w;
stix_oow_t multiplier;
for (radix = 2; radix <= 36; radix++) for (radix = 2; radix <= 36; radix++)
{ {
w = 0; w = 0;
ub = (stix_oow_t)STIX_TYPE_MAX(stix_oohw_t) / radix - (radix - 1); ub = (stix_oow_t)STIX_TYPE_MAX(atom_t) / radix - (radix - 1);
multiplier = 1; multiplier = 1;
for (safe_ndigits = 0; w <= ub; safe_ndigits++) safe_ndigits = 0;
while (w <= ub)
{ {
w = w * radix + (radix - 1); w = w * radix + (radix - 1);
multiplier *= radix; multiplier *= radix;
safe_ndigits++;
} }
/* safe_ndigits contains the number of digits that never /* safe_ndigits contains the number of digits that never