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

Diff for /OpenXM_contrib2/asir2000/builtin/poly.c between version 1.1.1.1 and 1.17

version 1.1.1.1, 1999/12/03 07:39:07 version 1.17, 2002/09/27 04:42:59
Line 1 
Line 1 
 /* $OpenXM: OpenXM/src/asir99/builtin/poly.c,v 1.1.1.1 1999/11/10 08:12:26 noro Exp $ */  /*
    * Copyright (c) 1994-2000 FUJITSU LABORATORIES LIMITED
    * All rights reserved.
    *
    * FUJITSU LABORATORIES LIMITED ("FLL") hereby grants you a limited,
    * non-exclusive and royalty-free license to use, copy, modify and
    * redistribute, solely for non-commercial and non-profit purposes, the
    * computer program, "Risa/Asir" ("SOFTWARE"), subject to the terms and
    * conditions of this Agreement. For the avoidance of doubt, you acquire
    * only a limited right to use the SOFTWARE hereunder, and FLL or any
    * third party developer retains all rights, including but not limited to
    * copyrights, in and to the SOFTWARE.
    *
    * (1) FLL does not grant you a license in any way for commercial
    * purposes. You may use the SOFTWARE only for non-commercial and
    * non-profit purposes only, such as academic, research and internal
    * business use.
    * (2) The SOFTWARE is protected by the Copyright Law of Japan and
    * international copyright treaties. If you make copies of the SOFTWARE,
    * with or without modification, as permitted hereunder, you shall affix
    * to all such copies of the SOFTWARE the above copyright notice.
    * (3) An explicit reference to this SOFTWARE and its copyright owner
    * shall be made on your publication or presentation in any form of the
    * results obtained by use of the SOFTWARE.
    * (4) In the event that you modify the SOFTWARE, you shall notify FLL by
    * e-mail at risa-admin@sec.flab.fujitsu.co.jp of the detailed specification
    * for such modification or the source code of the modified part of the
    * SOFTWARE.
    *
    * THE SOFTWARE IS PROVIDED AS IS WITHOUT ANY WARRANTY OF ANY KIND. FLL
    * MAKES ABSOLUTELY NO WARRANTIES, EXPRESSED, IMPLIED OR STATUTORY, AND
    * EXPRESSLY DISCLAIMS ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS
    * FOR A PARTICULAR PURPOSE OR NONINFRINGEMENT OF THIRD PARTIES'
    * RIGHTS. NO FLL DEALER, AGENT, EMPLOYEES IS AUTHORIZED TO MAKE ANY
    * MODIFICATIONS, EXTENSIONS, OR ADDITIONS TO THIS WARRANTY.
    * UNDER NO CIRCUMSTANCES AND UNDER NO LEGAL THEORY, TORT, CONTRACT,
    * OR OTHERWISE, SHALL FLL BE LIABLE TO YOU OR ANY OTHER PERSON FOR ANY
    * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, PUNITIVE OR CONSEQUENTIAL
    * DAMAGES OF ANY CHARACTER, INCLUDING, WITHOUT LIMITATION, DAMAGES
    * ARISING OUT OF OR RELATING TO THE SOFTWARE OR THIS AGREEMENT, DAMAGES
    * FOR LOSS OF GOODWILL, WORK STOPPAGE, OR LOSS OF DATA, OR FOR ANY
    * DAMAGES, EVEN IF FLL SHALL HAVE BEEN INFORMED OF THE POSSIBILITY OF
    * SUCH DAMAGES, OR FOR ANY CLAIM BY ANY OTHER PARTY. EVEN IF A PART
    * OF THE SOFTWARE HAS BEEN DEVELOPED BY A THIRD PARTY, THE THIRD PARTY
    * DEVELOPER SHALL HAVE NO LIABILITY IN CONNECTION WITH THE USE,
    * PERFORMANCE OR NON-PERFORMANCE OF THE SOFTWARE.
    *
    * $OpenXM: OpenXM_contrib2/asir2000/builtin/poly.c,v 1.16 2001/10/09 01:36:06 noro Exp $
   */
 #include "ca.h"  #include "ca.h"
 #include "parse.h"  #include "parse.h"
 #include "base.h"  #include "base.h"
Line 11  void Pord(), Pcoef0(), Pcoef(), Pdeg(), Pmindeg(), Pse
Line 59  void Pord(), Pcoef0(), Pcoef(), Pdeg(), Pmindeg(), Pse
 void Pcoef_gf2n();  void Pcoef_gf2n();
 void getcoef(), getdeglist(), mergedeglist(), change_mvar(), restore_mvar();  void getcoef(), getdeglist(), mergedeglist(), change_mvar(), restore_mvar();
   
 void Pp_mag();  void Pp_mag(),Pmaxblen();
 void Pmergelist(), Pch_mv(), Pre_mv(), Pdeglist();  void Pmergelist(), Pch_mv(), Pre_mv(), Pdeglist();
 void Pptomp(),Pmptop();  void Pptomp(),Pmptop();
 void Pptolmp(),Plmptop();  void Pptolmp(),Plmptop();
   void Pptosfp(),Psfptop(),Psf_galois_action(),Psf_embed(),Psf_find_root();
   void Psf_minipoly(),Psf_log();
 void Pptogf2n(),Pgf2ntop(),Pgf2ntovect();  void Pptogf2n(),Pgf2ntop(),Pgf2ntovect();
 void Pptogfpn(),Pgfpntop();  void Pptogfpn(),Pgfpntop();
 void Pfind_root_gf2n();  void Pfind_root_gf2n();
   void Pumul_specialmod(),Pusquare_specialmod(),Putmul_specialmod();
   
 void Pureverse(),Putrunc(),Pudecomp(),Purembymul(),Purembymul_precomp();  void Pureverse(),Putrunc(),Pudecomp(),Purembymul(),Purembymul_precomp();
 void Puinvmod(),Purevinvmod();  void Puinvmod(),Purevinvmod();
