1 #define PERL_NO_GET_CONTEXT
7 /* for Perl prior to v5.7.1 */
9 # define SvUOK(sv) SvIOK_UV(sv)
12 /* for Perl v5.6 (RT #63859) */
13 #ifndef croak_xs_usage
14 # define croak_xs_usage croak
18 double XS_BASE_LEN = 0;
20 MODULE = Math::BigInt::FastCalc PACKAGE = Math::BigInt::FastCalc
24 #############################################################################
25 # 2002-08-12 0.03 Tels unreleased
26 # * is_zero/is_one/is_odd/is_even/len work now (pass v1.61 tests)
27 # 2002-08-13 0.04 Tels unreleased
28 # * returns no/yes for is_foo() methods to be faster
29 # 2002-08-18 0.06alpha
30 # * added _num(), _inc() and _dec()
31 # 2002-08-25 0.06 Tels
32 # * added __strip_zeros(), _copy()
33 # 2004-08-13 0.07 Tels
34 # * added _is_two(), _is_ten(), _ten()
35 # 2007-04-02 0.08 Tels
36 # * plug leaks by creating mortals
37 # 2007-05-27 0.09 Tels
40 #define RETURN_MORTAL_INT(value) \
41 ST(0) = sv_2mortal(newSViv(value)); \
47 croak("Usage: Math::BigInt::FastCalc::BOOT(package, version, base_len, base)");
48 XS_BASE_LEN = SvIV(ST(2));
49 XS_BASE = SvNV(ST(3));
52 ##############################################################################
65 if (SvUOK(x) && SvUV(x) < XS_BASE)
67 /* shortcut for integer arguments */
68 av_push (av, newSVuv( SvUV(x) ));
72 /* split the input (as string) into XS_BASE_LEN long parts */
74 [ reverse(unpack("a" . ($il % $BASE_LEN+1)
75 . ("a$BASE_LEN" x ($il / $BASE_LEN)), $_[1])) ];
77 cur = SvPV(x, len); /* convert to string & store length */
78 cur += len; /* doing "cur = SvEND(x)" does not work! */
79 # process the string from the back
82 /* use either BASE_LEN or the amount of remaining digits */
83 part_len = (STRLEN) XS_BASE_LEN;
88 /* processed so many digits */
91 /* printf ("part '%s' (part_len: %i, len: %i, BASE_LEN: %i)\n", cur, part_len, len, XS_BASE_LEN); */
94 av_push (av, newSVpvn(cur, part_len) );
98 RETVAL = newRV_noinc((SV *)av);
102 ##############################################################################
114 a = (AV*)SvRV(x); /* ref to aray, don't check ref */
115 elems = av_len(a); /* number of elems in array */
116 a2 = (AV*)sv_2mortal((SV*)newAV());
117 av_extend (a2, elems); /* pre-padd */
120 /* av_store( a2, elems, newSVsv( (SV*)*av_fetch(a, elems, 0) ) ); */
122 /* looking and trying to preserve IV is actually slower when copying */
123 /* temp = (SV*)*av_fetch(a, elems, 0);
126 av_store( a2, elems, newSViv( SvIV( (SV*)*av_fetch(a, elems, 0) )));
130 av_store( a2, elems, newSVnv( SvNV( (SV*)*av_fetch(a, elems, 0) )));
133 av_store( a2, elems, newSVnv( SvNV( (SV*)*av_fetch(a, elems, 0) )));
136 ST(0) = sv_2mortal( newRV_inc((SV*) a2) );
138 ##############################################################################
139 # __strip_zeros (also check for empty arrays from div)
151 a = (AV*)SvRV(x); /* ref to aray, don't check ref */
152 elems = av_len(a); /* number of elems in array */
153 ST(0) = x; /* we return x */
156 av_push (a, newSViv(0)); /* correct empty arrays */
161 XSRETURN(1); /* nothing to do since only one elem */
166 temp = *av_fetch(a, index, 0); /* fetch ptr to current element */
175 index = elems - index;
183 ##############################################################################
184 # decrement (subtract one)
197 a = (AV*)SvRV(x); /* ref to aray, don't check ref */
198 elems = av_len(a); /* number of elems in array */
199 ST(0) = x; /* we return x */
203 while (index <= elems)
205 temp = *av_fetch(a, index, 0); /* fetch ptr to current element */
206 sv_setnv (temp, SvNV(temp)-1); /* decrement */
209 break; /* early out */
211 sv_setnv (temp, MAX); /* overflow, so set this to $MAX */
214 /* do have more than one element? */
215 /* (more than one because [0] should be kept as single-element) */
218 temp = *av_fetch(a, elems, 0); /* fetch last element */
219 if (SvIV(temp) == 0) /* did last elem overflow? */
221 av_pop(a); /* yes, so shrink array */
222 /* aka remove leading zeros */
225 XSRETURN(1); /* return x */
227 ##############################################################################
228 # increment (add one)
241 a = (AV*)SvRV(x); /* ref to aray, don't check ref */
242 elems = av_len(a); /* number of elems in array */
243 ST(0) = x; /* we return x */
247 while (index <= elems)
249 temp = *av_fetch(a, index, 0); /* fetch ptr to current element */
250 sv_setnv (temp, SvNV(temp)+1);
251 if (SvNV(temp) < BASE)
253 XSRETURN(1); /* return (early out) */
255 sv_setiv (temp, 0); /* overflow, so set this elem to 0 */
258 temp = *av_fetch(a, elems, 0); /* fetch last element */
259 if (SvIV(temp) == 0) /* did last elem overflow? */
261 av_push(a, newSViv(1)); /* yes, so extend array by 1 */
263 XSRETURN(1); /* return x */
265 ##############################################################################
276 av_push (av, newSViv( ix ));
277 RETVAL = newRV_noinc((SV *)av);
281 ##############################################################################
293 a = (AV*)SvRV(x); /* ref to aray, don't check ref */
294 temp = *av_fetch(a, 0, 0); /* fetch first element */
295 ST(0) = sv_2mortal(boolSV((SvIV(temp) & 1) == ix));
297 ##############################################################################
310 a = (AV*)SvRV(x); /* ref to aray, don't check ref */
313 ST(0) = &PL_sv_no; /* len != 1, can't be '0' */
317 SV *const temp = *av_fetch(a, 0, 0); /* fetch first element */
318 ST(0) = boolSV(SvIV(temp) == ix);
322 ##############################################################################
334 a = (AV*)SvRV(x); /* ref to aray, don't check ref */
335 elems = av_len(a); /* number of elems in array */
336 temp = *av_fetch(a, elems, 0); /* fetch last element */
337 SvPV(temp, len); /* convert to string & store length */
338 len += (IV) XS_BASE_LEN * elems;
339 ST(0) = sv_2mortal(newSViv(len));
341 ##############################################################################
344 _acmp(class, cx, cy);
350 SSize_t elemsx, elemsy, diff;
359 array_x = (AV*)SvRV(cx); /* ref to aray, don't check ref */
360 array_y = (AV*)SvRV(cy); /* ref to aray, don't check ref */
361 elemsx = av_len(array_x);
362 elemsy = av_len(array_y);
363 diff = elemsx - elemsy; /* difference */
367 RETURN_MORTAL_INT(1); /* len differs: X > Y */
371 RETURN_MORTAL_INT(-1); /* len differs: X < Y */
373 /* both have same number of elements, so check length of last element
374 and see if it differs */
375 tempx = *av_fetch(array_x, elemsx, 0); /* fetch last element */
376 tempy = *av_fetch(array_y, elemsx, 0); /* fetch last element */
377 SvPV(tempx, lenx); /* convert to string & store length */
378 SvPV(tempy, leny); /* convert to string & store length */
379 diff_str = (SSize_t)lenx - (SSize_t)leny;
382 RETURN_MORTAL_INT(1); /* same len, but first elems differs in len */
386 RETURN_MORTAL_INT(-1); /* same len, but first elems differs in len */
388 /* same number of digits, so need to make a full compare */
392 tempx = *av_fetch(array_x, elemsx, 0); /* fetch curr x element */
393 tempy = *av_fetch(array_y, elemsx, 0); /* fetch curr y element */
394 diff_nv = SvNV(tempx) - SvNV(tempy);
403 RETURN_MORTAL_INT(1);
407 RETURN_MORTAL_INT(-1);
409 ST(0) = sv_2mortal(newSViv(0)); /* X and Y are equal */