[BACK]Return to Hgfs.c CVS log [TXT][DIR] Up to [local] / OpenXM_contrib2 / asir2000 / engine

Annotation of OpenXM_contrib2/asir2000/engine/Hgfs.c, Revision 1.34

1.34    ! noro        1: /* $OpenXM: OpenXM_contrib2/asir2000/engine/Hgfs.c,v 1.33 2015/08/14 13:51:54 fujimoto Exp $ */
1.1       noro        2:
                      3: #include "ca.h"
1.22      noro        4: #include "inline.h"
1.1       noro        5:
1.30      noro        6: int debug_sfbfctr;
                      7:
1.18      noro        8: void lnfsf(int n,UM p0,UM p1,struct p_pair *list,UM np0,UM np1);
1.20      noro        9: void extractcoefbm(BM f,int dx,UM r);
1.1       noro       10:
1.18      noro       11: int comp_dum(DUM a,DUM b)
1.12      noro       12: {
1.34    ! noro       13:   if ( DEG(a->f) > DEG(b->f) )
        !            14:     return -1;
        !            15:   else if ( DEG(a->f) < DEG(b->f) )
        !            16:     return 1;
        !            17:   else
        !            18:     return 0;
1.12      noro       19: }
                     20:
1.25      noro       21: void ufctrsf(P p,DCP *dcp)
1.1       noro       22: {
1.34    ! noro       23:   int n,i,j,k;
        !            24:   DCP dc,dc0;
        !            25:   P lc;
        !            26:   UM mp;
        !            27:   UM *tl;
        !            28:   Obj obj;
        !            29:   struct oDUM *udc,*udc1;
        !            30:
        !            31:   simp_ff((Obj)p,&obj); p = (P)obj;
        !            32:   if ( !p ) {
        !            33:     NEWDC(dc); COEF(dc) = 0; DEG(dc) = ONE;
        !            34:     NEXT(dc) = 0; *dcp = dc;
        !            35:     return;
        !            36:   }
        !            37:   mp = W_UMALLOC(UDEG(p));
        !            38:   ptosfum(p,mp);
        !            39:   if ( (n = DEG(mp)) < 0 ) {
        !            40:     *dcp = 0; return;
        !            41:   } else if ( n == 0 ) {
        !            42:     NEWDC(dc); COEF(dc) = p; DEG(dc) = ONE;
        !            43:     NEXT(dc) = 0; *dcp = dc;
        !            44:     return;
        !            45:   }
        !            46:   lc = COEF(DC(p));
        !            47:   if ( !_isonesf(COEF(mp)[n]) ) {
        !            48:     monicsfum(mp);
        !            49:   }
        !            50:
        !            51:   W_CALLOC(n+1,struct oDUM,udc);
        !            52:   gensqfrsfum(mp,udc);
        !            53:
        !            54:   tl = (UM *)ALLOCA((n+1)*sizeof(UM));
        !            55:   W_CALLOC(DEG(mp)+1,struct oDUM,udc1);
        !            56:
        !            57:   for ( i = 0,j = 0; udc[i].f; i++ )
        !            58:     if ( DEG(udc[i].f) == 1 ) {
        !            59:       udc1[j].f = udc[i].f; udc1[j].n = udc[i].n; j++;
        !            60:     } else {
        !            61:       bzero((char *)tl,(n+1)*sizeof(UM));
        !            62:       czsfum(udc[i].f,tl);
        !            63:       for ( k = 0; tl[k]; k++, j++ ) {
        !            64:         udc1[j].f = tl[k]; udc1[j].n = udc[i].n;
        !            65:       }
        !            66:     }
        !            67:   udc = udc1;
        !            68:   for ( i = 0; udc[i].f; i++ );
        !            69:   qsort(udc,i,sizeof(struct oDUM),
        !            70:     (int (*)(const void *,const void *))comp_dum);
        !            71:
        !            72:   NEWDC(dc0); COEF(dc0) = lc; DEG(dc0) = ONE; dc = dc0;
        !            73:   for ( n = 0; udc[n].f; n++ ) {
        !            74:     NEWDC(NEXT(dc)); dc = NEXT(dc);
        !            75:     STOQ(udc[n].n,DEG(dc)); sfumtop(VR(p),udc[n].f,&COEF(dc));
        !            76:   }
        !            77:   NEXT(dc) = 0; *dcp = dc0;
1.1       noro       78: }
                     79:
1.18      noro       80: void gensqfrsfum(UM p,struct oDUM *dc)
1.1       noro       81: {
1.34    ! noro       82:   int n,i,j,d,mod;
        !            83:   UM t,s,g,f,f1,b;
        !            84:   GFS u,v;
        !            85:
        !            86:   if ( (n = DEG(p)) == 1 ) {
        !            87:     dc[0].f = UMALLOC(DEG(p)); cpyum(p,dc[0].f); dc[0].n = 1;
        !            88:     return;
        !            89:   }
        !            90:   t = W_UMALLOC(n); s = W_UMALLOC(n); g = W_UMALLOC(n);
        !            91:   f = W_UMALLOC(n); f1 = W_UMALLOC(n); b = W_UMALLOC(n);
        !            92:   diffsfum(p,t); cpyum(p,s); gcdsfum(t,s,g);
        !            93:   if ( !DEG(g) ) {
        !            94:     dc[0].f = UMALLOC(DEG(p)); cpyum(p,dc[0].f); dc[0].n = 1;
        !            95:     return;
        !            96:   }
        !            97:   cpyum(p,b); cpyum(p,t); divsfum(t,g,f);
        !            98:   for ( i = 0, d = 0; DEG(f); i++ ) {
        !            99:     while ( 1 ) {
        !           100:       cpyum(b,t);
        !           101:       if ( divsfum(t,f,s) >= 0 )
        !           102:         break;
        !           103:       else {
        !           104:         cpyum(s,b); d++;
        !           105:       }
        !           106:     }
        !           107:     cpyum(b,t); cpyum(f,s); gcdsfum(t,s,f1);
        !           108:     divsfum(f,f1,s); cpyum(f1,f);
        !           109:     dc[i].f = UMALLOC(DEG(s)); cpyum(s,dc[i].f); dc[i].n = d;
        !           110:   }
        !           111:   mod = characteristic_sf();
        !           112:   if ( DEG(b) > 0 ) {
        !           113:     d = 1;
        !           114:     while ( 1 ) {
        !           115:       cpyum(b,t);
        !           116:       for ( j = DEG(t); j >= 0; j-- )
        !           117:         if ( COEF(t)[j] && (j % mod) )
        !           118:           break;
        !           119:       if ( j >= 0 )
        !           120:         break;
        !           121:       else {
        !           122:         DEG(s) = DEG(t)/mod;
        !           123:         for ( j = 0; j <= DEG(t); j++ ) {
        !           124:           iftogfs(COEF(t)[j*mod],&u);
        !           125:           pthrootgfs(u,&v);
        !           126:           COEF(s)[j] = v?FTOIF(CONT(v)):0;
        !           127:         }
        !           128:         cpyum(s,b); d *= mod;
        !           129:       }
        !           130:     }
        !           131:     gensqfrsfum(b,dc+i);
        !           132:     for ( j = i; dc[j].f; j++ )
        !           133:       dc[j].n *= d;
        !           134:   }
1.1       noro      135: }
                    136:
1.18      noro      137: void randsfum(int d,UM p)
1.1       noro      138: {
1.34    ! noro      139:   int i;
1.1       noro      140:
1.34    ! noro      141:   for ( i = 0; i < d; i++ )
        !           142:     COEF(p)[i] = _randomsf();
        !           143:   for ( i = d-1; i >= 0 && !COEF(p)[i]; i-- );
        !           144:   p->d = i;
1.1       noro      145: }
                    146:
1.18      noro      147: void pwrmodsfum(UM p,int e,UM f,UM pr)
1.1       noro      148: {
1.34    ! noro      149:   UM wt,ws,q;
1.1       noro      150:
1.34    ! noro      151:   if ( e == 0 ) {
        !           152:     DEG(pr) = 0; COEF(pr)[0] = _onesf();
        !           153:   } else if ( DEG(p) < 0 )
        !           154:     DEG(pr) = -1;
        !           155:   else if ( e == 1 ) {
        !           156:     q = W_UMALLOC(DEG(p)); cpyum(p,pr);
        !           157:     DEG(pr) = divsfum(pr,f,q);
        !           158:   } else if ( DEG(p) == 0 ) {
        !           159:     DEG(pr) = 0; COEF(pr)[0] = _pwrsf(COEF(p)[0],e);
        !           160:   } else {
        !           161:     wt = W_UMALLOC(2*DEG(f)); ws = W_UMALLOC(2*DEG(f));
        !           162:     q = W_UMALLOC(2*DEG(f));
        !           163:     pwrmodsfum(p,e/2,f,wt);
        !           164:     if ( !(e%2) )  {
        !           165:       mulsfum(wt,wt,pr); DEG(pr) = divsfum(pr,f,q);
        !           166:     } else {
        !           167:       mulsfum(wt,wt,ws);
        !           168:       DEG(ws) = divsfum(ws,f,q);
        !           169:       mulsfum(ws,p,pr);
        !           170:       DEG(pr) = divsfum(pr,f,q);
        !           171:     }
        !           172:   }
1.1       noro      173: }
                    174:
1.18      noro      175: void spwrsfum(UM m,UM f,N e,UM r)
1.1       noro      176: {
1.34    ! noro      177:   UM t,s,q;
        !           178:   N e1;
        !           179:   int a;
        !           180:
        !           181:   if ( !e ) {
        !           182:     DEG(r) = 0; COEF(r)[0] = _onesf();
        !           183:   } else if ( UNIN(e) )
        !           184:     cpyum(f,r);
        !           185:   else {
        !           186:     a = divin(e,2,&e1);
        !           187:     t = W_UMALLOC(2*DEG(m)); spwrsfum(m,f,e1,t);
        !           188:     s = W_UMALLOC(2*DEG(m)); q = W_UMALLOC(2*DEG(m));
        !           189:     mulsfum(t,t,s); DEG(s) = divsfum(s,m,q);
        !           190:     if ( a ) {
        !           191:       mulsfum(s,f,t); DEG(t) = divsfum(t,m,q); cpyum(t,r);
1.1       noro      192:         } else
1.34    ! noro      193:       cpyum(s,r);
        !           194:   }
1.1       noro      195: }
                    196:
