source: git/libpolys/coeffs/numbers.cc @ 81384b

spielwiese
Last change on this file since 81384b was 21b9ef, checked in by Oleksandr Motsak <motsak@…>, 11 years ago
Added n_Print for printing numbers during debugging
  • Property mode set to 100644
File size: 13.1 KB
Line 
1/*****************************************
2*  Computer Algebra System SINGULAR      *
3*****************************************/
4
5/*
6* ABSTRACT: interface to coefficient aritmetics
7*/
8
9#include <string.h>
10#include <stdlib.h>
11
12#include "config.h"
13#include <misc/auxiliary.h>
14
15#ifdef HAVE_FACTORY
16#include <factory/factory.h>
17#endif
18
19
20#include "coeffs.h"
21#include <coeffs/numbers.h>
22
23#include <reporter/reporter.h>
24#include <omalloc/omalloc.h>
25#include <coeffs/numbers.h>
26#include <coeffs/longrat.h>
27#include <coeffs/modulop.h>
28#include <coeffs/gnumpfl.h>
29#include <coeffs/gnumpc.h>
30#include <coeffs/ffields.h>
31#include <coeffs/shortfl.h>
32
33#ifdef HAVE_RINGS
34#include <coeffs/rmodulo2m.h>
35#include <coeffs/rmodulon.h>
36#include <coeffs/rintegers.h>
37#endif
38
39#ifdef HAVE_POLYEXTENSIONS
40#include <polys/ext_fields/algext.h>
41#include <polys/ext_fields/transext.h>
42#endif
43
44
45
46//static int characteristic = 0;
47//extern int IsPrime(int p);
48
49n_Procs_s *cf_root=NULL;
50
51void   nNew(number* d) { *d=NULL; }
52void   ndDelete(number* d, const coeffs) { *d=NULL; }
53void   ndInpMult(number &a, number b, const coeffs r)
54{
55  number n=n_Mult(a,b,r);
56  n_Delete(&a,r);
57  a=n;
58}
59void ndInpAdd(number &a, number b, const coeffs r)
60{
61  number n=n_Add(a,b,r);
62  n_Delete(&a,r);
63  a=n;
64}
65
66#ifdef LDEBUG
67void   nDBDummy1(number* d,char *, int) { *d=NULL; }
68BOOLEAN ndDBTest(number, const char *, const int, const coeffs)
69{
70  return TRUE;
71}
72#endif
73
74number ndFarey(number,number,const coeffs r)
75{
76  Werror("farey not implemented for (c=%d)",getCoeffType(r));
77  return NULL;
78}
79number ndChineseRemainder(number *,number *,int,BOOLEAN,const coeffs r)
80{
81  Werror("ChineseRemainder not implemented for (c=%d)",getCoeffType(r));
82  return n_Init(0,r); 
83}
84
85static int ndParDeg(number n, const coeffs r)
86{
87  return (-n_IsZero(n,r));
88}
89
90static number ndParameter(const int, const coeffs r)
91{
92  Werror("ndParameter: n_Parameter is not implemented/relevant for (coeff_type = %d)",getCoeffType(r));
93  return NULL;
94}
95
96BOOLEAN n_IsZeroDivisor( number a, const coeffs r)
97{
98  int c = n_GetChar(r);
99  BOOLEAN ret = n_IsZero(a, r);
100  if( (c != 0) && !ret )
101  {
102    number ch = n_Init( c, r ); 
103    number g = n_Gcd( ch, a, r );
104    ret = !n_IsOne (g, r);
105    n_Delete(&ch, r);
106    n_Delete(&g, r);
107  }
108  return ret; 
109}
110
111void   ndNormalize(number&, const coeffs) { }
112
113char * ndName(number, const coeffs) { return NULL; }
114
115number ndReturn0(number, const coeffs r) { return n_Init(0,r); }
116
117number ndGcd(number, number, const coeffs r) { return n_Init(1,r); }
118
119number ndIntMod(number, number, const coeffs r) { return n_Init(0,r); }
120
121number ndGetDenom(number &, const coeffs r) { return n_Init(1,r); }
122number ndGetNumerator(number &a,const coeffs r) { return n_Copy(a,r); }
123
124int ndSize(number a, const coeffs r) { return (int)n_IsZero(a,r)==FALSE; }
125
126void ndClearContent(ICoeffsEnumerator& numberCollectionEnumerator, number& c, const coeffs r)
127{
128  assume(r != NULL);
129
130  // no fractions
131  assume(!(  nCoeff_is_Q(r) ));
132  // all coeffs are given by integers!!!
133
134  numberCollectionEnumerator.Reset();
135
136  if( !numberCollectionEnumerator.MoveNext() ) // empty zero polynomial?
137  {
138    c = n_Init(1, r);
139    return;
140  } 
141
142  number &curr = numberCollectionEnumerator.Current();
143 
144#ifdef HAVE_RINGS
145  /// TODO: move to a separate implementation
146  if (nCoeff_is_Ring(r))
147  {
148    if (nCoeff_has_Units(r))
149    {
150      c = n_GetUnit(curr, r);
151     
152      if (!n_IsOne(c, r))
153      {
154        number inv = n_Invers(c, r);
155
156        n_InpMult(curr, inv, r);
157       
158        while( numberCollectionEnumerator.MoveNext() )
159        {
160          number &n = numberCollectionEnumerator.Current();
161          n_Normalize(n, r); // ?
162          n_InpMult(n, inv, r); // TODO: either this or directly divide!!!?
163        }
164
165        n_Delete(&inv, r);       
166      }     
167    } else c = n_Init(1, r);
168   
169    return;
170  }
171#endif
172
173  assume(!nCoeff_is_Ring(r));
174  assume(nCoeff_is_Zp(r) || nCoeff_is_numeric(r) || nCoeff_is_GF(r) || nCoeff_is_Zp_a(r) || nCoeff_is_Q_algext(r));
175
176  n_Normalize(curr, r); // Q: good/bad/ugly??
177
178  if (!n_IsOne(curr, r))
179  {
180    number t = curr; // takes over the curr! note: not a reference!!!
181
182    curr = n_Init(1, r); // ???
183   
184    number inv = n_Invers(t, r);
185   
186    while( numberCollectionEnumerator.MoveNext() )
187    {
188      number &n = numberCollectionEnumerator.Current();
189      n_InpMult(n, inv, r); // TODO: either this or directly divide!!!?
190//      n_Normalize(n, r); // ?
191    }
192   
193    n_Delete(&inv, r);
194
195    c = t;
196  } else
197    c = n_Copy(curr, r); // c == 1 and nothing else to do...
198}
199
200void ndClearDenominators(ICoeffsEnumerator& /*numberCollectionEnumerator*/, number& d, const coeffs r)
201{
202  assume( r != NULL );
203  assume( !(nCoeff_is_Q(r) || nCoeff_is_transExt(r) || nCoeff_is_algExt(r)) );
204  assume( nCoeff_is_Ring(r) || nCoeff_is_Zp(r) || nCoeff_is_numeric(r) || nCoeff_is_GF(r) );
205
206  d = n_Init(1, r);
207}
208
209number ndCopy(number a, const coeffs) { return a; }
210number ndCopyMap(number a, const coeffs aRing, const coeffs r)
211{
212  assume( getCoeffType(r) == getCoeffType(aRing) );
213  if ( nCoeff_has_simple_Alloc(r) && nCoeff_has_simple_Alloc(aRing) )
214    return a;
215        else
216    return n_Copy(a, r);
217}
218void ndKillChar(coeffs) {}
219void ndSetChar(const coeffs) {}
220
221number nd_Copy(number a, const coeffs r) { return n_Copy(a, r); }
222
223#ifdef HAVE_RINGS
224BOOLEAN ndDivBy(number, number, const coeffs) { return TRUE; } // assume a,b !=0
225int ndDivComp(number, number, const coeffs) { return 2; }
226BOOLEAN ndIsUnit(number a, const coeffs r) { return !n_IsZero(a,r); }
227number  ndExtGcd (number, number, number *, number *, const coeffs r) { return n_Init(1,r); }
228#endif
229
230#ifdef HAVE_FACTORY
231CanonicalForm ndConvSingNFactoryN( number, BOOLEAN /*setChar*/, const coeffs)
232{
233  CanonicalForm term(0);
234  Werror("no conversion to factory");
235  return term;
236}
237
238number ndConvFactoryNSingN( const CanonicalForm, const coeffs)
239{
240  Werror("no conversion from factory");
241  return NULL;
242}
243#endif
244
245number  ndInit_bigint(number, const coeffs, const coeffs)
246{
247  Werror("no conversion from bigint to this field");
248  return NULL;
249}
250
251/**< [in, out] a bigint number >= 0  */
252/**< [out] the GMP equivalent    */
253/// Converts a non-negative bigint number into a GMP number.
254void ndMPZ(mpz_t result, number &n, const coeffs r)
255{
256  mpz_init_set_si( result, n_Int(n, r) );
257}
258
259number ndInitMPZ(mpz_t m, const coeffs r)
260{ 
261  return n_Init( mpz_get_si(m), r);
262}
263
264
265BOOLEAN ndCoeffIsEqual(const coeffs r, n_coeffType n, void *)
266{
267  /* test, if r is an instance of nInitCoeffs(n,parameter) */
268  /* if paramater is not needed */
269  return (n==r->type);
270}
271
272static n_coeffType nLastCoeffs=n_CF;
273cfInitCharProc nInitCharTableDefault[]=
274{ NULL,        /*n_unknown */
275 npInitChar,   /* n_Zp */
276 nlInitChar,   /* n_Q */
277 nrInitChar,   /* n_R */
278 nfInitChar,   /* n_GF */
279 ngfInitChar,  /* n_long_R */
280 #ifdef HAVE_POLYEXTENSIONS
281 naInitChar,  /* n_algExt */
282 ntInitChar,  /* n_transExt */
283 #else
284 NULL,        /* n_algExt */
285 NULL,        /* n_transExt */
286 #endif   
287 ngcInitChar,  /* n_long_C */
288 #ifdef HAVE_RINGS
289 nrzInitChar,  /* n_Z */
290 nrnInitChar,  /* n_Zn */
291 nrnInitChar,  /* n_Znm */
292 nr2mInitChar, /* n_Z2m */
293 #else
294 NULL,         /* n_Z */
295 NULL,         /* n_Zn */
296 NULL,         /* n_Znm */
297 NULL,         /* n_Z2m */
298 #endif
299 NULL   /* n_CF */
300};
301
302static cfInitCharProc *nInitCharTable=nInitCharTableDefault;
303/*2
304* init operations for coeffs r
305*/
306coeffs nInitChar(n_coeffType t, void * parameter)
307{
308  n_Procs_s *n=cf_root;
309
310  while((n!=NULL) && (n->nCoeffIsEqual!=NULL) && (!n->nCoeffIsEqual(n,t,parameter)))
311      n=n->next;
312
313  if (n==NULL)
314  {
315    n=(n_Procs_s*)omAlloc0(sizeof(n_Procs_s));
316    n->next=cf_root;
317    n->ref=1;
318    n->type=t;
319
320    // default entries (different from NULL) for some routines:
321    n->nCoeffIsEqual = ndCoeffIsEqual;
322    n->cfSize = ndSize;
323    n->cfGetDenom= ndGetDenom;
324    n->cfGetNumerator= ndGetNumerator;
325    n->cfName =  ndName;
326    n->cfImPart=ndReturn0;
327    n->cfDelete= ndDelete;
328    n->cfInpMult=ndInpMult;
329    n->cfCopy = ndCopy;
330    n->cfIntMod=ndIntMod; /* dummy !! */
331    n->cfNormalize=ndNormalize;
332    n->cfGcd  = ndGcd;
333    n->cfLcm  = ndGcd; /* tricky, isn't it ?*/
334    n->cfInit_bigint = ndInit_bigint;
335    n->cfInitMPZ = ndInitMPZ;
336    n->cfMPZ = ndMPZ;
337
338    //n->cfKillChar = ndKillChar; /* dummy */
339    n->cfSetChar = ndSetChar; /* dummy */
340    // temp. removed to catch all the coeffs which miss to implement this!
341
342    n->cfChineseRemainder = ndChineseRemainder;
343    n->cfFarey = ndFarey;
344    n->cfParDeg = ndParDeg;
345   
346    n->cfParameter = ndParameter;
347
348    n->cfClearContent = ndClearContent;
349    n->cfClearDenominators = ndClearDenominators;
350
351#ifdef HAVE_RINGS
352    n->cfDivComp = ndDivComp;
353    n->cfDivBy = ndDivBy;
354    n->cfIsUnit = ndIsUnit;
355    n->cfExtGcd = ndExtGcd;
356    //n->cfGetUnit = (nMapFunc)NULL;
357#endif
358
359#ifdef fACTORY
360    n->convSingNFactoryN=ndConvSingNFactoryN;
361    n->convFactoryNSingN=ndConvFactoryNSingN;
362#endif
363   
364    BOOLEAN nOK=TRUE;
365    // init
366    if ((t<=nLastCoeffs) && (nInitCharTable[t]!=NULL))
367      nOK = (nInitCharTable[t])(n,parameter);
368    else
369       Werror("Sorry: the coeff type [%d] was not registered: it is missing in nInitCharTable", (int)t);
370    if (nOK)
371    {
372      omFreeSize(n,sizeof(*n));
373      return NULL;
374    }
375    cf_root=n;
376    // post init settings:
377    if (n->cfRePart==NULL) n->cfRePart=n->cfCopy;
378    if (n->cfIntDiv==NULL) n->cfIntDiv=n->cfDiv;
379   
380#ifdef HAVE_RINGS
381    if (n->cfGetUnit==NULL) n->cfGetUnit=n->cfCopy;
382#endif
383
384    if(n->cfWriteShort==NULL)
385      n->cfWriteShort = n->cfWriteLong;
386
387    assume(n->nCoeffIsEqual!=NULL);
388    assume(n->cfSetChar!=NULL);
389    assume(n->cfMult!=NULL);
390    assume(n->cfSub!=NULL);
391    assume(n->cfAdd!=NULL);
392    assume(n->cfDiv!=NULL);
393    assume(n->cfIntDiv!=NULL);
394    assume(n->cfIntMod!=NULL);
395    assume(n->cfExactDiv!=NULL);
396    assume(n->cfInit!=NULL);
397    assume(n->cfInitMPZ!=NULL);
398    assume(n->cfSize!=NULL);
399    assume(n->cfInt!=NULL);
400    assume(n->cfMPZ!=NULL);
401    //assume(n->n->cfDivComp!=NULL);
402    //assume(n->cfIsUnit!=NULL);
403    //assume(n->cfGetUnit!=NULL);
404    //assume(n->cfExtGcd!=NULL);
405    assume(n->cfNeg!=NULL);
406    assume(n->cfCopy!=NULL);
407    assume(n->cfRePart!=NULL);
408    assume(n->cfImPart!=NULL);
409
410    assume(n->cfWriteLong!=NULL);
411    assume(n->cfWriteShort!=NULL);
412
413    assume(n->iNumberOfParameters>= 0);
414
415    assume( (n->iNumberOfParameters == 0 && n->pParameterNames == NULL) ||
416            (n->iNumberOfParameters >  0 && n->pParameterNames != NULL) );           
417
418    assume(n->cfParameter!=NULL);
419    assume(n->cfParDeg!=NULL);
420     
421    assume(n->cfRead!=NULL);
422    assume(n->cfNormalize!=NULL);
423    assume(n->cfGreater!=NULL);
424    //assume(n->cfDivBy!=NULL);
425    assume(n->cfEqual!=NULL);
426    assume(n->cfIsZero!=NULL);
427    assume(n->cfIsOne!=NULL);
428    assume(n->cfIsMOne!=NULL);
429    assume(n->cfGreaterZero!=NULL);
430    assume(n->cfPower!=NULL);
431    assume(n->cfGetDenom!=NULL);
432    assume(n->cfGetNumerator!=NULL);
433    assume(n->cfGcd!=NULL);
434    assume(n->cfLcm!=NULL);
435    assume(n->cfDelete!=NULL);
436    assume(n->cfSetMap!=NULL);
437    assume(n->cfName!=NULL);
438    assume(n->cfInpMult!=NULL);
439//    assume(n->cfInit_bigint!=NULL);
440    assume(n->cfCoeffWrite != NULL);
441
442    assume(n->cfClearContent != NULL);
443    assume(n->cfClearDenominators != NULL);
444   
445#ifdef LDEBUG
446    assume(n->cfDBTest!=NULL);
447#endif
448    assume(n->type==t);
449     
450#ifndef NDEBUG
451    if(n->cfKillChar==NULL) Warn("cfKillChar is NULL for coeff %d",t);
452    if(n->cfWriteLong==NULL) Warn("cfWrite is NULL for coeff %d",t);
453    if(n->cfWriteShort==NULL) Warn("cfWriteShort is NULL for coeff %d",t);
454#endif
455     
456   if( n->nNULL == NULL )
457     n->nNULL = n_Init(0, n); // may still remain NULL
458  }
459  else
460  {
461    n->ref++;
462  }
463  return n;
464}
465
466void nKillChar(coeffs r)
467{
468  if (r!=NULL)
469  {
470    r->ref--;
471    if (r->ref<=0)
472    {
473      n_Procs_s tmp;
474      n_Procs_s* n=&tmp;
475      tmp.next=cf_root;
476      while((n->next!=NULL) && (n->next!=r)) n=n->next;
477      if (n->next==r)
478      {
479        n->next=n->next->next;
480        if (cf_root==r) cf_root=n->next;
481        r->cfDelete(&(r->nNULL),r);
482        if (r->cfKillChar!=NULL) r->cfKillChar(r);
483        omFreeSize((void *)r, sizeof(n_Procs_s));
484        r=NULL;
485      }
486      else
487      {
488        WarnS("cf_root list destroyed");
489      }
490    }
491  }
492}
493
494
495n_coeffType nRegister(n_coeffType n, cfInitCharProc p)
496{
497  if (n==n_unknown)
498  {
499    nLastCoeffs=(n_coeffType)(int(nLastCoeffs)+1);
500    if (nInitCharTable==nInitCharTableDefault)
501    {
502      nInitCharTable=(cfInitCharProc*)omAlloc0(
503                                          nLastCoeffs*sizeof(cfInitCharProc));
504      memcpy(nInitCharTable,nInitCharTableDefault,
505              (nLastCoeffs-1)*sizeof(cfInitCharProc));
506    }
507    else
508    {
509      nInitCharTable=(cfInitCharProc*)omReallocSize(nInitCharTable,
510                                          (((int)nLastCoeffs)-1)*sizeof(cfInitCharProc),
511                                          ((int)nLastCoeffs)*sizeof(cfInitCharProc));
512    }
513
514    nInitCharTable[nLastCoeffs]=p;
515    return nLastCoeffs;
516  }
517  else
518  {
519    if (nInitCharTable[n]!=NULL) Print("coeff %d already initialized\n",n);
520    nInitCharTable[n]=p;
521    return n;
522  }
523}
524
525
526void n_Print(number& a,  const coeffs r)
527{ 
528   assume(r != NULL); 
529   n_Test(a,r); 
530   
531   StringSetS(""); 
532   n_Write(a, r); 
533   { char* s = StringEndS(); Print("%s", s); omFree(s); }
534}
Note: See TracBrowser for help on using the repository browser.