Line 48  void Pbininv_gf2n();
Line 99  void Pbininv_gf2n();
 void Prinvtest_gf2n();  void Prinvtest_gf2n();
 void Pis_irred_gf2();  void Pis_irred_gf2();
 void Pis_irred_ddd_gf2();  void Pis_irred_ddd_gf2();
   void Pget_next_fft_prime();
   void Puadj_coef();
   void Preorder();
   void Phomogeneous_part();
   void Phomogeneous_deg();
 void simp_ff(Obj,Obj *);  void simp_ff(Obj,Obj *);
 void ranp(int,UP *);  void ranp(int,UP *);
 void field_order_ff(N *);  void field_order_ff(N *);
   
 extern int current_mod;  
 extern GEN_UP2 current_mod_gf2n;  
 extern int lm_lazy;  
   
 int current_ff;  int current_ff;
   
 struct ftab poly_tab[] = {  struct ftab poly_tab[] = {
           {"homogeneous_deg",Phomogeneous_deg,-2},
           {"homogeneous_part",Phomogeneous_part,-3},
           {"reorder",Preorder,3},
           {"uadj_coef",Puadj_coef,3},
         {"ranp",Pranp,2},          {"ranp",Pranp,2},
         {"p_mag",Pp_mag,1},          {"p_mag",Pp_mag,1},
           {"maxblen",Pmaxblen,1},
         {"ord",Pord,-1},          {"ord",Pord,-1},
         {"coef0",Pcoef0,-3},          {"coef0",Pcoef0,-3},
         {"coef",Pcoef,-3},          {"coef",Pcoef,-3},
Line 72  struct ftab poly_tab[] = {
Line 128  struct ftab poly_tab[] = {
   
         {"sparsemod_gf2n",Psparsemod_gf2n,-1},          {"sparsemod_gf2n",Psparsemod_gf2n,-1},
   
         {"setmod_ff",Psetmod_ff,-2},          {"setmod_ff",Psetmod_ff,-3},
         {"simp_ff",Psimp_ff,1},          {"simp_ff",Psimp_ff,1},
         {"extdeg_ff",Pextdeg_ff,0},          {"extdeg_ff",Pextdeg_ff,0},
         {"characteristic_ff",Pcharacteristic_ff,0},          {"characteristic_ff",Pcharacteristic_ff,0},
Line 91  struct ftab poly_tab[] = {
Line 147  struct ftab poly_tab[] = {
         {"ptolmp",Pptolmp,1},          {"ptolmp",Pptolmp,1},
         {"lmptop",Plmptop,1},          {"lmptop",Plmptop,1},
   
           {"sf_galois_action",Psf_galois_action,2},
           {"sf_find_root",Psf_find_root,1},
           {"sf_minipoly",Psf_minipoly,2},
           {"sf_embed",Psf_embed,3},
           {"sf_log",Psf_log,1},
   
           {"ptosfp",Pptosfp,1},
           {"sfptop",Psfptop,1},
         {"ptogf2n",Pptogf2n,1},          {"ptogf2n",Pptogf2n,1},
         {"gf2ntop",Pgf2ntop,-2},          {"gf2ntop",Pgf2ntop,-2},
         {"gf2ntovect",Pgf2ntovect,1},          {"gf2ntovect",Pgf2ntovect,1},
Line 107  struct ftab poly_tab[] = {
Line 171  struct ftab poly_tab[] = {
         {"ureverse_inv_as_power_series",Purevinvmod,2},          {"ureverse_inv_as_power_series",Purevinvmod,2},
         {"uinv_as_power_series",Puinvmod,2},          {"uinv_as_power_series",Puinvmod,2},
   
           {"umul_specialmod",Pumul_specialmod,3},
           {"usquare_specialmod",Pusquare_specialmod,2},
           {"utmul_specialmod",Putmul_specialmod,4},
   
         {"utmul",Putmul,3},          {"utmul",Putmul,3},
         {"umul_ff",Pumul_ff,2},          {"umul_ff",Pumul_ff,2},
         {"usquare_ff",Pusquare_ff,1},          {"usquare_ff",Pusquare_ff,1},
Line 118  struct ftab poly_tab[] = {
Line 186  struct ftab poly_tab[] = {
   
         {"utrunc",Putrunc,2},          {"utrunc",Putrunc,2},
         {"udecomp",Pudecomp,2},          {"udecomp",Pudecomp,2},
         {"ureverse",Pureverse,1},          {"ureverse",Pureverse,-2},
         {"urembymul",Purembymul,2},          {"urembymul",Purembymul,2},
         {"urembymul_precomp",Purembymul_precomp,3},          {"urembymul_precomp",Purembymul_precomp,3},
   
Line 150  struct ftab poly_tab[] = {
Line 218  struct ftab poly_tab[] = {
         {"bininv_gf2n",Pbininv_gf2n,2},          {"bininv_gf2n",Pbininv_gf2n,2},
         {"invtest_gf2n",Pinvtest_gf2n,1},          {"invtest_gf2n",Pinvtest_gf2n,1},
         {"rinvtest_gf2n",Prinvtest_gf2n,0},          {"rinvtest_gf2n",Prinvtest_gf2n,0},
           {"get_next_fft_prime",Pget_next_fft_prime,2},
         {0,0,0},          {0,0,0},
 };  };
   
 extern V up_var;  void Phomogeneous_part(NODE arg,P *rp)
   {
           if ( argc(arg) == 2 )
                   exthp(CO,(P)ARG0(arg),QTOS((Q)ARG1(arg)),rp);
           else
                   exthpc_generic(CO,(P)ARG0(arg),QTOS((Q)ARG2(arg)),
                           VR((P)ARG1(arg)),rp);
   }
   
 void Pranp(arg,rp)  void Phomogeneous_deg(NODE arg,Q *rp)
 NODE arg;  
 P *rp;  
 {  {
           int d;
   
           if ( argc(arg) == 1 )
                   d = homdeg((P)ARG0(arg));
           else
                   d = getchomdeg(VR((P)ARG1(arg)),(P)ARG0(arg));
           STOQ(d,*rp);
   }
   
   /*
           p1 = reorder(p,ovl,nvl) => p1 is 'sorted accoding to nvl.
   */
   
   void Preorder(NODE arg,P *rp)
   {
           VL ovl,nvl,tvl;
           NODE n;
   
           for ( ovl = 0, n = BDY((LIST)ARG1(arg)); n; n = NEXT(n) ) {
                   if ( !ovl ) {
                           NEWVL(ovl); tvl = ovl;
                   } else {
                           NEWVL(NEXT(tvl)); tvl = NEXT(tvl);
                   }
                           VR(tvl) = VR((P)BDY(n));
           }
           for ( nvl = 0, n = BDY((LIST)ARG2(arg)); n; n = NEXT(n) ) {
                   if ( !nvl ) {
                           NEWVL(nvl); tvl = nvl;
                   } else {
                           NEWVL(NEXT(tvl)); tvl = NEXT(tvl);
                   }
                           VR(tvl) = VR((P)BDY(n));
           }
           reorderp(nvl,ovl,(P)ARG0(arg),rp);
   }
   
   /*
           uadj_coef(F,M,M2)
           if ( F is a non-negative integer )
                   return F > M2 ? F-M : M2;
           else
                   F = CN*V^N+...+C0
                   return uadj_coef(CN,M,M2)*V^N+...+uadj_coef(C0,M,M2);
   */
   
   void Puadj_coef(NODE arg,P *rp)
   {
           UP f,r;
           N m,m2;
   
           ptoup((P)ARG0(arg),&f);
           m = NM((Q)ARG1(arg));
           m2 = NM((Q)ARG2(arg));
           adj_coefup(f,m,m2,&r);
           uptop(r,rp);
   }
   
   /*
           get_next_fft_prime(StartIndex,Bits)
           tries to find smallest Index >= StartIndex s.t.
                   2^(Bits-1)|FFTprime[Index]-1
           return [Index,Mod] or 0 (not exist)
   */
   
   void Pget_next_fft_prime(NODE arg,LIST *rp)
   {
           unsigned int mod,d;
           int start,bits,i;
           NODE n;
           Q q,ind;
   
           start = QTOS((Q)ARG0(arg));
           bits = QTOS((Q)ARG1(arg));
           for ( i = start; ; i++ ) {
                   get_fft_prime(i,&mod,&d);
                   if ( !mod ) {
                           *rp = 0; return;
                   }
                   if ( bits <= (int)d ) {
                           UTOQ(mod,q);
                           UTOQ(i,ind);
                           n = mknode(2,ind,q);
                           MKLIST(*rp,n);
                           return;
                   }
           }
   }
   
   void Pranp(NODE arg,P *rp)
   {
         int n;          int n;
         UP c;          UP c;
   
Line 171  P *rp;
Line 336  P *rp;
                 *rp = 0;                  *rp = 0;
 }  }
   
 void ranp(n,nr)  void ranp(int n,UP *nr)
 int n;  
 UP *nr;  
 {  {
         int i;          int i;
         unsigned int r;          unsigned int r;
Line 193  UP *nr;
Line 356  UP *nr;
                 *nr = 0;                  *nr = 0;
 }  }
   
 void Pp_mag(arg,rp)  void Pmaxblen(NODE arg,Q *rp)
 NODE arg;  
 Q *rp;  
 {  {
         int l;          int l;
           l = maxblenp(ARG0(arg));
           STOQ(l,*rp);
   }
   
   void Pp_mag(NODE arg,Q *rp)
   {
           int l;
         l = p_mag(ARG0(arg));          l = p_mag(ARG0(arg));
         STOQ(l,*rp);          STOQ(l,*rp);
 }  }
   
 void Pord(arg,listp)  void Pord(NODE arg,LIST *listp)
 NODE arg;  
 LIST *listp;  
 {  {
         NODE n,tn;          NODE n,tn;
         LIST l;          LIST l;
Line 254  LIST *listp;
Line 420  LIST *listp;
         NEXT(tn) = 0; MKLIST(l,n); *listp = l;          NEXT(tn) = 0; MKLIST(l,n); *listp = l;
 }  }
   
 void Pcoef0(arg,rp)  void Pcoef0(NODE arg,Obj *rp)
 NODE arg;  
 Obj *rp;  
 {  {
         Obj t,n;          Obj t,n;
         P s;          P s;
Line 297  Obj *rp;
Line 461  Obj *rp;
         }          }
 }  }
   
 void Pcoef(arg,rp)  void Pcoef(NODE arg,Obj *rp)
 NODE arg;  
 Obj *rp;  
 {  {
         Obj t,n;          Obj t,n;
         P s;          P s;
Line 339  Obj *rp;
Line 501  Obj *rp;
         }          }
 }  }
   
 void Pcoef_gf2n(arg,rp)  void Pcoef_gf2n(NODE arg,Obj *rp)
 NODE arg;  
 Obj *rp;  
 {  {
         Obj t,n;          Obj t,n;
         int id,d;          int id,d;
Line 362  Obj *rp;
Line 522  Obj *rp;
                 *rp = 0;                  *rp = 0;
 }  }
   
 void Pdeg(arg,rp)  void Pdeg(NODE arg,Q *rp)
 NODE arg;  
 Q *rp;  
 {  {
         Obj t,v;          Obj t,v;
         int d;          int d;
Line 390  Q *rp;
Line 548  Q *rp;
                 degp(VR((P)ARG1(arg)),(P)ARG0(arg),rp);                  degp(VR((P)ARG1(arg)),(P)ARG0(arg),rp);
 }  }
   
 void Pmindeg(arg,rp)  void Pmindeg(NODE arg,Q *rp)
 NODE arg;  
 Q *rp;  
 {  {
         Obj t;          Obj t;
   
Line 402  Q *rp;
Line 558  Q *rp;
                 getmindeg(VR((P)ARG1(arg)),(P)ARG0(arg),rp);                  getmindeg(VR((P)ARG1(arg)),(P)ARG0(arg),rp);
 }  }
   
 void Psetmod(arg,rp)  void Psetmod(NODE arg,Q *rp)
 NODE arg;  
 Q *rp;  
 {  {
         if ( arg ) {          if ( arg ) {
                 asir_assert(ARG0(arg),O_N,"setmod");                  asir_assert(ARG0(arg),O_N,"setmod");
Line 413  Q *rp;
Line 567  Q *rp;
         STOQ(current_mod,*rp);          STOQ(current_mod,*rp);
 }  }
   
 void Psparsemod_gf2n(arg,rp)  void Psparsemod_gf2n(NODE arg,Q *rp)
 NODE arg;  
 Q *rp;  
 {  {
         int id;          int id;
   
Line 428  Q *rp;
Line 580  Q *rp;
         STOQ(id,*rp);          STOQ(id,*rp);
 }  }
   
 void Pmultest_gf2n(arg,rp)  void Pmultest_gf2n(NODE arg,GF2N *rp)
 NODE arg;  
 GF2N *rp;  
 {  {
         GF2N a,b,c;          GF2N a,b,c;
         int i;          int i;
Line 441  GF2N *rp;
Line 591  GF2N *rp;
         *rp = c;          *rp = c;
 }  }
   
 void Psquaretest_gf2n(arg,rp)  void Psquaretest_gf2n(NODE arg,GF2N *rp)
 NODE arg;  
 GF2N *rp;  
 {  {
         GF2N a,c;          GF2N a,c;
         int i;          int i;
Line 454  GF2N *rp;
Line 602  GF2N *rp;
         *rp = c;          *rp = c;
 }  }
   
 void Pinvtest_gf2n(arg,rp)  void Pinvtest_gf2n(NODE arg,GF2N *rp)
 NODE arg;  
 GF2N *rp;  
 {  {
         GF2N a,c;          GF2N a,c;
         int i;          int i;
Line 467  GF2N *rp;
Line 613  GF2N *rp;
         *rp = c;          *rp = c;
 }  }
   
 void Pbininv_gf2n(arg,rp)  void Pbininv_gf2n(NODE arg,GF2N *rp)
 NODE arg;  
 GF2N *rp;  
 {  {
         UP2 a,inv;          UP2 a,inv;
         int n;          int n;
Line 480  GF2N *rp;
Line 624  GF2N *rp;
         MKGF2N(inv,*rp);          MKGF2N(inv,*rp);
 }  }
   
 void Prinvtest_gf2n(rp)  void Prinvtest_gf2n(Real *rp)
 Real *rp;  
 {  {
         GF2N *a;          GF2N *a;
         GF2N c;          GF2N c;
Line 501  Real *rp;
Line 644  Real *rp;
         MKReal(r,*rp);          MKReal(r,*rp);
 }  }
   
 void Pfind_root_gf2n(arg,rp)  void Pfind_root_gf2n(NODE arg,GF2N *rp)
 NODE arg;  
 GF2N *rp;  
 {  {
   
 #if 0  #if 0
Line 519  GF2N *rp;
Line 660  GF2N *rp;
 #endif  #endif
 }  }
   
 void Pis_irred_gf2(arg,rp)  void Pis_irred_gf2(NODE arg,Q *rp)
 NODE arg;  
 Q *rp;  
 {  {
         UP2 t;          UP2 t;
   
Line 529  Q *rp;
Line 668  Q *rp;
         *rp = irredcheckup2(t) ? ONE : 0;          *rp = irredcheckup2(t) ? ONE : 0;
 }  }
   
 void Pis_irred_ddd_gf2(arg,rp)  void Pis_irred_ddd_gf2(NODE arg,Q *rp)
 NODE arg;  
 Q *rp;  
 {  {
         UP2 t;          UP2 t;
         int ret;          int ret;
Line 541  Q *rp;
Line 678  Q *rp;
         STOQ(ret,*rp);          STOQ(ret,*rp);
 }  }
   
 void Psetmod_ff(arg,rp)  void Psetmod_ff(NODE arg,Obj *rp)
 NODE arg;  
 Obj *rp;  
 {  {
         int ac;          int ac;
           int d;
         Obj mod,defpoly;          Obj mod,defpoly;
         N n;          N n;
         UP up;          UP up;
         UP2 up2;          UP2 up2;
         Q q;          UM dp;
         P p;          Q q,r;
           P p,p1,y;
         NODE n0,n1;          NODE n0,n1;
         LIST list;          LIST list;
   
Line 563  Obj *rp;
Line 700  Obj *rp;
                 switch ( OID(mod) ) {                  switch ( OID(mod) ) {
                         case O_N:                          case O_N:
                                 current_ff = FF_GFP;                                  current_ff = FF_GFP;
                                 setmod_lm(NM((Q)mod)); break;                                  setmod_lm(NM((Q)mod));
                                   break;
                         case O_P:                          case O_P:
                                 current_ff = FF_GF2N;                                  current_ff = FF_GF2N;
                                 setmod_gf2n((P)mod); break;                                  setmod_gf2n((P)mod); break;
Line 571  Obj *rp;
Line 709  Obj *rp;
                                 error("setmod_ff : invalid argument");                                  error("setmod_ff : invalid argument");
                 }                  }
         } else if ( ac == 2 ) {          } else if ( ac == 2 ) {
                 current_ff = FF_GFPN;                  if ( OID(ARG0(arg)) == O_N ) {
                 defpoly = (Obj)ARG0(arg);                          /* small finite field; primitive root representation */
                 mod = (Obj)ARG1(arg);                          current_ff = FF_GFS;
                 if ( !mod || !defpoly )                          setmod_sf(QTOS((Q)ARG0(arg)),QTOS((Q)ARG1(arg)));
                         error("setmod_ff : invalid argument");                  } else {
                 setmod_lm(NM((Q)mod));                          mod = (Obj)ARG1(arg);
                 setmod_gfpn((P)defpoly);                          current_ff = FF_GFPN;
                           defpoly = (Obj)ARG0(arg);
                           if ( !mod || !defpoly )
                                   error("setmod_ff : invalid argument");
                           setmod_lm(NM((Q)mod));
                           setmod_gfpn((P)defpoly);
                   }
           } else if ( ac == 3 ) {
                   /* finite extension of a small finite field */
                   current_ff = FF_GFS;
                   setmod_sf(QTOS((Q)ARG0(arg)),QTOS((Q)ARG1(arg)));
                   d = QTOS((Q)ARG2(arg));
                   generate_defpoly_sfum(d,&dp);
                   setmod_gfsn(dp);
                   current_ff = FF_GFSN;
         }          }
         switch ( current_ff ) {          switch ( current_ff ) {
                 case FF_GFP:                  case FF_GFP:
Line 590  Obj *rp;
Line 742  Obj *rp;
                         MKNODE(n1,q,0); MKNODE(n0,p,n1);                          MKNODE(n1,q,0); MKNODE(n0,p,n1);
                         MKLIST(list,n0);                          MKLIST(list,n0);
                         *rp = (Obj)list; break;                          *rp = (Obj)list; break;
                   case FF_GFS:
                   case FF_GFSN:
                           STOQ(current_gfs_p,q);
                           if ( current_gfs_ext )
                                   enc_to_p(current_gfs_p,current_gfs_iton[1],
                                           VR(current_gfs_ext),&p);
                           else {
                                   if ( current_gfs_p == 2 )
                                           r = ONE;
                                   else if ( !current_gfs_ntoi )
                                           r = 0;
                                   else
                                           STOQ(current_gfs_iton[1],r);
                                   p = (P)r;
                           }
                           switch ( current_ff ) {
                                   case FF_GFS:
                                           n0 = mknode(3,q,current_gfs_ext,p);
                                           break;
                                   case FF_GFSN:
                                           getmod_gfsn(&dp);
                                           makevar("y",&y);
                                           sfumtop(VR(y),dp,&p1);
                                           n0 = mknode(4,q,current_gfs_ext,p,p1);
                                           break;
                           }
                           MKLIST(list,n0);
                           *rp = (Obj)list; break;
                 default:                  default:
                         *rp = 0; break;                          *rp = 0; break;
         }          }
 }  }
   
 void Pextdeg_ff(rp)  void Pextdeg_ff(Q *rp)
 Q *rp;  
 {  {
         int d;          int d;
         UP2 up2;          UP2 up2;
         UP up;          UP up;
           UM dp;
   
         switch ( current_ff ) {          switch ( current_ff ) {
                 case FF_GFP:                  case FF_GFP:
Line 609  Q *rp;
Line 789  Q *rp;
                         getmod_gf2n(&up2); d = degup2(up2); STOQ(d,*rp); break;                          getmod_gf2n(&up2); d = degup2(up2); STOQ(d,*rp); break;
                 case FF_GFPN:                  case FF_GFPN:
                         getmod_gfpn(&up); STOQ(up->d,*rp); break;                          getmod_gfpn(&up); STOQ(up->d,*rp); break;
                   case FF_GFS:
                           if ( !current_gfs_ext )
                                   *rp = ONE;
                           else
                                   *rp = DEG(DC(current_gfs_ext));
                           break;
                   case FF_GFSN:
                           getmod_gfsn(&dp);
                           STOQ(DEG(dp),*rp);
                           break;
                 default:                  default:
                         error("extdeg_ff : current_ff is not set");                          error("extdeg_ff : current_ff is not set");
         }          }
 }  }
   
 void Pcharacteristic_ff(rp)  void Pcharacteristic_ff(Q *rp)
 Q *rp;  
 {  {
         N lm;          N lm;
   
Line 625  Q *rp;
Line 814  Q *rp;
                         getmod_lm(&lm); NTOQ(lm,1,*rp); break;                          getmod_lm(&lm); NTOQ(lm,1,*rp); break;
                 case FF_GF2N:                  case FF_GF2N:
                         STOQ(2,*rp); break;                          STOQ(2,*rp); break;
                   case FF_GFS:
                   case FF_GFSN:
                           STOQ(current_gfs_p,*rp); break;
                 default:                  default:
                         error("characteristic_ff : current_ff is not set");                          error("characteristic_ff : current_ff is not set");
         }          }
 }  }
   
 void Pfield_type_ff(rp)  void Pfield_type_ff(Q *rp)
 Q *rp;  
 {  {
         STOQ(current_ff,*rp);          STOQ(current_ff,*rp);
 }  }
   
 void Pfield_order_ff(rp)  void Pfield_order_ff(Q *rp)
 Q *rp;  
 {  {
         N n;          N n;
   
Line 645  Q *rp;
Line 835  Q *rp;
         NTOQ(n,1,*rp);          NTOQ(n,1,*rp);
 }  }
   
 void field_order_ff(order)  void Prandom_ff(Obj *rp)
 N *order;  
 {  {
         UP2 up2;  
         UP up;  
         N m;  
         int d,w;  
   
         switch ( current_ff ) {  
                 case FF_GFP:  
                         getmod_lm(order); break;  
                 case FF_GF2N:  
                         getmod_gf2n(&up2); d = degup2(up2);  
                         w = (d>>5)+1;  
                         *order = m = NALLOC(w);  
                         PL(m)=w;  
                         bzero((char *)BD(m),w*sizeof(unsigned int));  
                         BD(m)[d>>5] |= 1<<(d&31);  
                         break;  
                 case FF_GFPN:  
                         getmod_lm(&m);  
                         getmod_gfpn(&up); pwrn(m,up->d,order); break;  
                 default:  
                         error("field_order_ff : current_ff is not set");  
         }  
 }  
   
 void Prandom_ff(rp)  
 Obj *rp;  
 {  
         LM l;          LM l;
         GF2N g;          GF2N g;
         GFPN p;          GFPN p;
           GFS s;
           GFSN spn;
   
         switch ( current_ff ) {          switch ( current_ff ) {
                 case FF_GFP:                  case FF_GFP:
Line 686  Obj *rp;
Line 850  Obj *rp;
                         randomgf2n(&g); *rp = (Obj)g; break;                          randomgf2n(&g); *rp = (Obj)g; break;
                 case FF_GFPN:                  case FF_GFPN:
                         randomgfpn(&p); *rp = (Obj)p; break;                          randomgfpn(&p); *rp = (Obj)p; break;
                   case FF_GFS:
                           randomgfs(&s); *rp = (Obj)s; break;
                   case FF_GFSN:
                           randomgfsn(&spn); *rp = (Obj)spn; break;
                 default:                  default:
                         error("random_ff : current_ff is not set");                          error("random_ff : current_ff is not set");
         }          }
 }  }
   
 void Psimp_ff(arg,rp)  void Psimp_ff(NODE arg,Obj *rp)
 NODE arg;  
 Obj *rp;  
 {  {
         LM r;  
         GF2N rg;  
         extern lm_lazy;  
   
         simp_ff((Obj)ARG0(arg),rp);          simp_ff((Obj)ARG0(arg),rp);
 }  }
   
 void simp_ff(p,rp)  void getcoef(VL vl,P p,V v,Q d,P *r)
 Obj p;  
 Obj *rp;  
 {  {
         Num n;  
         LM r,s;  
         DCP dc,dcr0,dcr;  
         GF2N rg,sg;  
         GFPN rpn,spn;  
         P t;  
         Obj obj;  
   
         lm_lazy = 0;  
         if ( !p )  
                 *rp = 0;  
         else if ( OID(p) == O_N ) {  
                 switch ( current_ff ) {  
                         case FF_GFP:  
                                 ptolmp((P)p,&t); simplm((LM)t,&s); *rp = (Obj)s;  
                                 break;  
                         case FF_GF2N:  
                                 ptogf2n((Obj)p,&rg); simpgf2n((GF2N)rg,&sg); *rp = (Obj)sg;  
                                 break;  
                         case FF_GFPN:  
                                 ntogfpn((Obj)p,&rpn); simpgfpn((GFPN)rpn,&spn); *rp = (Obj)spn;  
                                 break;  
                         default:  
                                 *rp = (Obj)p;  
                                 break;  
                 }  
         } else if ( OID(p) == O_P ) {  
                 for ( dc = DC((P)p), dcr0 = 0; dc; dc = NEXT(dc) ) {  
                         simp_ff((Obj)COEF(dc),&obj);  
                         if ( obj ) {  
                                 NEXTDC(dcr0,dcr); DEG(dcr) = DEG(dc); COEF(dcr) = (P)obj;  
                         }  
                 }  
                 if ( !dcr0 )  
                         *rp = 0;  
                 else {  
                         NEXT(dcr) = 0; MKP(VR((P)p),dcr0,t); *rp = (Obj)t;  
                 }  
         } else  
                 error("simp_ff : not implemented yet");  
 }  
   
 void getcoef(vl,p,v,d,r)  
 VL vl;  
 P p;  
 V v;  
 Q d;  
 P *r;  
 {  
         P s,t,u,a,b,x;          P s,t,u,a,b,x;
         DCP dc;          DCP dc;
         V w;          V w;
Line 779  P *r;
Line 890  P *r;
         }          }
 }  }
   
 void Pdeglist(arg,rp)  void Pdeglist(NODE arg,LIST *rp)
 NODE arg;  
 LIST *rp;  
 {  {
         NODE d;          NODE d;
   
Line 789  LIST *rp;
Line 898  LIST *rp;
         MKLIST(*rp,d);          MKLIST(*rp,d);
 }  }
   
 void Pch_mv(arg,rp)  void Pch_mv(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         change_mvar(CO,(P)ARG0(arg),VR((P)ARG1(arg)),rp);          change_mvar(CO,(P)ARG0(arg),VR((P)ARG1(arg)),rp);
 }  }
   
 void Pre_mv(arg,rp)  void Pre_mv(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         restore_mvar(CO,(P)ARG0(arg),VR((P)ARG1(arg)),rp);          restore_mvar(CO,(P)ARG0(arg),VR((P)ARG1(arg)),rp);
 }  }
   
 void change_mvar(vl,p,v,r)  void change_mvar(VL vl,P p,V v,P *r)
 VL vl;  
 P p;  
 V v;  
 P *r;  
 {  {
         Q d;          Q d;
         DCP dc,dc0;          DCP dc,dc0;
Line 825  P *r;
Line 926  P *r;
         }          }
 }  }
   
 void restore_mvar(vl,p,v,r)  void restore_mvar(VL vl,P p,V v,P *r)
 VL vl;  
 P p;  
 V v;  
 P *r;  
 {  {
         P s,u,a,b,x;          P s,u,a,b,x;
         DCP dc;          DCP dc;
Line 846  P *r;
Line 943  P *r;
         }          }
 }  }
   
 void getdeglist(p,v,d)  void getdeglist(P p,V v,NODE *d)
 P p;  
 V v;  
 NODE *d;  
 {  {
         NODE n,n0,d0,d1,d2;          NODE n,n0,d0,d1,d2;
         DCP dc;          DCP dc;
Line 868  NODE *d;
Line 962  NODE *d;
                 *d = d0;                  *d = d0;
         }          }
 }  }
 void Pmergelist(arg,rp)  
 NODE arg;  void Pmergelist(NODE arg,LIST *rp)
 LIST *rp;  
 {  {
     NODE n;      NODE n;
   
Line 880  LIST *rp;
Line 973  LIST *rp;
         MKLIST(*rp,n);          MKLIST(*rp,n);
 }  }
   
 void mergedeglist(d0,d1,dr)  void mergedeglist(NODE d0,NODE d1,NODE *dr)
 NODE d0,d1,*dr;  
 {  {
         NODE t0,t,dt;          NODE t0,t,dt;
         Q d;          Q d;
Line 913  NODE d0,d1,*dr;
Line 1005  NODE d0,d1,*dr;
         }          }
 }  }
   
 void Pptomp(arg,rp)  void Pptomp(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         ptomp(QTOS((Q)ARG1(arg)),(P)ARG0(arg),rp);          ptomp(QTOS((Q)ARG1(arg)),(P)ARG0(arg),rp);
 }  }
   
 void Pmptop(arg,rp)  void Pmptop(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         mptop((P)ARG0(arg),rp);          mptop((P)ARG0(arg),rp);
 }  }
   
 void Pptolmp(arg,rp)  void Pptolmp(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         ptolmp((P)ARG0(arg),rp);          ptolmp((P)ARG0(arg),rp);
 }  }
   
 void Plmptop(arg,rp)  void Plmptop(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         lmptop((P)ARG0(arg),rp);          lmptop((P)ARG0(arg),rp);
 }  }
   
 void Pptogf2n(arg,rp)  void Psf_galois_action(NODE arg,P *rp)
 NODE arg;  
 GF2N *rp;  
 {  {
         ptogf2n((Obj)ARG0(arg),rp);          sf_galois_action(ARG0(arg),ARG1(arg),rp);
 }  }
   
 void Pgf2ntop(arg,rp)  /*
 NODE arg;    sf_embed(F,B,PM)
 P *rp;    F : an element of GF(pn)
     B : the image of the primitive root of GF(pn)
     PM : order of GF(pm)
   */
   
   void Psf_embed(NODE arg,P *rp)
 {  {
         extern V up2_var;          int k,pm;
   
           /* GF(pn)={0,1,a,a^2,...}->GF(pm)={0,1,b,b^2,...}; a->b^k */
           k = CONT((GFS)ARG1(arg));
           pm = QTOS((Q)ARG2(arg));
           sf_embed((P)ARG0(arg),k,pm,rp);
   }
   
   void Psf_log(NODE arg,Q *rp)
   {
           int k;
   
           if ( !ARG0(arg) )
                   error("sf_log : invalid armument");
           k = CONT((GFS)ARG0(arg));
           STOQ(k,*rp);
   }
   
   void Psf_find_root(NODE arg,GFS *rp)
   {
           P p;
           Obj t;
           int d;
           UM u;
           int *root;
   
           p = (P)ARG0(arg);
           simp_ff((Obj)p,&t); p = (P)t;
           d = getdeg(VR(p),p);
           u = W_UMALLOC(d);
           ptosfum(p,u);
           root = (int *)ALLOCA(d*sizeof(int));
           find_rootsf(u,root);
           MKGFS(IFTOF(root[0]),*rp);
   }
   
   void Psf_minipoly(NODE arg,P *rp)
   {
           Obj t;
           P p1,p2;
           int d1,d2;
           UM up1,up2,m;
   
           p1 = (P)ARG0(arg); simp_ff((Obj)p1,&t); p1 = (P)t;
           p2 = (P)ARG1(arg); simp_ff((Obj)p2,&t); p2 = (P)t;
           d1 = getdeg(VR(p1),p1); up1 = W_UMALLOC(d1); ptosfum(p1,up1);
           d2 = getdeg(VR(p2),p2); up2 = W_UMALLOC(d2); ptosfum(p2,up2);
           m = W_UMALLOC(d2);
           minipolysf(up1,up2,m);
           sfumtop(VR(p2),m,&p1);
           sfptop(p1,rp);
   }
   
   void Pptosfp(NODE arg,P *rp)
   {
           ptosfp(ARG0(arg),rp);
   }
   
   void Psfptop(NODE arg,P *rp)
   {
           sfptop((P)ARG0(arg),rp);
   }
   
   void Pptogf2n(NODE arg,GF2N *rp)
   {
           ptogf2n((Obj)ARG0(arg),rp);
   }
   
   void Pgf2ntop(NODE arg,P *rp)
   {
         if ( argc(arg) == 2 )          if ( argc(arg) == 2 )
                 up2_var = VR((P)ARG1(arg));                  up2_var = VR((P)ARG1(arg));
         gf2ntop((GF2N)ARG0(arg),rp);          gf2ntop((GF2N)ARG0(arg),rp);
 }  }
   
 void Pgf2ntovect(arg,rp)  void Pgf2ntovect(NODE arg,VECT *rp)
 NODE arg;  
 VECT *rp;  
 {  {
         gf2ntovect((GF2N)ARG0(arg),rp);          gf2ntovect((GF2N)ARG0(arg),rp);
 }  }
   
 void Pptogfpn(arg,rp)  void Pptogfpn(NODE arg,GFPN *rp)
 NODE arg;  
 GF2N *rp;  
 {  {
         ptogfpn((Obj)ARG0(arg),rp);          ptogfpn((Obj)ARG0(arg),rp);
 }  }
   
 void Pgfpntop(arg,rp)  void Pgfpntop(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         extern V up_var;  
   
         if ( argc(arg) == 2 )          if ( argc(arg) == 2 )
                 up_var = VR((P)ARG1(arg));                  up_var = VR((P)ARG1(arg));
         gfpntop((GFPN)ARG0(arg),rp);          gfpntop((GFPN)ARG0(arg),rp);
 }  }
   
 void Pureverse(arg,rp)  void Pureverse(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p,r;          UP p,r;
   
         ptoup((P)ARG0(arg),&p);          ptoup((P)ARG0(arg),&p);
         reverseup(p,p->d,&r);          if ( argc(arg) == 1 )
                   reverseup(p,p->d,&r);
           else
                   reverseup(p,QTOS((Q)ARG1(arg)),&r);
         uptop(r,rp);          uptop(r,rp);
 }  }
   
 void Putrunc(arg,rp)  void Putrunc(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p,r;          UP p,r;
   
Line 1006  P *rp;
Line 1152  P *rp;
         uptop(r,rp);          uptop(r,rp);
 }  }
   
 void Pudecomp(arg,rp)  void Pudecomp(NODE arg,LIST *rp)
 NODE arg;  
 LIST *rp;  
 {  {
         P u,l;          P u,l;
         UP p,up,low;          UP p,up,low;
Line 1022  LIST *rp;
Line 1166  LIST *rp;
         MKLIST(*rp,n0);          MKLIST(*rp,n0);
 }  }
   
 void Purembymul(arg,rp)  void Purembymul(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p1,p2,r;          UP p1,p2,r;
   
Line 1044  P *rp;
Line 1186  P *rp;
  * p2*inv = 1 mod x^d2   * p2*inv = 1 mod x^d2
  */   */
   
 void Purembymul_precomp(arg,rp)  void Purembymul_precomp(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p1,p2,inv,r;          UP p1,p2,inv,r;
   
Line 1066  P *rp;
Line 1206  P *rp;
         }          }
 }  }
   
 void Puinvmod(arg,rp)  void Puinvmod(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p,r;          UP p,r;
   
Line 1077  P *rp;
Line 1215  P *rp;
         uptop(r,rp);          uptop(r,rp);
 }  }
   
 void Purevinvmod(arg,rp)  void Purevinvmod(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p,pr,r;          UP p,pr,r;
   
Line 1089  P *rp;
Line 1225  P *rp;
         uptop(r,rp);          uptop(r,rp);
 }  }
   
 void Ppwrmod_ff(arg,rp)  void Ppwrmod_ff(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p1,p2;          UP p1,p2;
   
Line 1102  P *rp;
Line 1236  P *rp;
                 case FF_GF2N:                  case FF_GF2N:
                         powermodup_gf2n(p1,&p2); break;                          powermodup_gf2n(p1,&p2); break;
                 case FF_GFPN:                  case FF_GFPN:
                   case FF_GFS:
                   case FF_GFSN:
                         powermodup(p1,&p2); break;                          powermodup(p1,&p2); break;
                 default:                  default:
                         error("pwrmod_ff : current_ff is not set");                          error("pwrmod_ff : current_ff is not set");
Line 1109  P *rp;
Line 1245  P *rp;
         uptop(p2,rp);          uptop(p2,rp);
 }  }
   
 void Pgeneric_pwrmod_ff(arg,rp)  void Pgeneric_pwrmod_ff(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP g,f,r;          UP g,f,r;
   
Line 1123  P *rp;
Line 1257  P *rp;
                 case FF_GF2N:                  case FF_GF2N:
                         generic_powermodup_gf2n(g,f,(Q)ARG2(arg),&r); break;                          generic_powermodup_gf2n(g,f,(Q)ARG2(arg),&r); break;
                 case FF_GFPN:                  case FF_GFPN:
                   case FF_GFS:
                   case FF_GFSN:
                         generic_powermodup(g,f,(Q)ARG2(arg),&r); break;                          generic_powermodup(g,f,(Q)ARG2(arg),&r); break;
                 default:                  default:
                         error("generic_pwrmod_ff : current_ff is not set");                          error("generic_pwrmod_ff : current_ff is not set");
Line 1130  P *rp;
Line 1266  P *rp;
         uptop(r,rp);          uptop(r,rp);
 }  }
   
 void Ppwrtab_ff(arg,rp)  void Ppwrtab_ff(NODE arg,VECT *rp)
 NODE arg;  
 VECT *rp;  
 {  {
         UP f,xp;          UP f,xp;
         UP *tab;          UP *tab;
Line 1150  VECT *rp;
Line 1284  VECT *rp;
                 case FF_GF2N:                  case FF_GF2N:
                         powertabup_gf2n(f,xp,tab); break;                          powertabup_gf2n(f,xp,tab); break;
                 case FF_GFPN:                  case FF_GFPN:
                   case FF_GFS:
                   case FF_GFSN:
                         powertabup(f,xp,tab); break;                          powertabup(f,xp,tab); break;
                 default:                  default:
                         error("pwrtab_ff : current_ff is not set");                          error("pwrtab_ff : current_ff is not set");
Line 1159  VECT *rp;
Line 1295  VECT *rp;
                 uptop(tab[i],(P *)&BDY(r)[i]);                  uptop(tab[i],(P *)&BDY(r)[i]);
 }  }
   
 void Pkpwrmod_lm(arg,rp)  void Pkpwrmod_lm(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p1,p2;          UP p1,p2;
   
Line 1170  P *rp;
Line 1304  P *rp;
         uptop(p2,rp);          uptop(p2,rp);
 }  }
   
 void Pkgeneric_pwrmod_lm(arg,rp)  void Pkgeneric_pwrmod_lm(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP g,f,r;          UP g,f,r;
   
Line 1182  P *rp;
Line 1314  P *rp;
         uptop(r,rp);          uptop(r,rp);
 }  }
   
 void Pkpwrtab_lm(arg,rp)  void Pkpwrtab_lm(NODE arg,VECT *rp)
 NODE arg;  
 VECT *rp;  
 {  {
         UP f,xp;          UP f,xp;
         UP *tab;          UP *tab;
Line 1202  VECT *rp;
Line 1332  VECT *rp;
                 uptop(tab[i],(P *)&BDY(r)[i]);                  uptop(tab[i],(P *)&BDY(r)[i]);
 }  }
   
 void Plazy_lm(arg,rp)  void Plazy_lm(NODE arg,Q *rp)
 NODE arg;  
 Q *rp;  
 {  {
         lm_lazy = QTOS((Q)ARG0(arg));          lm_lazy = QTOS((Q)ARG0(arg));
         *rp = 0;          *rp = 0;
 }  }
   
 void Pkmul(arg,rp)  void Pkmul(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         P n1,n2;          P n1,n2;
   
Line 1222  P *rp;
Line 1348  P *rp;
         kmulp(CO,n1,n2,rp);          kmulp(CO,n1,n2,rp);
 }  }
   
 void Pksquare(arg,rp)  void Pksquare(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         P n1;          P n1;
   
Line 1233  P *rp;
Line 1357  P *rp;
         ksquarep(CO,n1,rp);          ksquarep(CO,n1,rp);
 }  }
   
 void Pktmul(arg,rp)  void Pktmul(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p1,p2,r;          UP p1,p2,r;
   
Line 1245  P *rp;
Line 1367  P *rp;
         uptop(r,rp);          uptop(r,rp);
 }  }
   
 void Pumul(arg,rp)  void Pumul(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         P a1,a2;          P a1,a2;
         UP p1,p2,r;          UP p1,p2,r;
Line 1256  P *rp;
Line 1376  P *rp;
         if ( !a1 || !a2 || NUM(a1) || NUM(a2) )          if ( !a1 || !a2 || NUM(a1) || NUM(a2) )
                 mulp(CO,a1,a2,rp);                  mulp(CO,a1,a2,rp);
         else {          else {
                 if ( !uzpcheck(a1) || !uzpcheck(a2) || VR(a1) != VR(a2) )                  if ( !uzpcheck((Obj)a1) || !uzpcheck((Obj)a2) || VR(a1) != VR(a2) )
                         error("umul : invalid argument");                          error("umul : invalid argument");
                 ptoup(a1,&p1);                  ptoup(a1,&p1);
                 ptoup(a2,&p2);                  ptoup(a2,&p2);
Line 1265  P *rp;
Line 1385  P *rp;
         }          }
 }  }
   
 void Pusquare(arg,rp)  void Pusquare(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p1,p2,r;          UP p1,r;
   
         ptoup((P)ARG0(arg),&p1);          ptoup((P)ARG0(arg),&p1);
         hybrid_squareup(0,p1,&r);          hybrid_squareup(0,p1,&r);
         uptop(r,rp);          uptop(r,rp);
 }  }
   
 void Putmul(arg,rp)  void Putmul(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p1,p2,r;          UP p1,p2,r;
   
Line 1288  P *rp;
Line 1404  P *rp;
         uptop(r,rp);          uptop(r,rp);
 }  }
   
 void Pumul_ff(arg,rp)  void Pumul_ff(NODE arg,Obj *rp)
 NODE arg;  
 Obj *rp;  
 {  {
         P a1,a2;          P a1,a2;
         UP p1,p2,r;          UP p1,p2,r;
Line 1304  Obj *rp;
Line 1418  Obj *rp;
         simp_ff((Obj)p,rp);          simp_ff((Obj)p,rp);
 }  }
   
 void Pusquare_ff(arg,rp)  void Pusquare_ff(NODE arg,Obj *rp)
 NODE arg;  
 Obj *rp;  
 {  {
         UP p1,p2,r;          UP p1,r;
         P p;          P p;
   
         ptoup((P)ARG0(arg),&p1);          ptoup((P)ARG0(arg),&p1);
Line 1317  Obj *rp;
Line 1429  Obj *rp;
         simp_ff((Obj)p,rp);          simp_ff((Obj)p,rp);
 }  }
   
 void Putmul_ff(arg,rp)  void Putmul_ff(NODE arg,Obj *rp)
 NODE arg;  
 Obj *rp;  
 {  {
         UP p1,p2,r;          UP p1,p2,r;
         P p;          P p;
Line 1331  Obj *rp;
Line 1441  Obj *rp;
         simp_ff((Obj)p,rp);          simp_ff((Obj)p,rp);
 }  }
   
 void Phfmul_lm(arg,rp)  void Phfmul_lm(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP p1,p2,r;          UP p1,p2,r;
         UP hi,lo,mid,t,s,p10,p11,p20,p21;          UP hi,lo,mid,t,s,p10,p11,p20,p21;
Line 1374  P *rp;
Line 1482  P *rp;
         uptop(r,rp);          uptop(r,rp);
 }  }
   
 void Pfmultest(arg,rp)  void Pfmultest(NODE arg,LIST *rp)
 NODE arg;  
 LIST *rp;  
 {  {
         P p1,p2,r;          P p1,p2,r;
         int d1,d2;          int d1,d2;
Line 1421  LIST *rp;
Line 1527  LIST *rp;
         }          }
 }  }
   
 void Pkmulum(arg,rp)  void Pkmulum(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         P p1,p2;          P p1,p2;
         int d1,d2,mod;          int d1,d2,mod;
Line 1438  P *rp;
Line 1542  P *rp;
         umtop(VR(p1),wr,rp);          umtop(VR(p1),wr,rp);
 }  }
   
 void Pksquareum(arg,rp)  void Pksquareum(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         P p1;          P p1;
         int d1,mod;          int d1,mod;
Line 1455  P *rp;
Line 1557  P *rp;
         umtop(VR(p1),wr,rp);          umtop(VR(p1),wr,rp);
 }  }
   
 void Ptracemod_gf2n(arg,rp)  void Ptracemod_gf2n(NODE arg,P *rp)
 NODE arg;  
 P *rp;  
 {  {
         UP g,f,r;          UP g,f,r;
   
Line 1465  P *rp;
Line 1565  P *rp;
         ptoup((P)ARG1(arg),&f);          ptoup((P)ARG1(arg),&f);
         tracemodup_gf2n(g,f,(Q)ARG2(arg),&r);          tracemodup_gf2n(g,f,(Q)ARG2(arg),&r);
         uptop(r,rp);          uptop(r,rp);
   }
   
   void Pumul_specialmod(NODE arg,P *rp)
   {
           P a1,a2;
           UP p1,p2,r;
           int i,nmod;
           int *modind;
           NODE t,n;
   
           a1 = (P)ARG0(arg); a2 = (P)ARG1(arg);
           if ( !a1 || !a2 || NUM(a1) || NUM(a2) )
                   mulp(CO,a1,a2,rp);
           else {
                   if ( !uzpcheck((Obj)a1) || !uzpcheck((Obj)a2) || VR(a1) != VR(a2) )
                           error("umul_specialmod : invalid argument");
                   ptoup(a1,&p1);
                   ptoup(a2,&p2);
                   n = BDY((LIST)ARG2(arg));
                   nmod = length(n);
                   modind = (int *)MALLOC_ATOMIC(nmod*sizeof(int));
                   for ( i = 0, t = n; i < nmod; i++, t = NEXT(t) )
                           modind[i] = QTOS((Q)BDY(t));
                   fft_mulup_specialmod_main(p1,p2,0,modind,nmod,&r);
                   uptop(r,rp);
           }
   }
   
   void Pusquare_specialmod(NODE arg,P *rp)
   {
           P a1;
           UP p1,r;
           int i,nmod;
           int *modind;
           NODE t,n;
   
           a1 = (P)ARG0(arg);
           if ( !a1 || NUM(a1) )
                   mulp(CO,a1,a1,rp);
           else {
                   if ( !uzpcheck((Obj)a1) )
                           error("usquare_specialmod : invalid argument");
                   ptoup(a1,&p1);
                   n = BDY((LIST)ARG1(arg));
                   nmod = length(n);
                   modind = (int *)MALLOC_ATOMIC(nmod*sizeof(int));
                   for ( i = 0, t = n; i < nmod; i++, t = NEXT(t) )
                           modind[i] = QTOS((Q)BDY(t));
                   fft_mulup_specialmod_main(p1,p1,0,modind,nmod,&r);
                   uptop(r,rp);
           }
   }
   
   void Putmul_specialmod(NODE arg,P *rp)
   {
           P a1,a2;
           UP p1,p2,r;
           int i,nmod;
           int *modind;
           NODE t,n;
   
           a1 = (P)ARG0(arg); a2 = (P)ARG1(arg);
           if ( !a1 || !a2 || NUM(a1) || NUM(a2) )
                   mulp(CO,a1,a2,rp);
           else {
                   if ( !uzpcheck((Obj)a1) || !uzpcheck((Obj)a2) || VR(a1) != VR(a2) )
                           error("utmul_specialmod : invalid argument");
                   ptoup(a1,&p1);
                   ptoup(a2,&p2);
                   n = BDY((LIST)ARG3(arg));
                   nmod = length(n);
                   modind = (int *)MALLOC_ATOMIC(nmod*sizeof(int));
                   for ( i = 0, t = n; i < nmod; i++, t = NEXT(t) )
                           modind[i] = QTOS((Q)BDY(t));
                   fft_mulup_specialmod_main(p1,p2,QTOS((Q)ARG2(arg))+1,modind,nmod,&r);
                   uptop(r,rp);
           }
 }  }

Legend:
Removed from v.1.1.1.1  
changed lines
  Added in v.1.17

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