1.18      noro      197: void tracemodsfum(UM m,UM f,int e,UM r)
1.2       noro      198: {
1.34    ! noro      199:   UM t,s,q,u;
        !           200:   int i;
1.2       noro      201:
1.34    ! noro      202:   q = W_UMALLOC(2*DEG(m)+DEG(f)); /* XXX */
        !           203:   t = W_UMALLOC(2*DEG(m));
        !           204:   s = W_UMALLOC(2*DEG(m));
        !           205:   u = W_UMALLOC(2*DEG(m));
        !           206:   DEG(f) = divsfum(f,m,q);
        !           207:   cpyum(f,s);
        !           208:   cpyum(f,t);
        !           209:   for ( i = 1; i < e; i++ ) {
        !           210:     mulsfum(t,t,u);
        !           211:     DEG(u) = divsfum(u,m,q); cpyum(u,t);
        !           212:     addsfum(t,s,u); cpyum(u,s);
        !           213:   }
        !           214:   cpyum(s,r);
1.2       noro      215: }
                    216:
1.18      noro      217: void make_qmatsf(UM p,UM *tab,int ***mp)
1.1       noro      218: {
1.34    ! noro      219:   int n,i,j;
        !           220:   int *c;
        !           221:   UM q,r;
        !           222:   int **mat;
        !           223:   int one;
        !           224:
        !           225:   n = DEG(p);
        !           226:   *mp = mat = almat(n,n);
        !           227:   for ( j = 0; j < n; j++ ) {
        !           228:     r = W_UMALLOC(DEG(tab[j])); q = W_UMALLOC(DEG(tab[j]));
        !           229:     cpyum(tab[j],r); DEG(r) = divsfum(r,p,q);
        !           230:     for ( i = 0, c = COEF(r); i <= DEG(r); i++ )
        !           231:       mat[i][j] = c[i];
        !           232:   }
        !           233:   one = _onesf();
        !           234:   for ( i = 0; i < n; i++ )
        !           235:     mat[i][i] = _subsf(mat[i][i],one);
1.1       noro      236: }
                    237:
1.18      noro      238: void nullsf(int **mat,int n,int *ind)
1.1       noro      239: {
1.34    ! noro      240:   int i,j,l,s,h,inv;
        !           241:   int *t,*u;
1.1       noro      242:
1.34    ! noro      243:   bzero((char *)ind,n*sizeof(int));
        !           244:   ind[0] = 0;
        !           245:   for ( i = j = 0; j < n; i++, j++ ) {
        !           246:     for ( ; j < n; j++ ) {
        !           247:       for ( l = i; l < n; l++ )
        !           248:         if ( mat[l][j] )
        !           249:           break;
        !           250:       if ( l < n ) {
        !           251:         t = mat[i]; mat[i] = mat[l]; mat[l] = t; break;
        !           252:       } else
        !           253:         ind[j] = 1;
        !           254:     }
        !           255:     if ( j == n )
        !           256:       break;
        !           257:     inv = _invsf(mat[i][j]);
        !           258:     for ( s = j, t = mat[i]; s < n; s++ )
        !           259:       t[s] = _mulsf(t[s],inv);
        !           260:     for ( l = 0; l < n; l++ ) {
        !           261:       if ( l == i )
        !           262:         continue;
        !           263:       u = mat[l]; h = _chsgnsf(u[j]);
        !           264:       for ( s = j; s < n; s++ )
        !           265:         u[s] = _addsf(_mulsf(h,t[s]),u[s]);
        !           266:     }
        !           267:   }
1.1       noro      268: }
                    269:
1.18      noro      270: void null_to_solsf(int **mat,int *ind,int n,UM *r)
1.1       noro      271: {
1.34    ! noro      272:   int i,j,k,l;
        !           273:   int *c;
        !           274:   UM w;
        !           275:
        !           276:   for ( i = 0, l = 0; i < n; i++ ) {
        !           277:     if ( !ind[i] )
        !           278:       continue;
        !           279:     w = UMALLOC(n);
        !           280:     for ( j = k = 0, c = COEF(w); j < n; j++ )
        !           281:       if ( ind[j] )
        !           282:         c[j] = 0;
        !           283:       else
        !           284:         c[j] = mat[k++][i];
        !           285:     c[i] = _chsgnsf(_onesf());
        !           286:     for ( j = n; j >= 0; j-- )
        !           287:       if ( c[j] )
        !           288:         break;
        !           289:     DEG(w) = j;
        !           290:     r[l++] = w;
        !           291:   }
1.1       noro      292: }
                    293: /*
                    294: make_qmatsf(p,tab,mp)
                    295: nullsf(mat,n,ind)
                    296: null_to_solsf(ind,n,r)
                    297: */
                    298:
1.18      noro      299: void czsfum(UM f,UM *r)
1.1       noro      300: {
1.34    ! noro      301:   int i,j;
        !           302:   int d,n,ord;
        !           303:   UM s,t,u,v,w,g,x,m,q;
        !           304:   UM *base;
        !           305:
        !           306:   n = DEG(f); base = (UM *)ALLOCA(n*sizeof(UM));
        !           307:   bzero((char *)base,n*sizeof(UM));
        !           308:
        !           309:   w = W_UMALLOC(2*n); q = W_UMALLOC(2*n); m = W_UMALLOC(2*n);
        !           310:
        !           311:   base[0] = W_UMALLOC(0); DEG(base[0]) = 0; COEF(base[0])[0] = _onesf();
        !           312:
        !           313:   t = W_UMALLOC(1); DEG(t) = 1; COEF(t)[0] = 0; COEF(t)[1] = _onesf();
        !           314:
        !           315:   ord = field_order_sf();
        !           316:   pwrmodsfum(t,ord,f,w);
        !           317:   base[1] = W_UMALLOC(DEG(w));
        !           318:   cpyum(w,base[1]);
        !           319:
        !           320:   for ( i = 2; i < n; i++ ) {
        !           321:     mulsfum(base[i-1],base[1],m);
        !           322:     DEG(m) = divsfum(m,f,q);
        !           323:     base[i] = W_UMALLOC(DEG(m)); cpyum(m,base[i]);
        !           324:   }
        !           325:
        !           326:   v = W_UMALLOC(n); cpyum(f,v);
        !           327:   DEG(w) = 1; COEF(w)[0] = 0; COEF(w)[1] = _onesf();
        !           328:   x = W_UMALLOC(1); DEG(x) = 1; COEF(x)[0] = 0; COEF(x)[1] = _onesf();
        !           329:   t = W_UMALLOC(n); s = W_UMALLOC(n); u = W_UMALLOC(n); g = W_UMALLOC(n);
        !           330:
        !           331:   for ( j = 0, d = 1; 2*d <= DEG(v); d++ ) {
        !           332:     for ( DEG(t) = -1, i = 0; i <= DEG(w); i++ )
        !           333:       if ( COEF(w)[i] ) {
        !           334:         mulssfum(base[i],COEF(w)[i],s);
        !           335:         addsfum(s,t,u); cpyum(u,t);
        !           336:       }
        !           337:     cpyum(t,w); cpyum(v,s); subsfum(w,x,t);
        !           338:     gcdsfum(s,t,g);
        !           339:     if ( DEG(g) >= 1 ) {
        !           340:       berlekampsf(g,d,base,r+j); j += DEG(g)/d;
        !           341:       divsfum(v,g,q); cpyum(q,v);
        !           342:       DEG(w) = divsfum(w,v,q);
        !           343:       for ( i = 0; i < DEG(v); i++ )
        !           344:         DEG(base[i]) = divsfum(base[i],v,q);
        !           345:     }
        !           346:   }
        !           347:   if ( DEG(v) ) {
        !           348:     r[j] = UMALLOC(DEG(v)); cpyum(v,r[j]); j++;
        !           349:   }
        !           350:   r[j] = 0;
1.1       noro      351: }
                    352:
1.18      noro      353: int berlekampsf(UM p,int df,UM *tab,UM *r)
1.1       noro      354: {
1.34    ! noro      355:   int n,i,j,k,nf,d,nr;
        !           356:   int **mat;
        !           357:   int *ind;
        !           358:   UM mp,w,q,gcd,w1,w2;
        !           359:   UM *u;
        !           360:   int *root;
        !           361:
        !           362:   n = DEG(p);
        !           363:   ind = ALLOCA(n*sizeof(int));
        !           364:   make_qmatsf(p,tab,&mat);
        !           365:   nullsf(mat,n,ind);
        !           366:   for ( i = 0, d = 0; i < n; i++ )
        !           367:     if ( ind[i] )
        !           368:       d++;
        !           369:   if ( d == 1 ) {
        !           370:     r[0] = UMALLOC(n); cpyum(p,r[0]); return 1;
        !           371:   }
        !           372:   u = ALLOCA(d*sizeof(UM *));
        !           373:   r[0] = UMALLOC(n); cpyum(p,r[0]);
        !           374:   null_to_solsf(mat,ind,n,u);
        !           375:   root = ALLOCA(d*sizeof(int));
        !           376:   w = W_UMALLOC(n); mp = W_UMALLOC(d);
        !           377:   w1 = W_UMALLOC(n); w2 = W_UMALLOC(n);
        !           378:   for ( i = 1, nf = 1; i < d; i++ ) {
        !           379:     minipolysf(u[i],p,mp);
        !           380:     nr = find_rootsf(mp,root);
        !           381:     for ( j = 0; j < nf; j++ ) {
        !           382:       if ( DEG(r[j]) == df )
        !           383:         continue;
        !           384:       for ( k = 0; k < nr; k++ ) {
        !           385:         cpyum(u[i],w1); cpyum(r[j],w2);
        !           386:         COEF(w1)[0] = _chsgnsf(root[k]);
        !           387:         gcdsfum(w1,w2,w);
        !           388:         if ( DEG(w) > 0 && DEG(w) < DEG(r[j]) ) {
        !           389:           gcd = UMALLOC(DEG(w));
        !           390:           q = UMALLOC(DEG(r[j])-DEG(w));
        !           391:           cpyum(w,gcd); divsfum(r[j],w,q);
        !           392:           r[j] = q; r[nf++] = gcd;
        !           393:         }
        !           394:         if ( nf == d )
        !           395:           return d;
        !           396:       }
        !           397:     }
        !           398:   }
        !           399:   /* NOT REACHED */
        !           400:   error("berlekampsf : cannot happen");
        !           401:   return 0;
1.1       noro      402: }
                    403:
