source: git/libpolys/coeffs/rmodulo2m.cc @ 73a9ffb

spielwiese
Last change on this file since 73a9ffb was 73a9ffb, checked in by Frank Seelisch <seelisch@…>, 13 years ago
made sure that ch is properly set everywhere, and ch >= 0; more ASSUMEs
  • Property mode set to 100644
File size: 17.2 KB
Line 
1/****************************************
2*  Computer Algebra System SINGULAR     *
3****************************************/
4/* $Id$ */
5/*
6* ABSTRACT: numbers modulo 2^m
7*/
8
9#include "config.h"
10#include <misc/auxiliary.h>
11
12#ifdef HAVE_RINGS
13
14#include <misc/mylimits.h>
15#include <coeffs/coeffs.h>
16#include <reporter/reporter.h>
17#include <omalloc/omalloc.h>
18#include <coeffs/numbers.h>
19#include <coeffs/longrat.h>
20#include <coeffs/mpr_complex.h>
21#include <coeffs/rmodulo2m.h>
22#include "si_gmp.h"
23
24#include <string.h>
25
26/// Our Type!
27static const n_coeffType ID = n_Z2m;
28
29extern omBin gmp_nrz_bin; /* init in rintegers*/
30
31void    nr2mCoeffWrite  (const coeffs r)
32{
33  Print("//   Z/2^%lu\n", r->modExponent);
34}
35
36BOOLEAN nr2mCoeffIsEqual(const coeffs r, n_coeffType n, void * p)
37{
38  if (n==n_Z2m)
39  {
40    int m=(int)(long)(p);
41    unsigned long mm=r->mod2mMask;
42    if ((mm>>m)==1L) return TRUE;
43  }
44  return FALSE;
45}
46/* for initializing function pointers */
47BOOLEAN nr2mInitChar (coeffs r, void* p)
48{
49  assume( getCoeffType(r) == ID );
50  nr2mInitExp((int)(long)(p), r);
51  r->cfKillChar    = ndKillChar; /* dummy*/
52  r->nCoeffIsEqual = nr2mCoeffIsEqual;
53
54  r->ringtype = 1;
55 
56  /* next cast may yield an overflow as mod2mMask is an unsigned long */
57  r->ch = (int)r->mod2mMask + 1;
58
59  r->cfInit        = nr2mInit;
60  r->cfCopy        = ndCopy;
61  r->cfInt         = nr2mInt;
62  r->cfAdd         = nr2mAdd;
63  r->cfSub         = nr2mSub;
64  r->cfMult        = nr2mMult;
65  r->cfDiv         = nr2mDiv;
66  r->cfIntDiv      = nr2mIntDiv;
67  r->cfIntMod      = nr2mMod;
68  r->cfExactDiv    = nr2mDiv;
69  r->cfNeg         = nr2mNeg;
70  r->cfInvers      = nr2mInvers;
71  r->cfDivBy       = nr2mDivBy;
72  r->cfDivComp     = nr2mDivComp;
73  r->cfGreater     = nr2mGreater;
74  r->cfEqual       = nr2mEqual;
75  r->cfIsZero      = nr2mIsZero;
76  r->cfIsOne       = nr2mIsOne;
77  r->cfIsMOne      = nr2mIsMOne;
78  r->cfGreaterZero = nr2mGreaterZero;
79  r->cfWrite       = nr2mWrite;
80  r->cfRead        = nr2mRead;
81  r->cfPower       = nr2mPower;
82  r->cfSetMap      = nr2mSetMap;
83  r->cfNormalize   = ndNormalize;
84  r->cfLcm         = nr2mLcm;
85  r->cfGcd         = nr2mGcd;
86  r->cfIsUnit      = nr2mIsUnit;
87  r->cfGetUnit     = nr2mGetUnit;
88  r->cfExtGcd      = nr2mExtGcd;
89  r->cfName        = ndName;
90  r->cfCoeffWrite  = nr2mCoeffWrite;
91#ifdef LDEBUG
92  r->cfDBTest      = nr2mDBTest;
93#endif
94  r->has_simple_Alloc=TRUE;
95  return FALSE;
96}
97
98/*
99 * Multiply two numbers
100 */
101number nr2mMult(number a, number b, const coeffs r)
102{
103  if (((NATNUMBER)a == 0) || ((NATNUMBER)b == 0))
104    return (number)0;
105  else
106    return nr2mMultM(a, b, r);
107}
108
109/*
110 * Give the smallest k, such that a * x = k = b * y has a solution
111 */
112number nr2mLcm(number a, number b, const coeffs r)
113{
114  NATNUMBER res = 0;
115  if ((NATNUMBER)a == 0) a = (number) 1;
116  if ((NATNUMBER)b == 0) b = (number) 1;
117  while ((NATNUMBER)a % 2 == 0)
118  {
119    a = (number)((NATNUMBER)a / 2);
120    if ((NATNUMBER)b % 2 == 0) b = (number)((NATNUMBER)b / 2);
121    res++;
122  }
123  while ((NATNUMBER)b % 2 == 0)
124  {
125    b = (number)((NATNUMBER)b / 2);
126    res++;
127  }
128  return (number)(1L << res);  // (2**res)
129}
130
131/*
132 * Give the largest k, such that a = x * k, b = y * k has
133 * a solution.
134 */
135number nr2mGcd(number a, number b, const coeffs r)
136{
137  NATNUMBER res = 0;
138  if ((NATNUMBER)a == 0 && (NATNUMBER)b == 0) return (number)1;
139  while ((NATNUMBER)a % 2 == 0 && (NATNUMBER)b % 2 == 0)
140  {
141    a = (number)((NATNUMBER)a / 2);
142    b = (number)((NATNUMBER)b / 2);
143    res++;
144  }
145//  if ((NATNUMBER)b % 2 == 0)
146//  {
147//    return (number)((1L << res)); // * (NATNUMBER) a);  // (2**res)*a    a is a unit
148//  }
149//  else
150//  {
151    return (number)((1L << res)); // * (NATNUMBER) b);  // (2**res)*b    b is a unit
152//  }
153}
154
155/*
156 * Give the largest k, such that a = x * k, b = y * k has
157 * a solution.
158 */
159number nr2mExtGcd(number a, number b, number *s, number *t, const coeffs r)
160{
161  NATNUMBER res = 0;
162  if ((NATNUMBER)a == 0 && (NATNUMBER)b == 0) return (number)1;
163  while ((NATNUMBER)a % 2 == 0 && (NATNUMBER)b % 2 == 0)
164  {
165    a = (number)((NATNUMBER)a / 2);
166    b = (number)((NATNUMBER)b / 2);
167    res++;
168  }
169  if ((NATNUMBER)b % 2 == 0)
170  {
171    *t = NULL;
172    *s = nr2mInvers(a,r);
173    return (number)((1L << res)); // * (NATNUMBER) a);  // (2**res)*a    a is a unit
174  }
175  else
176  {
177    *s = NULL;
178    *t = nr2mInvers(b,r);
179    return (number)((1L << res)); // * (NATNUMBER) b);  // (2**res)*b    b is a unit
180  }
181}
182
183void nr2mPower(number a, int i, number * result, const coeffs r)
184{
185  if (i == 0)
186  {
187    *(NATNUMBER *)result = 1;
188  }
189  else if (i == 1)
190  {
191    *result = a;
192  }
193  else
194  {
195    nr2mPower(a, i-1, result, r);
196    *result = nr2mMultM(a, *result, r);
197  }
198}
199
200/*
201 * create a number from int
202 */
203number nr2mInit(int i, const coeffs r)
204{
205  if (i == 0) return (number)(NATNUMBER)i;
206
207  long ii = i;
208  NATNUMBER j = (NATNUMBER)1;
209  if (ii < 0) { j = r->mod2mMask; ii = -ii; }
210  NATNUMBER k = (NATNUMBER)ii;
211  k = k & r->mod2mMask;
212  /* now we have: i = j * k mod 2^m */
213  return (number)nr2mMult((number)j, (number)k, r);
214}
215
216/*
217 * convert a number to an int in ]-k/2 .. k/2],
218 * where k = 2^m; i.e., an int in ]-2^(m-1) .. 2^(m-1)];
219 * note that the code computes a long which will then
220 * automatically casted to int
221 */
222int nr2mInt(number &n, const coeffs r)
223{
224  NATNUMBER nn = (unsigned long)(NATNUMBER)n & r->mod2mMask;
225  unsigned long l = r->mod2mMask >> 1; l++; /* now: l = 2^(m-1) */
226  if ((NATNUMBER)nn > l)
227    return (int)((NATNUMBER)nn - r->mod2mMask - 1);
228  else
229    return (int)((NATNUMBER)nn);
230}
231
232number nr2mAdd(number a, number b, const coeffs r)
233{
234  return nr2mAddM(a, b, r);
235}
236
237number nr2mSub(number a, number b, const coeffs r)
238{
239  return nr2mSubM(a, b, r);
240}
241
242BOOLEAN nr2mIsUnit(number a, const coeffs r)
243{
244  return ((NATNUMBER)a % 2 == 1);
245}
246
247number nr2mGetUnit(number k, const coeffs r)
248{
249  if (k == NULL) return (number)1;
250  NATNUMBER erg = (NATNUMBER)k;
251  while (erg % 2 == 0) erg = erg / 2;
252  return (number)erg;
253}
254
255BOOLEAN nr2mIsZero(number a, const coeffs r)
256{
257  return 0 == (NATNUMBER)a;
258}
259
260BOOLEAN nr2mIsOne(number a, const coeffs r)
261{
262  return 1 == (NATNUMBER)a;
263}
264
265BOOLEAN nr2mIsMOne(number a, const coeffs r)
266{
267  return (r->mod2mMask  == (NATNUMBER)a);
268}
269
270BOOLEAN nr2mEqual(number a, number b, const coeffs r)
271{
272  return (a == b);
273}
274
275BOOLEAN nr2mGreater(number a, number b, const coeffs r)
276{
277  return nr2mDivBy(a, b,r);
278}
279
280/* Is a divisible by b? There are two cases:
281   1) a = 0 mod 2^m; then TRUE iff b = 0 or b is a power of 2
282   2) a, b <> 0; then TRUE iff b/gcd(a, b) is a unit mod 2^m
283   TRUE iff b(gcd(a, b) is a unit */
284BOOLEAN nr2mDivBy (number a, number b, const coeffs r)
285{
286  if (a == NULL)
287  {
288    NATNUMBER c = r->mod2mMask + 1;
289    if (c != 0) /* i.e., if no overflow */
290      return (c % (NATNUMBER)b) == 0;
291    else
292    {
293      /* overflow: we need to check whether b
294         is zero or a power of 2: */
295      c = (NATNUMBER)b;
296      while (c != 0)
297      {
298        if ((c % 2) != 0) return FALSE;
299        c = c >> 1;
300      }
301      return TRUE;
302    }
303  }
304  else
305  {
306    number n = nr2mGcd(a, b, r);
307    n = nr2mDiv(b, n, r);
308    return nr2mIsUnit(n, r);
309  }
310}
311
312int nr2mDivComp(number as, number bs, const coeffs r)
313{
314  NATNUMBER a = (NATNUMBER)as;
315  NATNUMBER b = (NATNUMBER)bs;
316  assume(a != 0 && b != 0);
317  while (a % 2 == 0 && b % 2 == 0)
318  {
319    a = a / 2;
320    b = b / 2;
321  }
322  if (a % 2 == 0)
323  {
324    return -1;
325  }
326  else
327  {
328    if (b % 2 == 1)
329    {
330      return 2;
331    }
332    else
333    {
334      return 1;
335    }
336  }
337}
338
339/* TRUE iff 0 < k <= 2^m / 2 */
340BOOLEAN nr2mGreaterZero(number k, const coeffs r)
341{
342  if ((NATNUMBER)k == 0) return FALSE;
343  if ((NATNUMBER)k > ((r->mod2mMask >> 1) + 1)) return FALSE;
344  return TRUE;
345}
346
347/* assumes that 'a' is odd, i.e., a unit in Z/2^m, and computes
348   the extended gcd of 'a' and 2^m, in order to find some 's'
349   and 't' such that a * s + 2^m * t = gcd(a, 2^m) = 1;
350   this code will always find a positive 's' */
351void specialXGCD(unsigned long& s, unsigned long a, const coeffs r)
352{
353  int_number u = (int_number)omAlloc(sizeof(mpz_t));
354  mpz_init_set_ui(u, a);
355  int_number u0 = (int_number)omAlloc(sizeof(mpz_t));
356  mpz_init(u0);
357  int_number u1 = (int_number)omAlloc(sizeof(mpz_t));
358  mpz_init_set_ui(u1, 1);
359  int_number u2 = (int_number)omAlloc(sizeof(mpz_t));
360  mpz_init(u2);
361  int_number v = (int_number)omAlloc(sizeof(mpz_t));
362  mpz_init_set_ui(v, r->mod2mMask);
363  mpz_add_ui(v, v, 1); /* now: v = 2^m */
364  int_number v0 = (int_number)omAlloc(sizeof(mpz_t));
365  mpz_init(v0);
366  int_number v1 = (int_number)omAlloc(sizeof(mpz_t));
367  mpz_init(v1);
368  int_number v2 = (int_number)omAlloc(sizeof(mpz_t));
369  mpz_init_set_ui(v2, 1);
370  int_number q = (int_number)omAlloc(sizeof(mpz_t));
371  mpz_init(q);
372  int_number rr = (int_number)omAlloc(sizeof(mpz_t));
373  mpz_init(rr);
374
375  while (mpz_cmp_ui(v, 0) != 0) /* i.e., while v != 0 */
376  {
377    mpz_div(q, u, v);
378    mpz_mod(rr, u, v);
379    mpz_set(u, v);
380    mpz_set(v, rr);
381    mpz_set(u0, u2);
382    mpz_set(v0, v2);
383    mpz_mul(u2, u2, q); mpz_sub(u2, u1, u2); /* u2 = u1 - q * u2 */
384    mpz_mul(v2, v2, q); mpz_sub(v2, v1, v2); /* v2 = v1 - q * v2 */
385    mpz_set(u1, u0);
386    mpz_set(v1, v0);
387  }
388
389  while (mpz_cmp_ui(u1, 0) < 0) /* i.e., while u1 < 0 */
390  {
391    /* we add 2^m = (2^m - 1) + 1 to u1: */
392    mpz_add_ui(u1, u1, r->mod2mMask);
393    mpz_add_ui(u1, u1, 1);
394  }
395  s = mpz_get_ui(u1); /* now: 0 <= s <= 2^m - 1 */
396
397  mpz_clear(u);  omFree((ADDRESS)u);
398  mpz_clear(u0); omFree((ADDRESS)u0);
399  mpz_clear(u1); omFree((ADDRESS)u1);
400  mpz_clear(u2); omFree((ADDRESS)u2);
401  mpz_clear(v);  omFree((ADDRESS)v);
402  mpz_clear(v0); omFree((ADDRESS)v0);
403  mpz_clear(v1); omFree((ADDRESS)v1);
404  mpz_clear(v2); omFree((ADDRESS)v2);
405  mpz_clear(q); omFree((ADDRESS)q);
406  mpz_clear(rr); omFree((ADDRESS)rr);
407}
408
409NATNUMBER InvMod(NATNUMBER a, const coeffs r)
410{
411  assume((NATNUMBER)a % 2 != 0);
412  unsigned long s;
413  specialXGCD(s, a, r);
414  return s;
415}
416//#endif
417
418inline number nr2mInversM(number c, const coeffs r)
419{
420  assume((NATNUMBER)c % 2 != 0);
421  // Table !!!
422  NATNUMBER inv;
423  inv = InvMod((NATNUMBER)c,r);
424  return (number)inv;
425}
426
427number nr2mDiv(number a, number b, const coeffs r)
428{
429  if ((NATNUMBER)a == 0) return (number)0;
430  else if ((NATNUMBER)b % 2 == 0)
431  {
432    if ((NATNUMBER)b != 0)
433    {
434      while (((NATNUMBER)b % 2 == 0) && ((NATNUMBER)a % 2 == 0))
435      {
436        a = (number)((NATNUMBER)a / 2);
437        b = (number)((NATNUMBER)b / 2);
438      }
439    }
440    if ((NATNUMBER)b % 2 == 0)
441    {
442      WerrorS("Division not possible, even by cancelling zero divisors.");
443      WerrorS("Result is integer division without remainder.");
444      return (number) ((NATNUMBER) a / (NATNUMBER) b);
445    }
446  }
447  return (number)nr2mMult(a, nr2mInversM(b,r),r);
448}
449
450number nr2mMod(number a, number b, const coeffs r)
451{
452  /*
453    We need to return the number rr which is uniquely determined by the
454    following two properties:
455      (1) 0 <= rr < |b| (with respect to '<' and '<=' performed in Z x Z)
456      (2) There exists some k in the integers Z such that a = k * b + rr.
457    Consider g := gcd(2^m, |b|). Note that then |b|/g is a unit in Z/2^m.
458    Now, there are three cases:
459      (a) g = 1
460          Then |b| is a unit in Z/2^m, i.e. |b| (and also b) divides a.
461          Thus rr = 0.
462      (b) g <> 1 and g divides a
463          Then a = (a/g) * (|b|/g)^(-1) * b (up to sign), i.e. again rr = 0.
464      (c) g <> 1 and g does not divide a
465          Let's denote the division with remainder of a by g as follows:
466          a = s * g + t. Then t = a - s * g = a - s * (|b|/g)^(-1) * |b|
467          fulfills (1) and (2), i.e. rr := t is the correct result. Hence
468          in this third case, rr is the remainder of division of a by g in Z.
469    This algorithm is the same as for the case Z/n, except that we may
470    compute the gcd of |b| and 2^m "by hand": We just extract the highest
471    power of 2 (<= 2^m) that is contained in b.
472  */
473  assume((NATNUMBER)b != 0);
474  NATNUMBER g = 1;
475  NATNUMBER b_div = (NATNUMBER)b;
476  if (b_div < 0) b_div = -b_div; // b_div now represents |b|
477  NATNUMBER rr = 0;
478  while ((g < r->mod2mMask ) && (b_div > 0) && (b_div % 2 == 0))
479  {
480    b_div = b_div >> 1;
481    g = g << 1;
482  } // g is now the gcd of 2^m and |b|
483
484  if (g != 1) rr = (NATNUMBER)a % g;
485  return (number)rr;
486}
487
488number nr2mIntDiv(number a, number b, const coeffs r)
489{
490  if ((NATNUMBER)a == 0)
491  {
492    if ((NATNUMBER)b == 0)
493      return (number)1;
494    if ((NATNUMBER)b == 1)
495      return (number)0;
496    NATNUMBER c = r->mod2mMask + 1;
497    if (c != 0) /* i.e., if no overflow */
498      return (number)(c / (NATNUMBER)b);
499    else
500    {
501      /* overflow: c = 2^32 resp. 2^64, depending on platform */
502      int_number cc = (int_number)omAlloc(sizeof(mpz_t));
503      mpz_init_set_ui(cc, r->mod2mMask); mpz_add_ui(cc, cc, 1);
504      mpz_div_ui(cc, cc, (unsigned long)(NATNUMBER)b);
505      unsigned long s = mpz_get_ui(cc);
506      mpz_clear(cc); omFree((ADDRESS)cc);
507      return (number)(NATNUMBER)s;
508    }
509  }
510  else
511  {
512    if ((NATNUMBER)b == 0)
513      return (number)0;
514    return (number)((NATNUMBER) a / (NATNUMBER) b);
515  }
516}
517
518number nr2mInvers(number c, const coeffs r)
519{
520  if ((NATNUMBER)c % 2 == 0)
521  {
522    WerrorS("division by zero divisor");
523    return (number)0;
524  }
525  return nr2mInversM(c, r);
526}
527
528number nr2mNeg(number c, const coeffs r)
529{
530  if ((NATNUMBER)c == 0) return c;
531  return nr2mNegM(c, r);
532}
533
534number nr2mMapMachineInt(number from, const coeffs src, const coeffs dst)
535{
536  NATNUMBER i = ((NATNUMBER)from) % dst->mod2mMask ;
537  return (number)i;
538}
539
540number nr2mMapZp(number from, const coeffs src, const coeffs dst)
541{
542  NATNUMBER j = (NATNUMBER)1;
543  long ii = (long)from;
544  if (ii < 0) { j = dst->mod2mMask; ii = -ii; }
545  NATNUMBER i = (NATNUMBER)ii;
546  i = i & dst->mod2mMask;
547  /* now we have: from = j * i mod 2^m */
548  return (number)nr2mMult((number)i, (number)j, dst);
549}
550
551number nr2mMapQ(number from, const coeffs src, const coeffs dst)
552{
553  int_number erg = (int_number)omAllocBin(gmp_nrz_bin);
554  mpz_init(erg);
555  int_number k = (int_number)omAlloc(sizeof(mpz_t));
556  mpz_init_set_ui(k, dst->mod2mMask);
557
558  nlGMP(from, (number)erg, dst);
559  mpz_and(erg, erg, k);
560  number res = (number)mpz_get_ui(erg);
561
562  mpz_clear(erg); omFree((ADDRESS)erg);
563  mpz_clear(k);   omFree((ADDRESS)k);
564
565  return (number)res;
566}
567
568number nr2mMapGMP(number from, const coeffs src, const coeffs dst)
569{
570  int_number erg = (int_number)omAllocBin(gmp_nrz_bin);
571  mpz_init(erg);
572  int_number k = (int_number)omAlloc(sizeof(mpz_t));
573  mpz_init_set_ui(k, dst->mod2mMask);
574
575  mpz_and(erg, (int_number)from, k);
576  number res = (number) mpz_get_ui(erg);
577
578  mpz_clear(erg); omFree((ADDRESS)erg);
579  mpz_clear(k);   omFree((ADDRESS)k);
580
581  return (number)res;
582}
583
584nMapFunc nr2mSetMap(const coeffs src, const coeffs dst)
585{
586  if (nCoeff_is_Ring_2toM(src)
587     && (src->mod2mMask == dst->mod2mMask))
588  {
589    return ndCopyMap;
590  }
591  if (nCoeff_is_Ring_2toM(src)
592     && (src->mod2mMask < dst->mod2mMask))
593  { /* i.e. map an integer mod 2^s into Z mod 2^t, where t < s */
594    return nr2mMapMachineInt;
595  }
596  if (nCoeff_is_Ring_2toM(src)
597     && (src->mod2mMask > dst->mod2mMask))
598  { /* i.e. map an integer mod 2^s into Z mod 2^t, where t > s */
599    // to be done
600  }
601  if (nCoeff_is_Ring_Z(src))
602  {
603    return nr2mMapGMP;
604  }
605  if (nCoeff_is_Q(src))
606  {
607    return nr2mMapQ;
608  }
609  if (nCoeff_is_Zp(src) && (src->ch == 2))
610  {
611    return nr2mMapZp;
612  }
613  if (nCoeff_is_Ring_PtoM(src) || nCoeff_is_Ring_ModN(src))
614  {
615    // Computing the n of Z/n
616    int_number modul = (int_number)omAllocBin(gmp_nrz_bin);
617    mpz_init_set(modul, src->modNumber);
618    int_number twoToTheK = (int_number)omAllocBin(gmp_nrz_bin);
619    mpz_init_set_ui(twoToTheK, src->mod2mMask);
620    mpz_add_ui(twoToTheK, twoToTheK, 1);
621    if (mpz_divisible_p(modul, twoToTheK))
622    {
623      mpz_clear(modul);     omFree((void *)modul);
624      mpz_clear(twoToTheK); omFree((void *)twoToTheK);
625      return nr2mMapGMP;
626    }
627    mpz_clear(modul);     omFree((void *) modul);
628    mpz_clear(twoToTheK); omFree((void *)twoToTheK);
629  }
630  return NULL;      // default
631}
632
633/*
634 * set the exponent
635 */
636
637void nr2mSetExp(int m, coeffs r)
638{
639  if (m > 1)
640  {
641    /* we want mod2mMask to be the bit pattern
642       '111..1' consisting of m one's: */
643    r->mod2mMask = 1;
644    for (int i = 1; i < m; i++) r->mod2mMask = (r->mod2mMask << 1) + 1;
645  }
646  else
647  {
648    /* code unexpectedly called with m = 1; we continue with m = 2: */
649    r->mod2mMask = 3; /* i.e., '11' in binary representation */
650  }
651}
652
653void nr2mInitExp(int m, coeffs r)
654{
655  nr2mSetExp(m, r);
656  if (m < 2)
657    WarnS("nr2mInitExp unexpectedly called with m = 1 (we continue with Z/2^2");
658}
659
660#ifdef LDEBUG
661BOOLEAN nr2mDBTest (number a, const char *f, const int l, const coeffs r)
662{
663  if ((NATNUMBER)a < 0) return FALSE;
664  if (((NATNUMBER)a & r->mod2mMask) != (NATNUMBER)a) return FALSE;
665  return TRUE;
666}
667#endif
668
669void nr2mWrite (number &a, const coeffs r)
670{
671  int i = nr2mInt(a, r);
672  StringAppend("%d", i);
673}
674
675static const char* nr2mEati(const char *s, int *i, const coeffs r)
676{
677
678  if (((*s) >= '0') && ((*s) <= '9'))
679  {
680    (*i) = 0;
681    do
682    {
683      (*i) *= 10;
684      (*i) += *s++ - '0';
685      if ((*i) >= (MAX_INT_VAL / 10)) (*i) = (*i) & r->mod2mMask;
686    }
687    while (((*s) >= '0') && ((*s) <= '9'));
688    (*i) = (*i) & r->mod2mMask;
689  }
690  else (*i) = 1;
691  return s;
692}
693
694const char * nr2mRead (const char *s, number *a, const coeffs r)
695{
696  int z;
697  int n=1;
698
699  s = nr2mEati(s, &z,r);
700  if ((*s) == '/')
701  {
702    s++;
703    s = nr2mEati(s, &n,r);
704  }
705  if (n == 1)
706    *a = (number)z;
707  else
708      *a = nr2mDiv((number)z,(number)n,r);
709  return s;
710}
711#endif
712/* #ifdef HAVE_RINGS */
Note: See TracBrowser for help on using the repository browser.