source: git/libpolys/coeffs/rmodulo2m.cc @ 1950d7

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