1.18      noro      404: void minipolysf(UM f,UM p,UM mp)
1.1       noro      405: {
1.34    ! noro      406:   struct p_pair *list,*l,*l1,*lprev;
        !           407:   int n,d;
        !           408:   UM u,p0,p1,np0,np1,q,w;
        !           409:
        !           410:   list = (struct p_pair *)MALLOC(sizeof(struct p_pair));
        !           411:   list->p0 = u = W_UMALLOC(0); DEG(u) = 0; COEF(u)[0] = _onesf();
        !           412:   list->p1 = W_UMALLOC(0); cpyum(list->p0,list->p1);
        !           413:   list->next = 0;
        !           414:   n = DEG(p); w = UMALLOC(2*n);
        !           415:   p0 = UMALLOC(2*n); cpyum(list->p0,p0);
        !           416:   p1 = UMALLOC(2*n); cpyum(list->p1,p1);
        !           417:   q = W_UMALLOC(2*n);
        !           418:   while ( 1 ) {
        !           419:     COEF(p0)[DEG(p0)] = 0; DEG(p0)++; COEF(p0)[DEG(p0)] = _onesf();
        !           420:     mulsfum(f,p1,w); DEG(w) = divsfum(w,p,q); cpyum(w,p1);
        !           421:     np0 = UMALLOC(n); np1 = UMALLOC(n);
        !           422:     lnfsf(n,p0,p1,list,np0,np1);
        !           423:     if ( DEG(np1) < 0 ) {
        !           424:       cpyum(np0,mp); return;
        !           425:     } else {
        !           426:       l1 = (struct p_pair *)MALLOC(sizeof(struct p_pair));
        !           427:       l1->p0 = np0; l1->p1 = np1;
        !           428:       for ( l = list, lprev = 0, d = DEG(np1);
        !           429:         l && (DEG(l->p1) > d); lprev = l, l = l->next );
        !           430:       if ( lprev ) {
        !           431:         lprev->next = l1; l1->next = l;
        !           432:       } else {
        !           433:         l1->next = list; list = l1;
        !           434:       }
        !           435:     }
        !           436:   }
1.1       noro      437: }
                    438:
1.18      noro      439: void lnfsf(int n,UM p0,UM p1,struct p_pair *list,UM np0,UM np1)
1.1       noro      440: {
1.34    ! noro      441:   int h,d1;
        !           442:   UM t0,t1,s0,s1;
        !           443:   struct p_pair *l;
        !           444:
        !           445:   cpyum(p0,np0); cpyum(p1,np1);
        !           446:   t0 = W_UMALLOC(n); t1 = W_UMALLOC(n);
        !           447:   s0 = W_UMALLOC(n); s1 = W_UMALLOC(n);
        !           448:   for ( l = list; l; l = l->next ) {
        !           449:     d1 = DEG(np1);
        !           450:     if ( d1 == DEG(l->p1) ) {
        !           451:       h = _divsf(COEF(np1)[d1],_chsgnsf(COEF(l->p1)[d1]));
        !           452:       mulssfum(l->p0,h,t0); addsfum(np0,t0,s0); cpyum(s0,np0);
        !           453:       mulssfum(l->p1,h,t1); addsfum(np1,t1,s1); cpyum(s1,np1);
        !           454:     }
        !           455:   }
1.1       noro      456: }
                    457:
1.18      noro      458: int find_rootsf(UM p,int *root)
1.1       noro      459: {
1.34    ! noro      460:   UM *r;
        !           461:   int i,n;
1.1       noro      462:
1.34    ! noro      463:   n = DEG(p);
        !           464:   r = ALLOCA((DEG(p))*sizeof(UM));
        !           465:   canzassf(p,1,r);
        !           466:   for ( i = 0; i < n; i++ )
        !           467:     root[i] = _chsgnsf(COEF(r[i])[0]);
        !           468:   return n;
1.1       noro      469: }
                    470:
1.18      noro      471: void canzassf(UM f,int d,UM *r)
1.1       noro      472: {
1.34    ! noro      473:   UM t,s,u,w,g,o;
        !           474:   N n1,n2,n3,n4,n5;
        !           475:   UM *b;
        !           476:   int n,q,ed;
        !           477:
        !           478:   if ( DEG(f) == d ) {
        !           479:     r[0] = UMALLOC(d); cpyum(f,r[0]);
        !           480:     return;
        !           481:   } else {
        !           482:     n = DEG(f); b = (UM *)ALLOCA(n*sizeof(UM));
        !           483:     bzero((char *)b,n*sizeof(UM));
        !           484:
        !           485:     t = W_UMALLOC(2*d);
        !           486:     s = W_UMALLOC(DEG(f)); u = W_UMALLOC(DEG(f));
        !           487:     w = W_UMALLOC(DEG(f)); g = W_UMALLOC(DEG(f));
        !           488:     o = W_UMALLOC(0); DEG(o) = 0; COEF(o)[0] = _onesf();
        !           489:     q = field_order_sf();
        !           490:     if ( q % 2 ) {
        !           491:       STON(q,n1); pwrn(n1,d,&n2); subn(n2,ONEN,&n3);
        !           492:       STON(2,n4); divsn(n3,n4,&n5);
        !           493:     } else
        !           494:       ed = d*extdeg_sf();
        !           495:     while ( 1 ) {
        !           496:       randsfum(2*d,t);
        !           497:       if ( q % 2 ) {
        !           498:         spwrsfum(f,t,n5,s); subsfum(s,o,u);
        !           499:       } else
        !           500:         tracemodsfum(f,t,ed,u);
        !           501:       cpyum(f,w);
        !           502:       gcdsfum(w,u,g);
        !           503:       if ( (DEG(g) >= 1) && (DEG(g) < DEG(f)) ) {
        !           504:         canzassf(g,d,r);
        !           505:         cpyum(f,w); divsfum(w,g,s);
        !           506:         canzassf(s,d,r+DEG(g)/d);
        !           507:         return;
        !           508:       }
        !           509:     }
        !           510:   }
1.1       noro      511: }
                    512:
1.3       noro      513: /* Hensel related functions */
                    514:
1.28      noro      515: int sfberle(V,V,P,int,GFS *,DCP *);
1.3       noro      516: void sfgcdgen(P,ML,ML *);
1.5       noro      517: void sfhenmain2(BM,UM,UM,int,BM *);
                    518: void ptosfbm(int,P,BM);
1.28      noro      519: void sfhensel(int count,P f,V x,V y,int degbound,GFS *evp,P *sfp,ML *listp);
1.3       noro      520:
                    521: /* f = f(x,y) */
                    522:
1.28      noro      523: void sfhensel(int count,P f,V x,V y,int degbound,GFS *evp,P *sfp,ML *listp)
1.3       noro      524: {
1.34    ! noro      525:   int i;
        !           526:   int fn;
        !           527:   ML rlist;
        !           528:   BM fl;
        !           529:   VL vl,nvl;
        !           530:   int dx,dy,bound;
        !           531:   GFS ev;
        !           532:   P f1,t,c,sf;
        !           533:   DCP dc,dct,dc0;
        !           534:   UM q,fm,hm;
        !           535:   UM *gm;
        !           536:   struct oEGT tmp0,tmp1,eg_hensel,eg_hensel_t;
        !           537:
        !           538:   clctv(CO,f,&vl);
        !           539:   if ( vl->v != x ) {
        !           540:     reordvar(vl,x,&nvl); reorderp(nvl,vl,f,&f1);
        !           541:     vl = nvl; f = f1;
        !           542:   }
        !           543:   if ( vl->next )
        !           544:     y = vl->next->v;
        !           545:   dx = getdeg(x,f);
        !           546:   dy = getdeg(y,f);
        !           547:   if ( dx == 1 ) {
        !           548:     *listp = rlist = MLALLOC(1); rlist->n = 1; rlist->c[0] = 0;
        !           549:     return;
        !           550:   }
        !           551:   fn = sfberle(x,y,f,count,&ev,&dc);
        !           552:   if ( fn <= 1 ) {
        !           553:     /* fn == 0 => short of evaluation points */
        !           554:     *listp = rlist = MLALLOC(1); rlist->n = fn; rlist->c[0] = 0;
        !           555:     return;
        !           556:   }
        !           557:   if ( degbound >= 0 ) {
        !           558:     /*
        !           559:      * reconstruct dc so that
        !           560:      * dc[1],... : factors satisfy degree bound
        !           561:      * dc[0]     : product of others
        !           562:      */
        !           563:     c = dc->c; dc = NEXT(dc);
        !           564:     dc0 = 0;
        !           565:     fn = 0;
        !           566:     while ( dc ) {
        !           567:       if ( getdeg(x,COEF(dc)) <= degbound ) {
        !           568:         dct = NEXT(dc); NEXT(dc) = dc0; dc0 = dc; dc = dct;
        !           569:         fn++;
        !           570:       } else {
        !           571:         mulp(vl,COEF(dc),c,&t); c = t;
        !           572:         dc = NEXT(dc);
        !           573:       }
        !           574:     }
        !           575:     if ( OID(c) == O_P ) {
        !           576:       NEWDC(dc); COEF(dc) = c; DEG(dc) = ONE; NEXT(dc) = dc0;
        !           577:       fn++;
        !           578:     } else {
        !           579:       mulp(vl,dc0->c,c,&t); dc0->c = t; dc = dc0;
        !           580:     }
        !           581:   } else {
        !           582:     /* pass the the leading coeff. to the first element */
        !           583:     c = dc->c; dc = NEXT(dc);
        !           584:     mulp(vl,dc->c,c,&t); dc->c = t;
        !           585:   }
        !           586:
        !           587:   /* convert mod y-a factors into UM */
        !           588:   gm = (UM *)ALLOCA(fn*sizeof(UM));
        !           589:   for ( i = 0; i < fn; i++, dc = NEXT(dc) ) {
        !           590:     gm[i] = W_UMALLOC(UDEG(dc->c));
        !           591:     ptosfum(dc->c,gm[i]);
        !           592:   }
        !           593:
        !           594:   /* set bound */
        !           595:   /* g | f, lc_y(g) = lc_y(f) => deg_y(g) <= deg_y(f) */
        !           596:   /* so, bound = dy is sufficient, but we use slightly large value */
        !           597:   bound = dy+2;
        !           598:
        !           599:   /* f(x,y) -> f(x,y+ev) */
        !           600:   fl = BMALLOC(dx,bound);
        !           601:   ptosfbm(bound,f,fl);
        !           602:   if ( ev ) shiftsfbm(fl,FTOIF(CONT(ev)));
        !           603:
        !           604:   /* sf = f(x+ev) */
        !           605:   sfbmtop(fl,x,y,&sf);
        !           606:
        !           607:   /* fm = fl mod y */
        !           608:   fm = W_UMALLOC(dx);
        !           609:   cpyum(COEF(fl)[0],fm);
        !           610:   hm = W_UMALLOC(dx);
        !           611:
        !           612:   q = W_UMALLOC(dx);
        !           613:   rlist = MLALLOC(fn); rlist->n = fn; rlist->bound = bound;
        !           614:   if ( debug_sfbfctr )
        !           615:     fprintf(asir_out,"%d candidates\n",fn);
        !           616:   init_eg(&eg_hensel);
        !           617:   for ( i = 0; i < fn-1; i++ ) {
        !           618:     if ( debug_sfbfctr )
        !           619:       fprintf(asir_out,"deg(fm) = %d, deg(gm[%d]) = %d\n",
        !           620:       DEG(fm),i,DEG(gm[i]));
        !           621:     init_eg(&eg_hensel_t);
        !           622:     get_eg(&tmp0);
        !           623:     /* fl = gm[i]*hm mod y */
        !           624:     divsfum(fm,gm[i],hm);
        !           625:     /* fl is replaced by the cofactor of gk mod y^bound */
        !           626:     /* rlist->c[i] = gk */
        !           627:     sfhenmain2(fl,gm[i],hm,bound,(BM *)&rlist->c[i]);
        !           628:     cpyum(hm,fm);
        !           629:     get_eg(&tmp1); add_eg(&eg_hensel_t,&tmp0,&tmp1);
        !           630:     add_eg(&eg_hensel,&tmp0,&tmp1);
        !           631:     if ( debug_sfbfctr) {
        !           632:       print_eg("Hensel",&eg_hensel_t);
        !           633:       fprintf(asir_out,"\n");
        !           634:     }
        !           635:   }
        !           636:   if ( debug_sfbfctr) {
        !           637:     print_eg("Hensel total",&eg_hensel);
        !           638:     fprintf(asir_out,"\n");
        !           639:   }
        !           640:   /* finally, fl must be the lift of gm[fn-1] */
        !           641:   rlist->c[i] = fl;
1.4       noro      642:
1.8       noro      643: #if 0
1.34    ! noro      644:   /* y -> y-a */
        !           645:   mev = _chsgnsf(FTOIF(CONT(ev)));
        !           646:   for ( i = 0; i < fn; i++ )
        !           647:     shiftsfbm((BM)(rlist->c[i]),mev);
1.8       noro      648: #endif
1.34    ! noro      649:   *evp = ev;
        !           650:   *sfp = sf;
        !           651:   *listp = rlist;
1.3       noro      652: }
                    653:
                    654: /* main variable of f = x */
                    655:
1.28      noro      656: int sfberle(V x,V y,P f,int count,GFS *ev,DCP *dcp)
1.3       noro      657: {
1.34    ! noro      658:   UM wf,wf1,wf2,wfs,gcd;
        !           659:   int fn,n;
        !           660:   GFS m,fm;
        !           661:   DCP dc,dct,dc0;
        !           662:   VL vl;
        !           663:   P lc,lc0,f0;
        !           664:   Obj obj;
        !           665:   int j,q,index,i;
        !           666:
        !           667:   NEWVL(vl); vl->v = x;
        !           668:   NEWVL(NEXT(vl)); NEXT(vl)->v = y;
        !           669:   NEXT(NEXT(vl)) =0;
        !           670:   simp_ff((Obj)f,&obj); f = (P)obj;
        !           671:   n = QTOS(DEG(DC(f)));
        !           672:   wf = W_UMALLOC(n); wf1 = W_UMALLOC(n); wf2 = W_UMALLOC(n);
        !           673:   wfs = W_UMALLOC(n); gcd = W_UMALLOC(n);
        !           674:   q = field_order_sf();
        !           675:   lc = DC(f)->c;
        !           676:   for ( j = 0, fn = n + 1, index = 0;
        !           677:     index < q && j < count && fn > 1; index++ ) {
        !           678:     indextogfs(index,&m);
        !           679:     substp(vl,lc,y,(P)m,&lc0);
        !           680:     if ( lc0 ) {
        !           681:       substp(vl,f,y,(P)m,&f0);
        !           682:       ptosfum(f0,wf); cpyum(wf,wf1);
        !           683:       diffsfum(wf1,wf2); gcdsfum(wf1,wf2,gcd);
        !           684:       if ( DEG(gcd) == 0 ) {
        !           685:         ufctrsf(f0,&dc);
        !           686:         for ( dct = NEXT(dc), i = 0; dct; dct = NEXT(dct), i++ );
        !           687:         if ( i < fn ) {
        !           688:           dc0 = dc; fn = i; fm = m;
        !           689:         }
        !           690:         j++;
        !           691:       }
        !           692:     }
        !           693:   }
        !           694:   if ( index == q )
        !           695:     return 0;
        !           696:   else if ( fn == 1 )
        !           697:     return 1;
        !           698:   else {
        !           699:     *dcp = dc0;
        !           700:     *ev = fm;
        !           701:     return fn;
        !           702:   }
1.3       noro      703: }
                    704:
1.18      noro      705: void sfgcdgen(P f,ML blist,ML *clistp)
1.3       noro      706: {
1.34    ! noro      707:   int i;
        !           708:   int n,d,np;
        !           709:   UM wf,wm,wx,wy,wu,wv,wa,wb,wg,q,tum;
        !           710:   UM *in,*out;
        !           711:   ML clist;
        !           712:
        !           713:   n = UDEG(f); np = blist->n;
        !           714:   d = 2*n;
        !           715:   q = W_UMALLOC(d); wf = W_UMALLOC(d);
        !           716:   wm = W_UMALLOC(d); wx = W_UMALLOC(d);
        !           717:   wy = W_UMALLOC(d); wu = W_UMALLOC(d);
        !           718:   wv = W_UMALLOC(d); wg = W_UMALLOC(d);
        !           719:   wa = W_UMALLOC(d); wb = W_UMALLOC(d);
        !           720:   ptosfum(f,wf); DEG(wg) = 0; COEF(wg)[0] = _onesf();
        !           721:   *clistp = clist = MLALLOC(np); clist->n = np;
        !           722:   for ( i = 0, in = (UM *)blist->c, out = (UM *)clist->c; i < np; i++ ) {
        !           723:     divsfum(wf,in[i],q); tum = wf; wf = q; q = tum;
        !           724:     cpyum(wf,wx); cpyum(in[i],wy);
        !           725:     eucsfum(wx,wy,wa,wb); mulsfum(wa,wg,wm);
        !           726:     DEG(wm) = divsfum(wm,in[i],q); out[i] = UMALLOC(DEG(wm));
        !           727:     cpyum(wm,out[i]); mulsfum(q,wf,wu);
        !           728:     mulsfum(wg,wb,wv); addsfum(wu,wv,wg);
        !           729:   }
1.3       noro      730: }
                    731:
1.14      noro      732: /* f = g0*h0 mod y -> f = gk*hk mod y^(dy+1), f is replaced by hk */
1.3       noro      733:
1.18      noro      734: void sfhenmain2(BM f,UM g0,UM h0,int dy,BM *gp)
1.4       noro      735: {
1.34    ! noro      736:   int i,k;
        !           737:   int dx;
        !           738:   UM wt,wa,wb,q,w1,w2,wh1,wg1,ws;
        !           739:   UM wc,wd,we,wz;
        !           740:   BM wb0,wb1;
        !           741:   int dg,dh;
        !           742:   BM fk,gk,hk;
        !           743:
        !           744:   if ( DEG(f) < dy )
        !           745:     error("sfhenmain2 : invalid input");
        !           746:
        !           747:   dx = degbm(f);
        !           748:   dg = DEG(g0);
        !           749:   dh = DEG(h0);
        !           750:
        !           751:   W_BMALLOC(dx,dy,wb0); W_BMALLOC(dx,dy,wb1);
        !           752:   wt = W_UMALLOC(dx); ws = W_UMALLOC(dx); q = W_UMALLOC(2*dx);
        !           753:   wg1 = W_UMALLOC(2*dx); wh1 = W_UMALLOC(2*dx);
        !           754:
        !           755:   /* fk = gk*hk mod y^k */
        !           756:   W_BMALLOC(dx,dy,fk);
        !           757:   cpyum(COEF(f)[0],COEF(fk)[0]);
        !           758:   gk = BMALLOC(dg,dy);
        !           759:   cpyum(g0,COEF(gk)[0]);
        !           760:   W_BMALLOC(dh,dy,hk);
        !           761:   cpyum(h0,COEF(hk)[0]);
        !           762:
        !           763:   wc = W_UMALLOC(2*dx); wd = W_UMALLOC(2*dx);
        !           764:   we = W_UMALLOC(2*dx); wz = W_UMALLOC(2*dx);
        !           765:
        !           766:   /* compute wa,wb s.t. wa*g0+wb*h0 = 1 mod y */
        !           767:   w1 = W_UMALLOC(dg); cpyum(g0,w1);
        !           768:   w2 = W_UMALLOC(dh); cpyum(h0,w2);
        !           769:   wa = W_UMALLOC(2*dx); wb = W_UMALLOC(2*dx);  /* XXX */
        !           770:   eucsfum(w1,w2,wa,wb);
        !           771:
        !           772:   if ( debug_sfbfctr)
        !           773:     fprintf(stderr,"dy=%d\n",dy);
        !           774:   for ( k = 1; k <= dy; k++ ) {
        !           775:     if ( debug_sfbfctr)
        !           776:       fprintf(stderr,".");
        !           777:
        !           778:     /* at this point, f = gk*hk mod y^k */
        !           779:
        !           780:     /* clear wt */
        !           781:     clearum(wt,dx);
        !           782:
        !           783:     /* wt = (f-gk*hk)/y^k */
        !           784:     subsfum(COEF(f)[k],COEF(fk)[k],wt);
        !           785:
        !           786:     /* compute wf1,wg1 s.t. wh1*g0+wg1*h0 = wt */
        !           787:     mulsfum(wa,wt,wh1); DEG(wh1) = divsfum(wh1,h0,q);
        !           788:     mulsfum(wh1,g0,wc); subsfum(wt,wc,wd); DEG(wd) = divsfum(wd,h0,wg1);
1.4       noro      789:
1.34    ! noro      790:     /* check */
1.4       noro      791: #if 0
1.34    ! noro      792:     if ( DEG(wd) >= 0 || DEG(wg1) > ng )
        !           793:       error("henmain2 : cannot happen(adj)");
1.4       noro      794:
1.34    ! noro      795:     mulsfum(wg1,h0,wc); mulsfum(wh1,g0,wd); addsfum(wc,wd,we);
        !           796:     subsfum(we,wt,wz);
        !           797:     if ( DEG(wz) >= 0 )
        !           798:       error("henmain2 : cannot happen");
1.4       noro      799: #endif
                    800:
1.34    ! noro      801:     /* fk += ((wg1*hk+wh1*gk)*y^k+wg1*wh1*y^(2*k) mod y^(dy+1) */
        !           802:     /* wb0 = wh1*y^k */
        !           803:     clearbm(dx,wb0);
        !           804:     cpyum(wh1,COEF(wb0)[k]);
        !           805:
        !           806:     /* wb1 = gk*wb0 mod y^(dy+1) */
        !           807:     clearbm(dx,wb1);
        !           808:     mulsfbm(gk,wb0,wb1);
        !           809:     /* fk += wb1 */
        !           810:     addtosfbm(wb1,fk);
        !           811:
        !           812:     /* wb0 = wg1*y^k */
        !           813:     clearbm(dx,wb0);
        !           814:     cpyum(wg1,COEF(wb0)[k]);
        !           815:
        !           816:     /* wb1 = hk*wb0 mod y^(dy+1) */
        !           817:     clearbm(dx,wb1);
        !           818:     mulsfbm(hk,wb0,wb1);
        !           819:     /* fk += wb1 */
        !           820:     addtosfbm(wb1,fk);
        !           821:
        !           822:     /* fk += wg1*wh1*y^(2*k) mod y^(dy+1) */
        !           823:     if ( 2*k <= dy ) {
        !           824:       mulsfum(wg1,wh1,wt); addsfum(COEF(fk)[2*k],wt,ws);
        !           825:       cpyum(ws,COEF(fk)[2*k]);
        !           826:     }
        !           827:
        !           828:     /* gk += wg1*y^k, hk += wh1*y^k */
        !           829:     cpyum(wg1,COEF(gk)[k]);
        !           830:     cpyum(wh1,COEF(hk)[k]);
        !           831:   }
        !           832:   if ( debug_sfbfctr)
        !           833:     fprintf(stderr,"\n");
        !           834:   *gp = gk;
        !           835:   DEG(f) = dy;
        !           836:   for ( i = 0; i <= dy; i++ )
        !           837:     cpyum(COEF(hk)[i],COEF(f)[i]);
1.27      noro      838: }
                    839:
                    840: /* a0*g+b0*h = 1 mod y -> a*g+b*h = 1 mod y^(dy+1) */
                    841:
                    842: void sfexgcd_by_hensel(BM g,BM h,int dy,BM *ap,BM *bp)
                    843: {
1.34    ! noro      844:   int i,k;
        !           845:   int dx;
        !           846:   UM wt,wa,wb,q,w1,w2,ws;
        !           847:   UM wc,wd,we,wz,wa1,wb1;
        !           848:   BM wz0,wz1;
        !           849:   int dg,dh;
        !           850:   BM a,b,c;
        !           851:
        !           852:   dg = degbm(g);
        !           853:   dh = degbm(h);
        !           854:   dx = dg+dh;
        !           855:
        !           856:   a = BMALLOC(dh,dy);
        !           857:   b = BMALLOC(dg,dy);
        !           858:   /* c holds a*g+b*h-1 */
        !           859:   c = BMALLOC(dg+dh,dy);
        !           860:
        !           861:   W_BMALLOC(dx,dy,wz0); W_BMALLOC(dx,dy,wz1);
        !           862:
        !           863:   wt = W_UMALLOC(dx); ws = W_UMALLOC(dx); q = W_UMALLOC(2*dx);
        !           864:   wa1 = W_UMALLOC(2*dx); wb1 = W_UMALLOC(2*dx);
        !           865:   wc = W_UMALLOC(2*dx); wd = W_UMALLOC(2*dx);
        !           866:   we = W_UMALLOC(2*dx); wz = W_UMALLOC(2*dx);
        !           867:
        !           868:   /* compute wa,wb s.t. wa*g0+wb*h0 = 1 mod y */
        !           869:   w1 = W_UMALLOC(dg); cpyum(COEF(g)[0],w1);
        !           870:   w2 = W_UMALLOC(dh); cpyum(COEF(h)[0],w2);
        !           871:   wa = W_UMALLOC(2*dx); wb = W_UMALLOC(2*dx);  /* XXX */
        !           872:   eucsfum(w1,w2,wa,wb);
        !           873:   cpyum(wa,COEF(a)[0]); cpyum(wb,COEF(b)[0]);
        !           874:
        !           875:   /* initialize c to a*g+b*h-1 */
        !           876:   mulsfbm(a,g,c); mulsfbm(b,h,wz0); addtosfbm(wz0,c);
        !           877:   COEF(COEF(c)[0])[0] = 0;
        !           878:
        !           879:   if ( debug_sfbfctr)
        !           880:     fprintf(stderr,"dy=%d\n",dy);
        !           881:   for ( k = 1; k <= dy; k++ ) {
        !           882:     if ( debug_sfbfctr)
        !           883:       fprintf(stderr,".");
        !           884:
        !           885:     /* at this point, a*g+b*h = 1 mod y^k, c = a*g+b*h-1 */
        !           886:
        !           887:     /* wt = -((a*g+b*h-1)/y^k) */
        !           888:     cpyum(COEF(c)[k],wt);
        !           889:     for ( i = DEG(wt); i >= 0; i-- )
        !           890:       COEF(wt)[i] = _chsgnsf(COEF(wt)[i]);
        !           891:
        !           892:     /* compute wa1,wb1 s.t. wa1*g0+wb1*h0 = wt */
        !           893:     mulsfum(wa,wt,wa1); DEG(wa1) = divsfum(wa1,COEF(h)[0],q);
        !           894:     mulsfum(wa1,COEF(g)[0],wc); subsfum(wt,wc,wd);
        !           895:     DEG(wd) = divsfum(wd,COEF(h)[0],wb1);
        !           896:
        !           897:     /* c += ((wa1*g+wb1*h)*y^k mod y^(dy+1) */
        !           898:     /* wz0 = wa1*y^k */
        !           899:     clearbm(dx,wz0);
        !           900:     cpyum(wa1,COEF(wz0)[k]);
        !           901:
        !           902:     /* wz1 = wz0*g mod y^(dy+1) */
        !           903:     clearbm(dx,wz1);
        !           904:     mulsfbm(g,wz0,wz1);
        !           905:     /* c += wz1 */
        !           906:     addtosfbm(wz1,c);
        !           907:
        !           908:     /* wz0 = wb1*y^k */
        !           909:     clearbm(dx,wz0);
        !           910:     cpyum(wb1,COEF(wz0)[k]);
        !           911:
        !           912:     /* wz1 = wz0*h mod y^(dy+1) */
        !           913:     clearbm(dx,wz1);
        !           914:     mulsfbm(h,wz0,wz1);
        !           915:     /* c += wz1 */
        !           916:     addtosfbm(wz1,c);
        !           917:
        !           918:     /* a += wa1*y^k, b += wb1*y^k */
        !           919:     cpyum(wa1,COEF(a)[k]);
        !           920:     cpyum(wb1,COEF(b)[k]);
        !           921:   }
        !           922:   if ( debug_sfbfctr)
        !           923:     fprintf(stderr,"\n");
        !           924:   DEG(a) = dy;
        !           925:   DEG(b) = dy;
        !           926:   *ap = a;
        !           927:   *bp = b;
1.3       noro      928: }
                    929:
1.5       noro      930: /* fl->c[i] = coef_y(f,i) */
                    931:
1.18      noro      932: void ptosfbm(int dy,P f,BM fl)
1.5       noro      933: {
1.34    ! noro      934:   DCP dc;
        !           935:   int d,i,dx;
        !           936:   UM t;
        !           937:
        !           938:   dx = QTOS(DEG(DC(f)));
        !           939:   if ( DEG(fl) < dy )
        !           940:     error("ptosfbm : invalid input");
        !           941:   DEG(fl) = dy;
        !           942:   clearbm(dx,fl);
        !           943:   t = UMALLOC(dy);
        !           944:   for ( dc = DC(f); dc; dc = NEXT(dc) ) {
        !           945:     d = QTOS(DEG(dc));
        !           946:     ptosfum(COEF(dc),t);
        !           947:     for ( i = 0; i <= DEG(t); i++ )
        !           948:       COEF(COEF(fl)[i])[d] = COEF(t)[i];
        !           949:   }
        !           950:   for ( i = 0; i <= dy; i++ )
        !           951:     degum(COEF(fl)[i],dx);
1.5       noro      952: }
                    953:
                    954: /* x : main variable */
                    955:
1.18      noro      956: void sfbmtop(BM f,V x,V y,P *fp)
1.5       noro      957: {
1.34    ! noro      958:   UM *c;
        !           959:   int i,j,d,a,dy;
        !           960:   GFS b;
        !           961:   DCP dc0,dc,dct;
        !           962:
        !           963:   dy = DEG(f);
        !           964:   c = COEF(f);
        !           965:   d = degbm(f);
        !           966:
        !           967:   dc0 = 0;
        !           968:   for ( i = 0; i <= d; i++ ) {
        !           969:     dc = 0;
        !           970:     for ( j = 0; j <= dy; j++ ) {
        !           971:       if ( DEG(c[j]) >= i && (a = COEF(c[j])[i]) ) {
        !           972:         NEWDC(dct);
        !           973:         STOQ(j,DEG(dct));
        !           974:         iftogfs(a,&b);
        !           975:         COEF(dct) = (P)b;
        !           976:         NEXT(dct) = dc;
        !           977:         dc = dct;
        !           978:       }
        !           979:     }
        !           980:     if ( dc ) {
        !           981:       NEWDC(dct);
        !           982:       STOQ(i,DEG(dct));
        !           983:       MKP(y,dc,COEF(dct));
        !           984:       NEXT(dct) = dc0;
        !           985:       dc0 = dct;
        !           986:     }
        !           987:   }
        !           988:   if ( dc0 )
        !           989:     MKP(x,dc0,*fp);
        !           990:   else
        !           991:     *fp = 0;
1.8       noro      992: }
                    993:
1.18      noro      994: void sfsqfr(P f,DCP *dcp)
1.14      noro      995: {
1.34    ! noro      996:   Obj obj;
        !           997:   DCP dc;
        !           998:   VL vl;
        !           999:
        !          1000:   simp_ff((Obj)f,&obj); f = (P)obj;
        !          1001:   clctv(CO,f,&vl);
        !          1002:   if ( !vl ) {
        !          1003:     /* f is a const */
        !          1004:     NEWDC(dc); DEG(dc) = ONE; COEF(dc) = f; NEXT(dc) = 0; *dcp = dc;
        !          1005:   } else if ( !NEXT(vl) )
        !          1006:     sfusqfr(f,dcp);
        !          1007:   else
        !          1008:     sqfrsf(vl,f,dcp);
1.14      noro     1009: }
                   1010:
1.18      noro     1011: void sfusqfr(P f,DCP *dcp)
1.14      noro     1012: {
1.34    ! noro     1013:   DCP dc,dct;
        !          1014:   struct oDUM *udc;
        !          1015:   V x;
        !          1016:   P lc;
        !          1017:   int n,i;
        !          1018:   UM mf;
        !          1019:
        !          1020:   x = VR(f);
        !          1021:   n = getdeg(x,f);
        !          1022:   mf = W_UMALLOC(n);
        !          1023:   ptosfum(f,mf);
        !          1024:   lc = COEF(DC(f));
        !          1025:   if ( !_isonesf(COEF(mf)[n]) ) {
        !          1026:     monicsfum(mf);
        !          1027:   }
        !          1028:   W_CALLOC(n+1,struct oDUM,udc);
        !          1029:   gensqfrsfum(mf,udc);
        !          1030:   for ( i = 0, dc = 0; udc[i].f; i++ ) {
        !          1031:     NEWDC(dct); STOQ(udc[i].n,DEG(dct));
        !          1032:     sfumtop(x,udc[i].f,&COEF(dct));
        !          1033:     NEXT(dct) = dc; dc = dct;
        !          1034:   }
        !          1035:   NEWDC(dct); DEG(dct) = ONE; COEF(dct) = (P)lc; NEXT(dct) = dc;
        !          1036:   *dcp = dct;
1.14      noro     1037: }
                   1038:
1.24      noro     1039: #if 0
1.21      noro     1040: void sfbsqfrmain(P f,V x,V y,DCP *dcp)
                   1041: {
1.34    ! noro     1042:   /* XXX*/
1.21      noro     1043: }
                   1044:
                   1045: /* f is bivariate */
                   1046:
1.18      noro     1047: void sfbsqfr(P f,V x,V y,DCP *dcp)
1.14      noro     1048: {
1.34    ! noro     1049:   P t,rf,cx,cy;
        !          1050:   VL vl,rvl;
        !          1051:   DCP dcx,dcy,dct,dc;
        !          1052:   struct oVL vl0,vl1;
        !          1053:
        !          1054:   /* cy(y) = cont(f,x), f /= cy */
        !          1055:   cont_pp_sfp(vl,f,&cy,&t); f = t;
        !          1056:   /* rvl = [y,x] */
        !          1057:   reordvar(vl,y,&rvl); reorderp(rvl,vl,f,&rf);
        !          1058:   /* cx(x) = cont(rf,y), Rf /= cy */
        !          1059:   cont_pp_sfp(rvl,rf,&cx,&t); rf = t;
        !          1060:   reorderp(vl,rvl,rf,&f);
        !          1061:
        !          1062:   /* f -> cx*cy*f */
        !          1063:   sfsqfr(cx,&dcx); dcx = NEXT(dcx);
        !          1064:   sfsqfr(cy,&dcy); dcy = NEXT(dcy);
        !          1065:   if ( dcx ) {
        !          1066:     for ( dct = dcx; NEXT(dct); dct = NEXT(dct) );
        !          1067:     NEXT(dct) = dcy;
        !          1068:   } else
        !          1069:     dcx = dcy;
        !          1070:   if ( OID(f) == O_N )
        !          1071:     *dcp = dcx;
        !          1072:   else {
        !          1073:     /* f must be bivariate */
        !          1074:     sfbsqfrmain(f,x,y,&dc);
        !          1075:     if ( dcx ) {
        !          1076:       for ( dct = dcx; NEXT(dct); dct = NEXT(dct) );
        !          1077:       NEXT(dct) = dc;
        !          1078:     } else
        !          1079:       dcx = dc;
        !          1080:     *dcp = dcx;
        !          1081:   }
1.14      noro     1082: }
1.24      noro     1083: #endif
1.14      noro     1084:
1.8       noro     1085: void sfdtest(P,ML,V,V,DCP *);
                   1086:
1.21      noro     1087: /* if degbound >= 0 find factor s.t. deg_x(factor) <= degbound */
                   1088:
                   1089: void sfbfctr(P f,V x,V y,int degbound,DCP *dcp)
1.8       noro     1090: {
1.34    ! noro     1091:   ML list;
        !          1092:   P sf;
        !          1093:   GFS ev;
        !          1094:   DCP dc,dct;
        !          1095:   BM fl;
        !          1096:   int dx,dy;
        !          1097:
        !          1098:   /* sf(x) = f(x+ev) = list->c[0]*list->c[1]*... */
        !          1099:   sfhensel(5,f,x,y,degbound,&ev,&sf,&list);
        !          1100:   if ( list->n == 0 )
        !          1101:     error("sfbfctr : short of evaluation points");
        !          1102:   else if ( list->n == 1 ) {
        !          1103:     /* f is irreducible */
        !          1104:     NEWDC(dc); DEG(dc) = ONE; COEF(dc) = f; NEXT(dc) = 0;
        !          1105:     *dcp = dc;
        !          1106:     return;
        !          1107:   }
        !          1108:   sfdtest(sf,list,x,y,&dc);
        !          1109:   if ( ev ) {
        !          1110:     dx = getdeg(x,sf);
        !          1111:     dy = getdeg(y,sf);
        !          1112:     W_BMALLOC(dx,dy,fl);
        !          1113:     for ( dct = dc; dct; dct = NEXT(dct) ) {
        !          1114:       ptosfbm(dy,COEF(dct),fl);
        !          1115:       shiftsfbm(fl,_chsgnsf(FTOIF(CONT(ev))));
        !          1116:       sfbmtop(fl,x,y,&COEF(dct));
        !          1117:     }
        !          1118:   }
        !          1119:   *dcp = dc;
1.26      noro     1120: }
                   1121:
                   1122: /* returns shifted f, shifted factors and the eval pt */
                   1123:
                   1124: void sfbfctr_shift(P f,V x,V y,int degbound,GFS *evp,P *sfp,DCP *dcp)
                   1125: {
1.34    ! noro     1126:   ML list;
        !          1127:   P sf;
        !          1128:   GFS ev;
        !          1129:   DCP dc,dct;
        !          1130:   int dx,dy;
        !          1131:
        !          1132:   /* sf(x) = f(x+ev) = list->c[0]*list->c[1]*... */
        !          1133:   sfhensel(5,f,x,y,degbound,&ev,&sf,&list);
        !          1134:   if ( list->n == 0 )
        !          1135:     error("sfbfctr_shift : short of evaluation points");
        !          1136:   else if ( list->n == 1 ) {
        !          1137:     /* f is irreducible */
        !          1138:     NEWDC(dc); DEG(dc) = ONE; COEF(dc) = f; NEXT(dc) = 0;
        !          1139:     *evp = 0;
        !          1140:     *sfp = f;
        !          1141:     *dcp = dc;
        !          1142:   } else {
        !          1143:     sfdtest(sf,list,x,y,dcp);
        !          1144:     *evp = ev;
        !          1145:     *sfp = sf;
        !          1146:   }
1.8       noro     1147: }
                   1148:
1.14      noro     1149: /* f = f(x,y) = list->c[0]*list->c[1]*... mod y^(list->bound+1) */
1.8       noro     1150:
1.18      noro     1151: void sfdtest(P f,ML list,V x,V y,DCP *dcp)
1.8       noro     1152: {
1.34    ! noro     1153:   int np,dx,dy;
        !          1154:   int i,j,k,bound;
        !          1155:   int *win;
        !          1156:   P g,lcg,factor,cofactor,lcyx;
        !          1157:   P csum;
        !          1158:   DCP dcf,dcf0,dc;
        !          1159:   BM *c;
        !          1160:   BM lcy;
        !          1161:   UM lcg0,lcy0,w;
        !          1162:   UM *d1c;
        !          1163:   ML wlist;
        !          1164:   struct oVL vl1,vl0;
        !          1165:   VL vl;
        !          1166:   int z,dt,dtok;
        !          1167:
        !          1168:   /* vl = [x,y] */
        !          1169:   vl0.v = x; vl0.next = &vl1; vl1.v = y; vl1.next = 0; vl = &vl0;
        !          1170:
        !          1171:   /* setup various structures and arrays */
        !          1172:   dx = getdeg(x,f);
        !          1173:   dy = getdeg(y,f);
        !          1174:   np = list->n;
        !          1175:   win = W_ALLOC(np+1);
        !          1176:   wlist = W_MLALLOC(np);
        !          1177:   wlist->n = list->n;
        !          1178:   bound = wlist->bound = list->bound;
        !          1179:   c = (BM *)COEF(wlist);
        !          1180:   bcopy((char *)COEF(list),(char *)c,(int)(sizeof(BM)*np));
        !          1181:
        !          1182:   lcg0 = W_UMALLOC(2*dy);
        !          1183:
        !          1184:   /* initialize g by f */
        !          1185:   g = f;
        !          1186:
        !          1187:   /* initialize lcg */
        !          1188:   mulp(vl,g,COEF(DC(g)),&lcg);
        !          1189:
        !          1190:   /* initialize lcg0 */
        !          1191:   const_term(lcg,lcg0);
        !          1192:
        !          1193:   /* initialize csum = lcg(1) */
        !          1194:   sfcsump(vl,lcg,&csum);
        !          1195:
        !          1196:   /* initialize lcy by LC(f) */
        !          1197:   W_BMALLOC(0,dy,lcy);
        !          1198:   NEWDC(dc); COEF(dc) = COEF(DC(g)); DEG(dc) = 0;
        !          1199:   NEWP(lcyx); VR(lcyx) = x; DC(lcyx) = dc;
        !          1200:   ptosfbm(dy,lcyx,lcy);
        !          1201:
        !          1202:   /* initialize lcy0 by LC(f) */
        !          1203:   lcy0 = W_UMALLOC(bound);
        !          1204:   ptosfum(COEF(DC(g)),lcy0);
        !          1205:
        !          1206:   /* ((d-1 coefs)*lcy0 */
        !          1207:   d1c = (UM *)W_ALLOC(np*sizeof(UM));
        !          1208:   w = W_UMALLOC(2*bound);
        !          1209:   for ( i = 1; i < np; i++ ) {
        !          1210:     extractcoefbm(c[i],degbm(c[i])-1,w);
        !          1211:     d1c[i] = W_UMALLOC(2*bound);
        !          1212:     mulsfum(w,lcy0,d1c[i]);
        !          1213:     /* d1c[i] = d1c[i] mod y^(bound+1) */
        !          1214:     if ( DEG(d1c[i]) > bound ) {
        !          1215:       for ( j = DEG(d1c[i]); j > bound; j-- )
        !          1216:         COEF(d1c[i])[j] = 0;
        !          1217:       degum(d1c[i],bound);
        !          1218:     }
        !          1219:   }
        !          1220:
        !          1221:   if ( debug_sfbfctr)
        !          1222:     fprintf(stderr,"np = %d\n",np);
        !          1223:   dtok = 0;
        !          1224:   for ( g = f, k = 1, dcf = dcf0 = 0, win[0] = 1, --np, z = 0; ; z++ ) {
        !          1225:     if ( debug_sfbfctr && !(z % 1000) ) fprintf(stderr,".");
        !          1226:     dt = sfdegtest(dy,bound,d1c,k,win);
        !          1227:     if ( dt )
        !          1228:       dtok++;
        !          1229:     if ( dt && sfdtestmain(vl,lcg,lcg0,lcy,csum,wlist,
        !          1230:         k,win,&factor,&cofactor) ) {
        !          1231:       NEXTDC(dcf0,dcf); DEG(dcf) = ONE; COEF(dcf) = factor;
        !          1232:       g = cofactor;
        !          1233:
        !          1234:       /* update lcg */
        !          1235:       mulp(vl,g,COEF(DC(g)),&lcg);
        !          1236:
        !          1237:       /* update lcg0 */
        !          1238:       const_term(lcg,lcg0);
        !          1239:
        !          1240:       /* update csum */
        !          1241:       sfcsump(vl,lcg,&csum);
        !          1242:
        !          1243:       /* update dy */
        !          1244:       dy = getdeg(y,g);
        !          1245:
        !          1246:       /* update lcy */
        !          1247:       clearbm(0,lcy);
        !          1248:       COEF(dc) = COEF(DC(g));
        !          1249:       ptosfbm(dy,lcyx,lcy);
        !          1250:
        !          1251:       for ( i = 0; i < k - 1; i++ )
        !          1252:         for ( j = win[i] + 1; j < win[i + 1]; j++ )
        !          1253:           c[j-i-1] = c[j];
        !          1254:       for ( j = win[k-1] + 1; j <= np; j++ )
        !          1255:           c[j-k] = c[j];
        !          1256:       if ( ( np -= k ) < k )
        !          1257:         break;
        !          1258:       if ( np - win[0] + 1 < k )
        !          1259:         if ( ++k > np )
        !          1260:           break;
        !          1261:         else
        !          1262:           for ( i = 0; i < k; i++ )
        !          1263:             win[i] = i + 1;
        !          1264:       else
        !          1265:         for ( i = 1; i < k; i++ )
        !          1266:           win[i] = win[0] + i;
        !          1267:
        !          1268:
        !          1269:       /* update lcy0  */
        !          1270:       ptosfum(COEF(DC(g)),lcy0);
        !          1271:
        !          1272:       /* update d-1 coeffs */
        !          1273:       for ( i = 1; i <= np; i++ ) {
        !          1274:         extractcoefbm(c[i],degbm(c[i])-1,w);
        !          1275:         mulsfum(w,lcy0,d1c[i]);
        !          1276:         /* d1c[i] = d1c[1] mod y^(bound+1) */
        !          1277:         if ( DEG(d1c[i]) > bound ) {
        !          1278:           for ( j = DEG(d1c[i]); j > bound; j-- )
        !          1279:             COEF(d1c[i])[j] = 0;
        !          1280:           degum(d1c[i],bound);
        !          1281:         }
        !          1282:       }
        !          1283:     } else if ( !ncombi(1,np,k,win) )
        !          1284:       if ( k == np )
        !          1285:         break;
        !          1286:       else
        !          1287:         for ( i = 0, ++k; i < k; i++ )
        !          1288:           win[i] = i + 1;
        !          1289:   }
        !          1290:   if ( debug_sfbfctr )
        !          1291:     fprintf(stderr,"total %d, omitted by degtest %d\n",z,z-dtok);
        !          1292:   NEXTDC(dcf0,dcf); COEF(dcf) = g;
        !          1293:   DEG(dcf) = ONE; NEXT(dcf) = 0; *dcp = dcf0;
1.8       noro     1294: }
                   1295:
1.20      noro     1296: void extractcoefbm(BM f,int dx,UM r)
1.19      noro     1297: {
1.34    ! noro     1298:   int j;
        !          1299:   UM fj;
1.19      noro     1300:
1.34    ! noro     1301:   for ( j = DEG(f); j >= 0; j-- ) {
        !          1302:     fj = COEF(f)[j];
        !          1303:     if ( fj && DEG(fj) >= dx ) {
        !          1304:       COEF(r)[j] = COEF(fj)[dx];
        !          1305:     } else
        !          1306:       COEF(r)[j] = 0;
        !          1307:   }
        !          1308:   degum(r,DEG(f));
1.19      noro     1309: }
                   1310:
                   1311: /* deg_y(prod mod y^(bound+1)) <= dy ? */
                   1312:
1.20      noro     1313: int sfdegtest(int dy,int bound,UM *d1c,int k,int *in)
1.19      noro     1314: {
1.34    ! noro     1315:   int i,j;
        !          1316:   UM w,w1,wt;
        !          1317:   BM t;
        !          1318:
        !          1319:   w = W_UMALLOC(bound);
        !          1320:   w1 = W_UMALLOC(bound);
        !          1321:   clearum(w,bound);
        !          1322:   for ( i = 0; i < k; i++ ) {
        !          1323:     addsfum(w,d1c[in[i]],w1); wt = w; w = w1; w1 = wt;
        !          1324:   }
        !          1325:   return DEG(w) <= dy ? 1 : 0;
1.19      noro     1326: }
                   1327:
1.9       noro     1328: /* lcy = LC(g), lcg = lcy*g, lcg0 = const part of lcg */
1.19      noro     1329:
1.18      noro     1330: int sfdtestmain(VL vl,P lcg,UM lcg0,BM lcy,P csum,ML list,
1.34    ! noro     1331:   int k,int *in,P *fp,P *cofp)
1.8       noro     1332: {
1.34    ! noro     1333:   P fmul,csumg,q,cont;
        !          1334:   V x,y;
1.8       noro     1335:
1.34    ! noro     1336:   x = vl->v;
        !          1337:   y = vl->next->v;
        !          1338:   if (!sfctest(lcg0,lcy,list,k,in))
        !          1339:     return 0;
        !          1340:   mulsfbmarray(UDEG(lcg),lcy,list,k,in,x,y,&fmul);
        !          1341:   if ( csum ) {
        !          1342:     sfcsump(vl,fmul,&csumg);
        !          1343:     if ( csumg ) {
        !          1344:       if ( !divtp(vl,csum,csumg,&q) )
        !          1345:         return 0;
        !          1346:     }
        !          1347:   }
        !          1348:   if ( divtp_by_sfbm(vl,lcg,fmul,&q) ) {
        !          1349:     cont_pp_sfp(vl,fmul,&cont,fp);
        !          1350:     cont_pp_sfp(vl,q,&cont,cofp);
        !          1351:     return 1;
        !          1352:   } else
        !          1353:     return 0;
1.8       noro     1354: }
                   1355:
1.18      noro     1356: void const_term(P f,UM c)
1.9       noro     1357: {
1.34    ! noro     1358:   DCP dc;
1.9       noro     1359:
1.34    ! noro     1360:   for ( dc = DC(f); dc && DEG(dc); dc = NEXT(dc) );
        !          1361:   if ( dc )
        !          1362:     ptosfum(COEF(dc),c);
        !          1363:   else
        !          1364:     DEG(c) = -1;
1.9       noro     1365: }
                   1366:
1.18      noro     1367: void const_term_sfbm(BM f,UM c)
1.9       noro     1368: {
1.34    ! noro     1369:   int i,dy;
1.9       noro     1370:
1.34    ! noro     1371:   dy = DEG(f);
        !          1372:   for ( i = 0; i <= dy; i++ )
        !          1373:     if ( DEG(COEF(f)[i]) >= 0 )
        !          1374:       COEF(c)[i] = COEF(COEF(f)[i])[0];
        !          1375:     else
        !          1376:       COEF(c)[i] = 0;
        !          1377:   degum(c,dy);
1.9       noro     1378: }
                   1379:
                   1380: /* lcy*(product of const part) | lcg0 ? */
                   1381:
1.18      noro     1382: int sfctest(UM lcg0,BM lcy,ML list,int k,int *in)
1.8       noro     1383: {
1.34    ! noro     1384:   int dy,i,dr;
        !          1385:   UM t,s,u,w;
        !          1386:   BM *l;
        !          1387:
        !          1388:   dy = list->bound;
        !          1389:   t = W_UMALLOC(2*dy);
        !          1390:   s = W_UMALLOC(2*dy);
        !          1391:   u = W_UMALLOC(2*dy);
        !          1392:   const_term_sfbm(lcy,t);
        !          1393:   if ( DEG(t) < 0 )
        !          1394:     return 1;
        !          1395:
        !          1396:   l = (BM *)list->c;
        !          1397:   for ( i = 0; i < k; i++ ) {
        !          1398:     const_term_sfbm(l[in[i]],s);
        !          1399:     mulsfum(t,s,u);
        !          1400:     if ( DEG(u) > dy )
        !          1401:       degum(u,dy);
        !          1402:     w = t; t = u; u = w;
        !          1403:   }
        !          1404:   cpyum(lcg0,s);
        !          1405:   dr = divsfum(s,t,u);
        !          1406:   if ( dr >= 0 )
        !          1407:     return 0;
        !          1408:   else
        !          1409:     return 1;
1.8       noro     1410: }
                   1411:
                   1412: /* main var of f is x */
                   1413:
1.18      noro     1414: void mulsfbmarray(int dx,BM lcy,ML list,int k,int *in,V x,V y,P *g)
1.8       noro     1415: {
1.34    ! noro     1416:   int dy,i;
        !          1417:   BM wb0,wb1,t;
        !          1418:   BM *l;
        !          1419:
        !          1420:   dy = list->bound;
        !          1421:   W_BMALLOC(dx,dy,wb0); W_BMALLOC(dx,dy,wb1);
        !          1422:   l = (BM *)list->c;
        !          1423:   clearbm(dx,wb0);
        !          1424:   mulsfbm(lcy,l[in[0]],wb0);
        !          1425:   for ( i = 1; i < k; i++ ) {
        !          1426:     clearbm(dx,wb1);
        !          1427:     mulsfbm(l[in[i]],wb0,wb1);
        !          1428:     t = wb0; wb0 = wb1; wb1 = t;
        !          1429:   }
        !          1430:   sfbmtop(wb0,x,y,g);
1.8       noro     1431: }
                   1432:
1.18      noro     1433: void sfcsump(VL vl,P f,P *s)
1.8       noro     1434: {
1.34    ! noro     1435:   P t,u;
        !          1436:   DCP dc;
1.8       noro     1437:
1.34    ! noro     1438:   for ( dc = DC(f), t = 0; dc; dc = NEXT(dc) ) {
        !          1439:     addp(vl,COEF(dc),t,&u); t = u;
        !          1440:   }
        !          1441:   *s = t;
1.8       noro     1442: }
                   1443:
                   1444: /* *fp = primitive part of f w.r.t. x */
                   1445:
1.18      noro     1446: void cont_pp_sfp(VL vl,P f,P *cp,P *fp)
1.8       noro     1447: {
1.34    ! noro     1448:   V x,y;
        !          1449:   int d;
        !          1450:   UM t,s,gcd;
        !          1451:   DCP dc;
        !          1452:   GFS g;
        !          1453:
        !          1454:   x = vl->v;
        !          1455:   y = vl->next->v;
        !          1456:   d = getdeg(y,f);
        !          1457:   if ( d == 0 ) {
        !          1458:     itogfs(1,&g);
        !          1459:     *cp = (P)g;
        !          1460:     *fp = f;  /* XXX */
        !          1461:   } else {
        !          1462:     t = W_UMALLOC(2*d);
        !          1463:     s = W_UMALLOC(2*d);
        !          1464:     gcd = W_UMALLOC(2*d);
        !          1465:     dc = DC(f);
        !          1466:     ptosfum(COEF(dc),gcd);
        !          1467:     for ( dc = NEXT(dc); dc; dc = NEXT(dc) ) {
        !          1468:       ptosfum(COEF(dc),t);
        !          1469:       gcdsfum(gcd,t,s);
        !          1470:       cpyum(s,gcd);
        !          1471:     }
        !          1472:     sfumtop(y,gcd,cp);
        !          1473:     divsp(vl,f,*cp,fp);
        !          1474:   }
1.11      noro     1475: }
                   1476:
1.18      noro     1477: int divtp_by_sfbm(VL vl,P f,P g,P *qp)
1.11      noro     1478: {
1.34    ! noro     1479:   V x,y;
        !          1480:   int fx,fy,gx,gy;
        !          1481:   BM fl,gl,ql;
        !          1482:   UM *cf,*cg,*cq;
        !          1483:   UM hg,q,t,s;
        !          1484:   int i,j,dr;
        !          1485:
        !          1486:   x = vl->v; y = vl->next->v;
        !          1487:   fx = getdeg(x,f); fy = getdeg(y,f);
        !          1488:   gx = getdeg(x,g); gy = getdeg(y,g);
        !          1489:
        !          1490:   if ( fx < gx || fy < gy )
        !          1491:     return 0;
        !          1492:   W_BMALLOC(fx,fy,fl); ptosfbm(fy,f,fl); cf = COEF(fl);
        !          1493:   W_BMALLOC(gx,gy,gl); ptosfbm(gy,g,gl); cg = COEF(gl);
        !          1494:   W_BMALLOC(fx-gx,fy-gy,ql); cq = COEF(ql);
        !          1495:
        !          1496:   hg = cg[gy];
        !          1497:   q = W_UMALLOC(fx); t = W_UMALLOC(fx); s = W_UMALLOC(fx);
        !          1498:
        !          1499:   for ( i = fy; i >= gy; i-- ) {
        !          1500:     if ( DEG(cf[i]) < 0 )
        !          1501:       continue;
        !          1502:     dr = divsfum(cf[i],hg,q);
        !          1503:     if ( dr >= 0 )
        !          1504:       return 0;
        !          1505:     if ( DEG(q) > fx-gx )
        !          1506:       return 0;
        !          1507:     cpyum(q,cq[i-gy]);
        !          1508:     for ( j = 0; j <= gy; j++ ) {
        !          1509:       mulsfum(cg[j],q,t);
        !          1510:       subsfum(cf[j+i-gy],t,s);
        !          1511:       cpyum(s,cf[j+i-gy]);
        !          1512:     }
        !          1513:   }
        !          1514:   for ( j = gy-1; j >= 0 && DEG(cf[j]) < 0; j-- );
        !          1515:   if ( j >= 0 )
        !          1516:     return 0;
        !          1517:   sfbmtop(ql,x,y,qp);
        !          1518:   return 1;
1.16      noro     1519: }
                   1520:
                   1521: /* XXX generate an irreducible poly of degree n */
                   1522:
1.17      noro     1523: extern int current_gfs_q1;
1.22      noro     1524: extern int *current_gfs_ntoi;
1.17      noro     1525:
1.18      noro     1526: void generate_defpoly_sfum(int n,UM *dp)
1.16      noro     1527: {
1.34    ! noro     1528:   UM r,dr,t,g;
        !          1529:   UM *f;
        !          1530:   int *c,*w;
        !          1531:   int max,i,j;
        !          1532:
        !          1533:   *dp = r = UMALLOC(n);
        !          1534:   DEG(r) = n;
        !          1535:   c = COEF(r);
        !          1536:   c[n] = _onesf();
        !          1537:   max = current_gfs_q1;
        !          1538:   w = (int *)ALLOCA(n*sizeof(int));
        !          1539:   bzero(w,n*sizeof(int));
        !          1540:
        !          1541:   dr = W_UMALLOC(n); t = W_UMALLOC(n); g = W_UMALLOC(n);
        !          1542:   f = (UM *)ALLOCA((n+1)*sizeof(UM));
        !          1543:   while ( 1 ) {
        !          1544:     for ( i = 0; i < n && w[i] == max; i++ );
        !          1545:     if ( i == n ) {
        !          1546:       /* XXX cannot happen */
        !          1547:       error("generate_defpoly_sfum : cannot happen");
        !          1548:     }
        !          1549:     for ( j = 0; j < i; j++ )
        !          1550:       w[j] = 0;
        !          1551:     w[i]++;
        !          1552:     if ( !current_gfs_ntoi )
        !          1553:       for ( i = 0; i < n; i++ )
        !          1554:         c[i] = w[i]?FTOIF(w[i]):0;
        !          1555:     else
        !          1556:       for ( i = 0; i < n; i++ )
        !          1557:         c[i] = w[i]?FTOIF(w[i]-1):0;
        !          1558:     if ( !c[0] )
        !          1559:       continue;
        !          1560:     diffsfum(r,dr); cpyum(r,t); gcdsfum(t,dr,g);
        !          1561:     if ( DEG(g) > 0 )
        !          1562:       continue;
        !          1563:
        !          1564:     czsfum(r,f);
        !          1565:     for ( i = 0; f[i]; i++ );
        !          1566:     if ( i == 1 )
        !          1567:       return;
        !          1568:   }
1.3       noro     1569: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>