Code coverage tests

This page documents the degree to which the PARI/GP source code is tested by our public test suite, distributed with the source distribution in directory src/test/. This is measured by the gcov utility; we then process gcov output using the lcov frond-end.

We test a few variants depending on Configure flags on the pari.math.u-bordeaux.fr machine (x86_64 architecture), and agregate them in the final report:

The target is to exceed 90% coverage for all mathematical modules (given that branches depending on DEBUGLEVEL or DEBUGMEM are not covered). This script is run to produce the results below.

LCOV - code coverage report
Current view: top level - basemath - FpX.c (source / functions) Hit Total Coverage
Test: PARI/GP v2.12.0 lcov report (development 23712-7b25a218b) Lines: 1306 1450 90.1 %
Date: 2019-03-24 05:44:59 Functions: 154 164 93.9 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* Copyright (C) 2007  The PARI group.
       2             : 
       3             : This file is part of the PARI/GP package.
       4             : 
       5             : PARI/GP is free software; you can redistribute it and/or modify it under the
       6             : terms of the GNU General Public License as published by the Free Software
       7             : Foundation. It is distributed in the hope that it will be useful, but WITHOUT
       8             : ANY WARRANTY WHATSOEVER.
       9             : 
      10             : Check the License for details. You should have received a copy of it, along
      11             : with the package; see the file 'COPYING'. If not, write to the Free Software
      12             : Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */
      13             : 
      14             : #include "pari.h"
      15             : #include "paripriv.h"
      16             : 
      17             : /* Not so fast arithmetic with polynomials over Fp */
      18             : 
      19             : static GEN
      20    76201369 : get_FpX_red(GEN T, GEN *B)
      21             : {
      22    76201369 :   if (typ(T)!=t_VEC) { *B=NULL; return T; }
      23      124627 :   *B = gel(T,1); return gel(T,2);
      24             : }
      25             : 
      26             : /***********************************************************************/
      27             : /**                                                                   **/
      28             : /**                              FpX                                  **/
      29             : /**                                                                   **/
      30             : /***********************************************************************/
      31             : 
      32             : /* FpX are polynomials over Z/pZ represented as t_POL with
      33             :  * t_INT coefficients.
      34             :  * 1) Coefficients should belong to {0,...,p-1}, though non-reduced
      35             :  * coefficients should work but be slower.
      36             :  *
      37             :  * 2) p is not assumed to be prime, but it is assumed that impossible divisions
      38             :  *    will not happen.
      39             :  * 3) Theses functions let some garbage on the stack, but are gerepileupto
      40             :  * compatible.
      41             :  */
      42             : 
      43             : static ulong
      44    78287303 : to_Flx(GEN *P, GEN *Q, GEN p)
      45             : {
      46    78287303 :   ulong pp = uel(p,2);
      47    78287303 :   *P = ZX_to_Flx(*P, pp);
      48    78286806 :   if(Q) *Q = ZX_to_Flx(*Q, pp);
      49    78286918 :   return pp;
      50             : }
      51             : 
      52             : static ulong
      53      828469 : to_Flxq(GEN *P, GEN *T, GEN p)
      54             : {
      55      828469 :   ulong pp = uel(p,2);
      56      828469 :   if (P) *P = ZX_to_Flx(*P, pp);
      57      828458 :   *T = ZXT_to_FlxT(*T, pp); return pp;
      58             : }
      59             : 
      60             : GEN
      61        1711 : Z_to_FpX(GEN a, GEN p, long v)
      62             : {
      63        1711 :   pari_sp av = avma;
      64        1711 :   GEN z = cgetg(3, t_POL);
      65        1711 :   GEN x = modii(a, p);
      66        1711 :   if (!signe(x)) { set_avma(av); return pol_0(v); }
      67        1711 :   z[1] = evalsigne(1) | evalvarn(v);
      68        1711 :   gel(z,2) = x; return z;
      69             : }
      70             : 
      71             : /* z in Z[X], return lift(z * Mod(1,p)), normalized*/
      72             : GEN
      73    48420077 : FpX_red(GEN z, GEN p)
      74             : {
      75    48420077 :   long i, l = lg(z);
      76    48420077 :   GEN x = cgetg(l, t_POL);
      77    48455260 :   for (i=2; i<l; i++) gel(x,i) = modii(gel(z,i),p);
      78    48420816 :   x[1] = z[1]; return FpX_renormalize(x,l);
      79             : }
      80             : 
      81             : GEN
      82      292156 : FpXV_red(GEN x, GEN p)
      83      292156 : { pari_APPLY_type(t_VEC, FpX_red(gel(x,i), p)) }
      84             : 
      85             : GEN
      86     1535373 : FpXT_red(GEN x, GEN p)
      87             : {
      88     1535373 :   if (typ(x) == t_POL)
      89     1450794 :     return FpX_red(x, p);
      90             :   else
      91       84579 :     pari_APPLY_type(t_VEC, FpXT_red(gel(x,i), p))
      92             : }
      93             : 
      94             : GEN
      95      329185 : FpX_normalize(GEN z, GEN p)
      96             : {
      97      329185 :   GEN p1 = leading_coeff(z);
      98      329185 :   if (lg(z) == 2 || equali1(p1)) return z;
      99       57660 :   return FpX_Fp_mul_to_monic(z, Fp_inv(p1,p), p);
     100             : }
     101             : 
     102             : GEN
     103       16154 : FpX_center(GEN T, GEN p, GEN pov2)
     104             : {
     105       16154 :   long i, l = lg(T);
     106       16154 :   GEN P = cgetg(l,t_POL);
     107       16154 :   for(i=2; i<l; i++) gel(P,i) = Fp_center(gel(T,i), p, pov2);
     108       16154 :   P[1] = T[1]; return P;
     109             : }
     110             : GEN
     111     1015094 : FpX_center_i(GEN T, GEN p, GEN pov2)
     112             : {
     113     1015094 :   long i, l = lg(T);
     114     1015094 :   GEN P = cgetg(l,t_POL);
     115     1015130 :   for(i=2; i<l; i++) gel(P,i) = Fp_center_i(gel(T,i), p, pov2);
     116     1015151 :   P[1] = T[1]; return P;
     117             : }
     118             : 
     119             : GEN
     120     9524919 : FpX_add(GEN x,GEN y,GEN p)
     121             : {
     122     9524919 :   long lx = lg(x), ly = lg(y), i;
     123             :   GEN z;
     124     9524919 :   if (lx < ly) swapspec(x,y, lx,ly);
     125     9524919 :   z = cgetg(lx,t_POL); z[1] = x[1];
     126     9524919 :   for (i=2; i<ly; i++) gel(z,i) = Fp_add(gel(x,i),gel(y,i), p);
     127     9524919 :   for (   ; i<lx; i++) gel(z,i) = modii(gel(x,i), p);
     128     9524919 :   z = ZX_renormalize(z, lx);
     129     9524919 :   if (!lgpol(z)) { set_avma((pari_sp)(z + lx)); return pol_0(varn(x)); }
     130     9217332 :   return z;
     131             : }
     132             : 
     133             : static GEN
     134        7676 : Fp_red_FpX(GEN x, GEN p, long v)
     135             : {
     136             :   GEN z;
     137        7676 :   if (!signe(x)) return pol_0(v);
     138         484 :   z = cgetg(3, t_POL);
     139         484 :   gel(z,2) = Fp_red(x,p);
     140         484 :   z[1] = evalvarn(v);
     141         484 :   return FpX_renormalize(z, 3);
     142             : }
     143             : 
     144             : static GEN
     145         104 : Fp_neg_FpX(GEN x, GEN p, long v)
     146             : {
     147             :   GEN z;
     148         104 :   if (!signe(x)) return pol_0(v);
     149           0 :   z = cgetg(3, t_POL);
     150           0 :   gel(z,2) = Fp_neg(x,p);
     151           0 :   z[1] = evalvarn(v);
     152           0 :   return FpX_renormalize(z, 3);
     153             : }
     154             : 
     155             : GEN
     156      540095 : FpX_Fp_add(GEN y,GEN x,GEN p)
     157             : {
     158      540095 :   long i, lz = lg(y);
     159             :   GEN z;
     160      540095 :   if (lz == 2) return Fp_red_FpX(x,p,varn(y));
     161      532419 :   z = cgetg(lz,t_POL); z[1] = y[1];
     162      532419 :   gel(z,2) = Fp_add(gel(y,2),x, p);
     163      532419 :   if (lz == 3) z = FpX_renormalize(z,lz);
     164             :   else
     165      478239 :     for(i=3;i<lz;i++) gel(z,i) = icopy(gel(y,i));
     166      532419 :   return z;
     167             : }
     168             : GEN
     169           0 : FpX_Fp_add_shallow(GEN y,GEN x,GEN p)
     170             : {
     171           0 :   long i, lz = lg(y);
     172             :   GEN z;
     173           0 :   if (lz == 2) return scalar_ZX_shallow(x,varn(y));
     174           0 :   z = cgetg(lz,t_POL); z[1] = y[1];
     175           0 :   gel(z,2) = Fp_add(gel(y,2),x, p);
     176           0 :   if (lz == 3) z = FpX_renormalize(z,lz);
     177             :   else
     178           0 :     for(i=3;i<lz;i++) gel(z,i) = gel(y,i);
     179           0 :   return z;
     180             : }
     181             : GEN
     182      519409 : FpX_Fp_sub(GEN y,GEN x,GEN p)
     183             : {
     184      519409 :   long i, lz = lg(y);
     185             :   GEN z;
     186      519409 :   if (lz == 2) return Fp_neg_FpX(x,p,varn(y));
     187      519305 :   z = cgetg(lz,t_POL); z[1] = y[1];
     188      519305 :   gel(z,2) = Fp_sub(gel(y,2),x, p);
     189      519305 :   if (lz == 3) z = FpX_renormalize(z,lz);
     190             :   else
     191      126599 :     for(i=3;i<lz;i++) gel(z,i) = icopy(gel(y,i));
     192      519305 :   return z;
     193             : }
     194             : GEN
     195        1684 : FpX_Fp_sub_shallow(GEN y,GEN x,GEN p)
     196             : {
     197        1684 :   long i, lz = lg(y);
     198             :   GEN z;
     199        1684 :   if (lz == 2) return Fp_neg_FpX(x,p,varn(y));
     200        1684 :   z = cgetg(lz,t_POL); z[1] = y[1];
     201        1684 :   gel(z,2) = Fp_sub(gel(y,2),x, p);
     202        1684 :   if (lz == 3) z = FpX_renormalize(z,lz);
     203             :   else
     204        1548 :     for(i=3;i<lz;i++) gel(z,i) = gel(y,i);
     205        1684 :   return z;
     206             : }
     207             : 
     208             : GEN
     209       83464 : FpX_neg(GEN x,GEN p)
     210             : {
     211       83464 :   long i, lx = lg(x);
     212       83464 :   GEN y = cgetg(lx,t_POL);
     213       83464 :   y[1] = x[1];
     214       83464 :   for(i=2; i<lx; i++) gel(y,i) = Fp_neg(gel(x,i), p);
     215       83464 :   return ZX_renormalize(y, lx);
     216             : }
     217             : 
     218             : static GEN
     219     7021233 : FpX_subspec(GEN x,GEN y,GEN p, long nx, long ny)
     220             : {
     221             :   long i, lz;
     222             :   GEN z;
     223     7021233 :   if (nx >= ny)
     224             :   {
     225     5200393 :     lz = nx+2;
     226     5200393 :     z = cgetg(lz,t_POL); z[1] = 0; z += 2;
     227     5202103 :     for (i=0; i<ny; i++) gel(z,i) = Fp_sub(gel(x,i),gel(y,i), p);
     228     5200394 :     for (   ; i<nx; i++) gel(z,i) = modii(gel(x,i), p);
     229             :   }
     230             :   else
     231             :   {
     232     1820840 :     lz = ny+2;
     233     1820840 :     z = cgetg(lz,t_POL); z[1] = 0; z += 2;
     234     1820851 :     for (i=0; i<nx; i++) gel(z,i) = Fp_sub(gel(x,i),gel(y,i), p);
     235     1820840 :     for (   ; i<ny; i++) gel(z,i) = Fp_neg(gel(y,i), p);
     236             :   }
     237     7021174 :   z = FpX_renormalize(z-2, lz);
     238     7021232 :   if (!lgpol(z)) { set_avma((pari_sp)(z + lz)); return pol_0(0); }
     239     6906658 :   return z;
     240             : }
     241             : 
     242             : GEN
     243     6922924 : FpX_sub(GEN x,GEN y,GEN p)
     244             : {
     245     6922924 :   GEN z = FpX_subspec(x+2,y+2,p,lgpol(x),lgpol(y));
     246     6922933 :   setvarn(z, varn(x));
     247     6922933 :   return z;
     248             : }
     249             : 
     250             : GEN
     251       11004 : Fp_FpX_sub(GEN x, GEN y, GEN p)
     252             : {
     253       11004 :   long ly = lg(y), i;
     254             :   GEN z;
     255       11004 :   if (ly <= 3) {
     256         337 :     z = cgetg(3, t_POL);
     257         337 :     x = (ly == 3)? Fp_sub(x, gel(y,2), p): modii(x, p);
     258         337 :     if (!signe(x)) { set_avma((pari_sp)(z + 3)); return pol_0(varn(y)); }
     259         270 :     z[1] = evalsigne(1)|y[1]; gel(z,2) = x; return z;
     260             :   }
     261       10667 :   z = cgetg(ly,t_POL);
     262       10667 :   gel(z,2) = Fp_sub(x, gel(y,2), p);
     263       10667 :   for (i = 3; i < ly; i++) gel(z,i) = Fp_neg(gel(y,i), p);
     264       10667 :   z = ZX_renormalize(z, ly);
     265       10667 :   if (!lgpol(z)) { set_avma((pari_sp)(z + ly)); return pol_0(varn(x)); }
     266       10667 :   z[1] = y[1]; return z;
     267             : }
     268             : 
     269             : GEN
     270           0 : FpX_convol(GEN x, GEN y, GEN p)
     271             : {
     272           0 :   long lx = lg(x), ly = lg(y), i;
     273             :   GEN z;
     274           0 :   if (lx < ly) swapspec(x,y, lx,ly);
     275           0 :   z = cgetg(lx,t_POL); z[1] = x[1];
     276           0 :   for (i=2; i<ly; i++) gel(z,i) = Fp_mul(gel(x,i),gel(y,i), p);
     277           0 :   for (   ; i<lx; i++) gel(z,i) = modii(gel(x,i), p);
     278           0 :   z = ZX_renormalize(z, lx);
     279           0 :   if (!lgpol(z)) { set_avma((pari_sp)(z + lx)); return pol_0(varn(x)); }
     280           0 :   return z;
     281             : }
     282             : 
     283             : GEN
     284    49212984 : FpX_mul(GEN x,GEN y,GEN p)
     285             : {
     286    49212984 :   if (lgefint(p) == 3)
     287             :   {
     288    36996364 :     ulong pp = to_Flx(&x, &y, p);
     289    36996364 :     return Flx_to_ZX(Flx_mul(x, y, pp));
     290             :   }
     291    12216620 :   return FpX_red(ZX_mul(x, y), p);
     292             : }
     293             : 
     294             : GEN
     295     1944523 : FpX_mulspec(GEN a, GEN b, GEN p, long na, long nb)
     296     1944523 : { return FpX_red(ZX_mulspec(a, b, na, nb), p); }
     297             : 
     298             : GEN
     299     4215974 : FpX_sqr(GEN x,GEN p)
     300             : {
     301     4215974 :   if (lgefint(p) == 3)
     302             :   {
     303      164629 :     ulong pp = to_Flx(&x, NULL, p);
     304      164625 :     return Flx_to_ZX(Flx_sqr(x, pp));
     305             :   }
     306     4051345 :   return FpX_red(ZX_sqr(x), p);
     307             : }
     308             : 
     309             : GEN
     310     1243297 : FpX_mulu(GEN y, ulong x,GEN p)
     311             : {
     312             :   GEN z;
     313             :   long i, l;
     314     1243297 :   x = umodui(x, p);
     315     1243297 :   if (!x) return zeropol(varn(y));
     316     1243227 :   z = cgetg_copy(y, &l); z[1] = y[1];
     317     1243227 :   for(i=2; i<l; i++) gel(z,i) = Fp_mulu(gel(y,i), x, p);
     318     1243227 :   return z;
     319             : }
     320             : 
     321             : GEN
     322     3419024 : FpX_Fp_mulspec(GEN y,GEN x,GEN p,long ly)
     323             : {
     324             :   GEN z;
     325             :   long i;
     326     3419024 :   if (!signe(x)) return pol_0(0);
     327     3255015 :   z = cgetg(ly+2,t_POL); z[1] = evalsigne(1);
     328     3255015 :   for(i=0; i<ly; i++) gel(z,i+2) = Fp_mul(gel(y,i), x, p);
     329     3255015 :   return ZX_renormalize(z, ly+2);
     330             : }
     331             : 
     332             : GEN
     333     3412443 : FpX_Fp_mul(GEN y,GEN x,GEN p)
     334             : {
     335     3412443 :   GEN z = FpX_Fp_mulspec(y+2,x,p,lgpol(y));
     336     3412443 :   setvarn(z, varn(y)); return z;
     337             : }
     338             : 
     339             : GEN
     340       57660 : FpX_Fp_mul_to_monic(GEN y,GEN x,GEN p)
     341             : {
     342             :   GEN z;
     343             :   long i, l;
     344       57660 :   z = cgetg_copy(y, &l); z[1] = y[1];
     345       57660 :   for(i=2; i<l-1; i++) gel(z,i) = Fp_mul(gel(y,i), x, p);
     346       57660 :   gel(z,l-1) = gen_1; return z;
     347             : }
     348             : 
     349             : struct _FpXQ {
     350             :   GEN T, p, aut;
     351             : };
     352             : 
     353             : static GEN
     354       62229 : _FpX_sqr(void *data, GEN x)
     355             : {
     356       62229 :   struct _FpXQ *D = (struct _FpXQ*)data;
     357       62229 :   return FpX_sqr(x, D->p);
     358             : }
     359             : static GEN
     360       93596 : _FpX_mul(void *data, GEN x, GEN y)
     361             : {
     362       93596 :   struct _FpXQ *D = (struct _FpXQ*)data;
     363       93596 :   return FpX_mul(x,y, D->p);
     364             : }
     365             : 
     366             : GEN
     367      306917 : FpX_powu(GEN x, ulong n, GEN p)
     368             : {
     369             :   struct _FpXQ D;
     370      306917 :   if (n==0) return pol_1(varn(x));
     371       50129 :   D.p = p;
     372       50129 :   return gen_powu(x, n, (void *)&D, _FpX_sqr, _FpX_mul);
     373             : }
     374             : 
     375             : GEN
     376        1265 : FpX_halve(GEN y, GEN p)
     377             : {
     378             :   GEN z;
     379             :   long i, l;
     380        1265 :   z = cgetg_copy(y, &l); z[1] = y[1];
     381        1265 :   for(i=2; i<l; i++) gel(z,i) = Fp_halve(gel(y,i), p);
     382        1265 :   return z;
     383             : }
     384             : 
     385             : static GEN
     386    65910961 : FpX_divrem_basecase(GEN x, GEN y, GEN p, GEN *pr)
     387             : {
     388             :   long vx, dx, dy, dy1, dz, i, j, sx, lr;
     389             :   pari_sp av0, av;
     390             :   GEN z,p1,rem,lead;
     391             : 
     392    65910961 :   if (!signe(y)) pari_err_INV("FpX_divrem",y);
     393    65910961 :   vx = varn(x);
     394    65910961 :   dy = degpol(y);
     395    65910943 :   dx = degpol(x);
     396    65910941 :   if (dx < dy)
     397             :   {
     398       66287 :     if (pr)
     399             :     {
     400       66006 :       av0 = avma; x = FpX_red(x, p);
     401       66006 :       if (pr == ONLY_DIVIDES) { set_avma(av0); return signe(x)? NULL: pol_0(vx); }
     402       66006 :       if (pr == ONLY_REM) return x;
     403       66006 :       *pr = x;
     404             :     }
     405       66287 :     return pol_0(vx);
     406             :   }
     407    65844654 :   lead = leading_coeff(y);
     408    65844812 :   if (!dy) /* y is constant */
     409             :   {
     410      289840 :     if (pr && pr != ONLY_DIVIDES)
     411             :     {
     412      286114 :       if (pr == ONLY_REM) return pol_0(vx);
     413      269400 :       *pr = pol_0(vx);
     414             :     }
     415      273126 :     av0 = avma;
     416      273126 :     if (equali1(lead)) return FpX_red(x, p);
     417      268620 :     else return gerepileupto(av0, FpX_Fp_mul(x, Fp_inv(lead,p), p));
     418             :   }
     419    65554972 :   av0 = avma; dz = dx-dy;
     420    65554972 :   if (lgefint(p) == 3)
     421             :   { /* assume ab != 0 mod p */
     422    40590974 :     ulong pp = to_Flx(&x, &y, p);
     423    40590534 :     z = Flx_divrem(x, y, pp, pr);
     424    40589120 :     set_avma(av0); /* HACK: assume pr last on stack, then z */
     425    40589069 :     if (!z) return NULL;
     426    40588992 :     z = leafcopy(z);
     427    40589187 :     if (pr && pr != ONLY_DIVIDES && pr != ONLY_REM)
     428             :     {
     429     2288854 :       *pr = leafcopy(*pr);
     430     2288854 :       *pr = Flx_to_ZX_inplace(*pr);
     431             :     }
     432    40589187 :     return Flx_to_ZX_inplace(z);
     433             :   }
     434    24963998 :   lead = equali1(lead)? NULL: gclone(Fp_inv(lead,p));
     435    24963899 :   set_avma(av0);
     436    24963899 :   z=cgetg(dz+3,t_POL); z[1] = x[1];
     437    24963899 :   x += 2; y += 2; z += 2;
     438    24963899 :   for (dy1=dy-1; dy1>=0 && !signe(gel(y, dy1)); dy1--);
     439             : 
     440    24963899 :   p1 = gel(x,dx); av = avma;
     441    24963899 :   gel(z,dz) = lead? gerepileuptoint(av, Fp_mul(p1,lead, p)): icopy(p1);
     442    59046190 :   for (i=dx-1; i>=dy; i--)
     443             :   {
     444    34082291 :     av=avma; p1=gel(x,i);
     445   340290880 :     for (j=i-dy1; j<=i && j<=dz; j++)
     446   306208589 :       p1 = subii(p1, mulii(gel(z,j),gel(y,i-j)));
     447    34082291 :     if (lead) p1 = mulii(p1,lead);
     448    34082291 :     gel(z,i-dy) = gerepileuptoint(av,modii(p1, p));
     449             :   }
     450    24963899 :   if (!pr) { guncloneNULL(lead); return z-2; }
     451             : 
     452    24930308 :   rem = (GEN)avma; av = (pari_sp)new_chunk(dx+3);
     453    25822202 :   for (sx=0; ; i--)
     454             :   {
     455    26714096 :     p1 = gel(x,i);
     456    85461711 :     for (j=maxss(0,i-dy1); j<=i && j<=dz; j++)
     457    59639509 :       p1 = subii(p1, mulii(gel(z,j),gel(y,i-j)));
     458    25822202 :     p1 = modii(p1,p); if (signe(p1)) { sx = 1; break; }
     459      975162 :     if (!i) break;
     460      891894 :     set_avma(av);
     461             :   }
     462    24930308 :   if (pr == ONLY_DIVIDES)
     463             :   {
     464           0 :     guncloneNULL(lead);
     465           0 :     if (sx) return gc_NULL(av0);
     466           0 :     set_avma((pari_sp)rem); return z-2;
     467             :   }
     468    24930308 :   lr=i+3; rem -= lr;
     469    24930308 :   rem[0] = evaltyp(t_POL) | evallg(lr);
     470    24930308 :   rem[1] = z[-1];
     471    24930308 :   p1 = gerepileuptoint((pari_sp)rem, p1);
     472    24930308 :   rem += 2; gel(rem,i) = p1;
     473    90724196 :   for (i--; i>=0; i--)
     474             :   {
     475    65793888 :     av=avma; p1 = gel(x,i);
     476   450857216 :     for (j=maxss(0,i-dy1); j<=i && j<=dz; j++)
     477   385063328 :       p1 = subii(p1, mulii(gel(z,j),gel(y,i-j)));
     478    65793888 :     gel(rem,i) = gerepileuptoint(av, modii(p1,p));
     479             :   }
     480    24930308 :   rem -= 2;
     481    24930308 :   guncloneNULL(lead);
     482    24930308 :   if (!sx) (void)FpX_renormalize(rem, lr);
     483    24930308 :   if (pr == ONLY_REM) return gerepileupto(av0,rem);
     484      576125 :   *pr = rem; return z-2;
     485             : }
     486             : 
     487             : GEN
     488       28302 : FpX_div_by_X_x(GEN a, GEN x, GEN p, GEN *r)
     489             : {
     490       28302 :   long l = lg(a)-1, i;
     491       28302 :   GEN z = cgetg(l, t_POL);
     492       28302 :   z[1] = evalsigne(1) | evalvarn(0);
     493       28302 :   gel(z, l-1) = gel(a,l);
     494      723374 :   for (i=l-2; i>1; i--) /* z[i] = a[i+1] + x*z[i+1] */
     495      695072 :     gel(z, i) = Fp_addmul(gel(a,i+1), x, gel(z,i+1), p);
     496       28302 :   if (r) *r = Fp_addmul(gel(a,2), x, gel(z,2), p);
     497       28302 :   return z;
     498             : }
     499             : 
     500             : static GEN
     501       70056 : _FpX_divrem(void * E, GEN x, GEN y, GEN *r)
     502             : {
     503       70056 :   struct _FpXQ *D = (struct _FpXQ*) E;
     504       70056 :   return FpX_divrem(x, y, D->p, r);
     505             : }
     506             : static GEN
     507       10241 : _FpX_add(void * E, GEN x, GEN y) {
     508       10241 :   struct _FpXQ *D = (struct _FpXQ*) E;
     509       10241 :   return FpX_add(x, y, D->p);
     510             : }
     511             : 
     512             : static struct bb_ring FpX_ring = { _FpX_add,_FpX_mul,_FpX_sqr };
     513             : 
     514             : GEN
     515        5915 : FpX_digits(GEN x, GEN T, GEN p)
     516             : {
     517        5915 :   pari_sp av = avma;
     518             :   struct _FpXQ D;
     519        5915 :   long d = degpol(T), n = (lgpol(x)+d-1)/d;
     520             :   GEN z;
     521        5915 :   D.p = p;
     522        5915 :   z = gen_digits(x,T,n,(void *)&D, &FpX_ring, _FpX_divrem);
     523        5915 :   return gerepileupto(av, z);
     524             : }
     525             : 
     526             : GEN
     527        2303 : FpXV_FpX_fromdigits(GEN x, GEN T, GEN p)
     528             : {
     529        2303 :   pari_sp av = avma;
     530             :   struct _FpXQ D;
     531             :   GEN z;
     532        2303 :   D.p = p;
     533        2303 :   z = gen_fromdigits(x,T,(void *)&D, &FpX_ring);
     534        2303 :   return gerepileupto(av, z);
     535             : }
     536             : 
     537             : long
     538       27447 : FpX_valrem(GEN x, GEN t, GEN p, GEN *py)
     539             : {
     540       27447 :   pari_sp av=avma;
     541             :   long k;
     542             :   GEN r, y;
     543             : 
     544       80559 :   for (k=0; ; k++)
     545             :   {
     546      133671 :     y = FpX_divrem(x, t, p, &r);
     547       80559 :     if (signe(r)) break;
     548       53112 :     x = y;
     549             :   }
     550       27447 :   *py = gerepilecopy(av,x);
     551       27447 :   return k;
     552             : }
     553             : 
     554             : static GEN
     555         443 : FpX_halfgcd_basecase(GEN a, GEN b, GEN p)
     556             : {
     557         443 :   pari_sp av=avma;
     558             :   GEN u,u1,v,v1;
     559         443 :   long vx = varn(a);
     560         443 :   long n = lgpol(a)>>1;
     561         443 :   u1 = v = pol_0(vx);
     562         443 :   u = v1 = pol_1(vx);
     563        4586 :   while (lgpol(b)>n)
     564             :   {
     565        3700 :     GEN r, q = FpX_divrem(a,b,p, &r);
     566        3700 :     a = b; b = r; swap(u,u1); swap(v,v1);
     567        3700 :     u1 = FpX_sub(u1, FpX_mul(u, q, p), p);
     568        3700 :     v1 = FpX_sub(v1, FpX_mul(v, q ,p), p);
     569        3700 :     if (gc_needed(av,2))
     570             :     {
     571           0 :       if (DEBUGMEM>1) pari_warn(warnmem,"FpX_halfgcd (d = %ld)",degpol(b));
     572           0 :       gerepileall(av,6, &a,&b,&u1,&v1,&u,&v);
     573             :     }
     574             :   }
     575         443 :   return gerepilecopy(av, mkmat2(mkcol2(u,u1), mkcol2(v,v1)));
     576             : }
     577             : static GEN
     578         388 : FpX_addmulmul(GEN u, GEN v, GEN x, GEN y, GEN p)
     579             : {
     580         388 :   return FpX_add(FpX_mul(u, x, p),FpX_mul(v, y, p), p);
     581             : }
     582             : 
     583             : static GEN
     584         190 : FpXM_FpX_mul2(GEN M, GEN x, GEN y, GEN p)
     585             : {
     586         190 :   GEN res = cgetg(3, t_COL);
     587         190 :   gel(res, 1) = FpX_addmulmul(gcoeff(M,1,1), gcoeff(M,1,2), x, y, p);
     588         190 :   gel(res, 2) = FpX_addmulmul(gcoeff(M,2,1), gcoeff(M,2,2), x, y, p);
     589         190 :   return res;
     590             : }
     591             : 
     592             : static GEN
     593         169 : FpXM_mul2(GEN A, GEN B, GEN p)
     594             : {
     595         169 :   GEN A11=gcoeff(A,1,1),A12=gcoeff(A,1,2), B11=gcoeff(B,1,1),B12=gcoeff(B,1,2);
     596         169 :   GEN A21=gcoeff(A,2,1),A22=gcoeff(A,2,2), B21=gcoeff(B,2,1),B22=gcoeff(B,2,2);
     597         169 :   GEN M1 = FpX_mul(FpX_add(A11,A22, p), FpX_add(B11,B22, p), p);
     598         169 :   GEN M2 = FpX_mul(FpX_add(A21,A22, p), B11, p);
     599         169 :   GEN M3 = FpX_mul(A11, FpX_sub(B12,B22, p), p);
     600         169 :   GEN M4 = FpX_mul(A22, FpX_sub(B21,B11, p), p);
     601         169 :   GEN M5 = FpX_mul(FpX_add(A11,A12, p), B22, p);
     602         169 :   GEN M6 = FpX_mul(FpX_sub(A21,A11, p), FpX_add(B11,B12, p), p);
     603         169 :   GEN M7 = FpX_mul(FpX_sub(A12,A22, p), FpX_add(B21,B22, p), p);
     604         169 :   GEN T1 = FpX_add(M1,M4, p), T2 = FpX_sub(M7,M5, p);
     605         169 :   GEN T3 = FpX_sub(M1,M2, p), T4 = FpX_add(M3,M6, p);
     606         169 :   retmkmat2(mkcol2(FpX_add(T1,T2, p), FpX_add(M2,M4, p)),
     607             :             mkcol2(FpX_add(M3,M5, p), FpX_add(T3,T4, p)));
     608             : }
     609             : 
     610             : /* Return [0,1;1,-q]*M */
     611             : static GEN
     612         165 : FpX_FpXM_qmul(GEN q, GEN M, GEN p)
     613             : {
     614         165 :   GEN u, v, res = cgetg(3, t_MAT);
     615         165 :   u = FpX_sub(gcoeff(M,1,1), FpX_mul(gcoeff(M,2,1), q, p), p);
     616         165 :   gel(res,1) = mkcol2(gcoeff(M,2,1), u);
     617         165 :   v = FpX_sub(gcoeff(M,1,2), FpX_mul(gcoeff(M,2,2), q, p), p);
     618         165 :   gel(res,2) = mkcol2(gcoeff(M,2,2), v);
     619         165 :   return res;
     620             : }
     621             : 
     622             : static GEN
     623           4 : matid2_FpXM(long v)
     624             : {
     625           4 :   retmkmat2(mkcol2(pol_1(v),pol_0(v)),
     626             :             mkcol2(pol_0(v),pol_1(v)));
     627             : }
     628             : 
     629             : static GEN
     630         186 : FpX_halfgcd_split(GEN x, GEN y, GEN p)
     631             : {
     632         186 :   pari_sp av=avma;
     633             :   GEN R, S, V;
     634             :   GEN y1, r, q;
     635         186 :   long l = lgpol(x), n = l>>1, k;
     636         186 :   if (lgpol(y)<=n) return matid2_FpXM(varn(x));
     637         186 :   R = FpX_halfgcd(RgX_shift_shallow(x,-n),RgX_shift_shallow(y,-n),p);
     638         186 :   V = FpXM_FpX_mul2(R,x,y,p); y1 = gel(V,2);
     639         186 :   if (lgpol(y1)<=n) return gerepilecopy(av, R);
     640         165 :   q = FpX_divrem(gel(V,1), y1, p, &r);
     641         165 :   k = 2*n-degpol(y1);
     642         165 :   S = FpX_halfgcd(RgX_shift_shallow(y1,-k), RgX_shift_shallow(r,-k),p);
     643         165 :   return gerepileupto(av, FpXM_mul2(S,FpX_FpXM_qmul(q,R,p),p));
     644             : }
     645             : 
     646             : /* Return M in GL_2(Fp[X]) such that:
     647             : if [a',b']~=M*[a,b]~ then degpol(a')>= (lgpol(a)>>1) >degpol(b')
     648             : */
     649             : 
     650             : static GEN
     651         629 : FpX_halfgcd_i(GEN x, GEN y, GEN p)
     652             : {
     653         629 :   if (lg(x)<=FpX_HALFGCD_LIMIT) return FpX_halfgcd_basecase(x,y,p);
     654         186 :   return FpX_halfgcd_split(x,y,p);
     655             : }
     656             : 
     657             : GEN
     658         748 : FpX_halfgcd(GEN x, GEN y, GEN p)
     659             : {
     660         748 :   pari_sp av = avma;
     661             :   GEN M,q,r;
     662         748 :   if (lgefint(p)==3)
     663             :   {
     664         119 :     ulong pp = to_Flx(&x, &y, p);
     665         119 :     M = FlxM_to_ZXM(Flx_halfgcd(x, y, pp));
     666             :   }
     667             :   else
     668             :   {
     669         629 :     if (!signe(x))
     670             :     {
     671           0 :       long v = varn(x);
     672           0 :       retmkmat2(mkcol2(pol_0(v),pol_1(v)),
     673             :                 mkcol2(pol_1(v),pol_0(v)));
     674             :     }
     675         629 :     if (degpol(y)<degpol(x)) return FpX_halfgcd_i(x,y,p);
     676          11 :     q = FpX_divrem(y,x,p,&r);
     677          11 :     M = FpX_halfgcd_i(x,r,p);
     678          11 :     gcoeff(M,1,1) = FpX_sub(gcoeff(M,1,1), FpX_mul(q, gcoeff(M,1,2), p), p);
     679          11 :     gcoeff(M,2,1) = FpX_sub(gcoeff(M,2,1), FpX_mul(q, gcoeff(M,2,2), p), p);
     680             :   }
     681         130 :   return gerepilecopy(av, M);
     682             : }
     683             : 
     684             : static GEN
     685       33783 : FpX_gcd_basecase(GEN a, GEN b, GEN p)
     686             : {
     687       33783 :   pari_sp av = avma, av0=avma;
     688      432472 :   while (signe(b))
     689             :   {
     690             :     GEN c;
     691      365005 :     if (gc_needed(av0,2))
     692             :     {
     693           0 :       if (DEBUGMEM>1) pari_warn(warnmem,"FpX_gcd (d = %ld)",degpol(b));
     694           0 :       gerepileall(av0,2, &a,&b);
     695             :     }
     696      365005 :     av = avma; c = FpX_rem(a,b,p); a=b; b=c;
     697             :   }
     698       33684 :   set_avma(av); return a;
     699             : }
     700             : 
     701             : GEN
     702      426576 : FpX_gcd(GEN x, GEN y, GEN p)
     703             : {
     704      426576 :   pari_sp av = avma;
     705      426576 :   if (lgefint(p)==3)
     706             :   {
     707             :     ulong pp;
     708      392281 :     (void)new_chunk((lg(x) + lg(y)) << 2); /* scratch space */
     709      392281 :     pp = to_Flx(&x, &y, p);
     710      392281 :     x = Flx_gcd(x, y, pp);
     711      392281 :     set_avma(av); return Flx_to_ZX(x);
     712             :   }
     713       34295 :   x = FpX_red(x, p);
     714       34295 :   y = FpX_red(y, p);
     715       34295 :   if (!signe(x)) return gerepileupto(av, y);
     716       67566 :   while (lg(y)>FpX_GCD_LIMIT)
     717             :   {
     718             :     GEN c;
     719           0 :     if (lgpol(y)<=(lgpol(x)>>1))
     720             :     {
     721           0 :       GEN r = FpX_rem(x, y, p);
     722           0 :       x = y; y = r;
     723             :     }
     724           0 :     c = FpXM_FpX_mul2(FpX_halfgcd(x,y, p), x, y, p);
     725           0 :     x = gel(c,1); y = gel(c,2);
     726           0 :     gerepileall(av,2,&x,&y);
     727             :   }
     728       33783 :   return gerepileupto(av, FpX_gcd_basecase(x,y,p));
     729             : }
     730             : 
     731             : /* Return NULL if gcd can be computed else return a factor of p */
     732             : GEN
     733         128 : FpX_gcd_check(GEN x, GEN y, GEN p)
     734             : {
     735         128 :   pari_sp av = avma;
     736             :   GEN a,b,c;
     737             : 
     738         128 :   a = FpX_red(x, p);
     739         128 :   b = FpX_red(y, p);
     740        1684 :   while (signe(b))
     741             :   {
     742        1491 :     GEN g = gcdii(p, leading_coeff(b));
     743        1491 :     if (!equali1(g)) return gerepileuptoint(av,g);
     744        1428 :     c = FpX_rem(a,b,p); a = b; b = c;
     745        1428 :     if (gc_needed(av,1))
     746             :     {
     747           0 :       if (DEBUGMEM>1)
     748           0 :         pari_warn(warnmem,"FpX_gcd_check (d = %ld)",degpol(b));
     749           0 :       gerepileall(av,2,&a,&b);
     750             :     }
     751             :   }
     752          65 :   return gc_NULL(av);
     753             : }
     754             : 
     755             : static GEN
     756      269400 : FpX_extgcd_basecase(GEN a, GEN b, GEN p, GEN *ptu, GEN *ptv)
     757             : {
     758      269400 :   pari_sp av=avma;
     759             :   GEN u,v,d,d1,v1;
     760      269400 :   long vx = varn(a);
     761      269400 :   d = a; d1 = b;
     762      269400 :   v = pol_0(vx); v1 = pol_1(vx);
     763     1193799 :   while (signe(d1))
     764             :   {
     765      654999 :     GEN r, q = FpX_divrem(d,d1,p, &r);
     766      654999 :     v = FpX_sub(v,FpX_mul(q,v1,p),p);
     767      654999 :     u=v; v=v1; v1=u;
     768      654999 :     u=r; d=d1; d1=u;
     769      654999 :     if (gc_needed(av,2))
     770             :     {
     771           0 :       if (DEBUGMEM>1) pari_warn(warnmem,"FpX_extgcd (d = %ld)",degpol(d));
     772           0 :       gerepileall(av,5, &d,&d1,&u,&v,&v1);
     773             :     }
     774             :   }
     775      269400 :   if (ptu) *ptu = FpX_div(FpX_sub(d,FpX_mul(b,v,p),p),a,p);
     776      269400 :   *ptv = v; return d;
     777             : }
     778             : 
     779             : static GEN
     780           4 : FpX_extgcd_halfgcd(GEN x, GEN y, GEN p, GEN *ptu, GEN *ptv)
     781             : {
     782           4 :   pari_sp av=avma;
     783           4 :   GEN u,v,R = matid2_FpXM(varn(x));
     784          12 :   while (lg(y)>FpX_EXTGCD_LIMIT)
     785             :   {
     786             :     GEN M, c;
     787           4 :     if (lgpol(y)<=(lgpol(x)>>1))
     788             :     {
     789           0 :       GEN r, q = FpX_divrem(x, y, p, &r);
     790           0 :       x = y; y = r;
     791           0 :       R = FpX_FpXM_qmul(q, R, p);
     792             :     }
     793           4 :     M = FpX_halfgcd(x,y, p);
     794           4 :     c = FpXM_FpX_mul2(M, x,y, p);
     795           4 :     R = FpXM_mul2(M, R, p);
     796           4 :     x = gel(c,1); y = gel(c,2);
     797           4 :     gerepileall(av,3,&x,&y,&R);
     798             :   }
     799           4 :   y = FpX_extgcd_basecase(x,y,p,&u,&v);
     800           4 :   if (ptu) *ptu = FpX_addmulmul(u,v,gcoeff(R,1,1),gcoeff(R,2,1),p);
     801           4 :   *ptv = FpX_addmulmul(u,v,gcoeff(R,1,2),gcoeff(R,2,2),p);
     802           4 :   return y;
     803             : }
     804             : 
     805             : /* x and y in Z[X], return lift(gcd(x mod p, y mod p)). Set u and v st
     806             :  * ux + vy = gcd (mod p) */
     807             : GEN
     808      409843 : FpX_extgcd(GEN x, GEN y, GEN p, GEN *ptu, GEN *ptv)
     809             : {
     810             :   GEN d;
     811      409843 :   pari_sp ltop=avma;
     812      409843 :   if (lgefint(p)==3)
     813             :   {
     814      140443 :     ulong pp = to_Flx(&x, &y, p);
     815      140443 :     d = Flx_extgcd(x,y, pp, ptu,ptv);
     816      140443 :     d = Flx_to_ZX(d);
     817      140443 :     if (ptu) *ptu=Flx_to_ZX(*ptu);
     818      140443 :     *ptv=Flx_to_ZX(*ptv);
     819             :   }
     820             :   else
     821             :   {
     822      269400 :     x = FpX_red(x, p);
     823      269400 :     y = FpX_red(y, p);
     824      269400 :     if (lg(y)>FpX_EXTGCD_LIMIT)
     825           4 :       d = FpX_extgcd_halfgcd(x, y, p, ptu, ptv);
     826             :     else
     827      269396 :       d = FpX_extgcd_basecase(x, y, p, ptu, ptv);
     828             :   }
     829      409843 :   gerepileall(ltop,ptu?3:2,&d,ptv,ptu);
     830      409843 :   return d;
     831             : }
     832             : 
     833             : GEN
     834       15316 : FpX_rescale(GEN P, GEN h, GEN p)
     835             : {
     836       15316 :   long i, l = lg(P);
     837       15316 :   GEN Q = cgetg(l,t_POL), hi = h;
     838       15316 :   gel(Q,l-1) = gel(P,l-1);
     839       68736 :   for (i=l-2; i>=2; i--)
     840             :   {
     841       68736 :     gel(Q,i) = Fp_mul(gel(P,i), hi, p);
     842       68736 :     if (i == 2) break;
     843       53420 :     hi = Fp_mul(hi,h, p);
     844             :   }
     845       15316 :   Q[1] = P[1]; return Q;
     846             : }
     847             : 
     848             : GEN
     849      362233 : FpX_deriv(GEN x, GEN p) { return FpX_red(ZX_deriv(x), p); }
     850             : 
     851             : GEN
     852         861 : FpX_integ(GEN x, GEN p)
     853             : {
     854         861 :   long i, lx = lg(x);
     855             :   GEN y;
     856         861 :   if (lx == 2) return ZX_copy(x);
     857         679 :   y = cgetg(lx+1, t_POL); y[1] = x[1];
     858         679 :   gel(y,2) = gen_0;
     859        5600 :   for (i=3; i<=lx; i++)
     860        4921 :     gel(y,i) = signe(gel(x,i-1))? Fp_divu(gel(x,i-1), i-2, p): gen_0;
     861         679 :   return ZX_renormalize(y, lx+1);;
     862             : }
     863             : 
     864             : INLINE GEN
     865       45173 : FpXn_recip(GEN P, long n)
     866       45173 : { return RgXn_recip_shallow(P, n); }
     867             : 
     868             : GEN
     869       44991 : FpX_Newton(GEN P, long n, GEN p)
     870             : {
     871       44991 :   pari_sp av = avma;
     872       44991 :   GEN dP = FpX_deriv(P, p);
     873       44991 :   GEN Q = FpXn_recip(FpX_div(RgX_shift(dP,n), P, p), n);
     874       44991 :   return gerepilecopy(av, Q);
     875             : }
     876             : 
     877             : GEN
     878         182 : FpX_fromNewton(GEN P, GEN p)
     879             : {
     880         182 :   pari_sp av = avma;
     881         182 :   long n = itos(modii(constant_coeff(P), p))+1;
     882         182 :   GEN z = FpX_neg(FpX_integ(RgX_shift(P,-1),p),p);
     883         182 :   GEN Q = FpXn_recip(FpXn_exp(z, n, p), n);
     884         182 :   return gerepilecopy(av, Q);
     885             : }
     886             : 
     887             : GEN
     888         364 : FpX_invLaplace(GEN x, GEN p)
     889             : {
     890         364 :   pari_sp av = avma;
     891         364 :   long i, e = 0, l = lg(x);
     892         364 :   GEN t = gen_1, y = cgetg(l,t_POL);
     893         364 :   y[1] = x[1];
     894        4452 :   for (i=2; i<l; i++)
     895             :   {
     896        4088 :     gel(y,i) = Fp_div(gel(x,i), t, p);
     897        4088 :     e++; t = Fp_mulu(t,e,p);
     898             :   }
     899         364 :   return gerepilecopy(av, y);
     900             : }
     901             : 
     902             : GEN
     903         182 : FpX_Laplace(GEN x, GEN p)
     904             : {
     905         182 :   pari_sp av = avma;
     906         182 :   long i, e = 0, l = lg(x);
     907         182 :   GEN t = gen_1, y = cgetg(l,t_POL);
     908         182 :   y[1] = x[1];
     909        2226 :   for (i=2; i<l; i++)
     910             :   {
     911        2044 :     gel(y,i) = Fp_mul(gel(x,i), t, p);
     912        2044 :     e++; t = Fp_mulu(t,e,p);
     913             :   }
     914         182 :   return gerepilecopy(av, y);
     915             : }
     916             : 
     917             : int
     918        2892 : FpX_is_squarefree(GEN f, GEN p)
     919             : {
     920        2892 :   pari_sp av = avma;
     921        2892 :   GEN z = FpX_gcd(f,FpX_deriv(f,p),p);
     922        2892 :   set_avma(av);
     923        2892 :   return degpol(z)==0;
     924             : }
     925             : 
     926             : GEN
     927       33682 : random_FpX(long d1, long v, GEN p)
     928             : {
     929       33682 :   long i, d = d1+2;
     930       33682 :   GEN y = cgetg(d,t_POL); y[1] = evalsigne(1) | evalvarn(v);
     931       33682 :   for (i=2; i<d; i++) gel(y,i) = randomi(p);
     932       33682 :   return FpX_renormalize(y,d);
     933             : }
     934             : 
     935             : GEN
     936        5662 : FpX_dotproduct(GEN x, GEN y, GEN p)
     937             : {
     938        5662 :   long i, l = minss(lg(x), lg(y));
     939             :   pari_sp av;
     940             :   GEN c;
     941        5662 :   if (l == 2) return gen_0;
     942        5585 :   av = avma; c = mulii(gel(x,2),gel(y,2));
     943        5585 :   for (i=3; i<l; i++) c = addii(c, mulii(gel(x,i),gel(y,i)));
     944        5585 :   return gerepileuptoint(av, modii(c,p));
     945             : }
     946             : 
     947             : /* Evaluation in Fp
     948             :  * x a ZX and y an Fp, return x(y) mod p
     949             :  *
     950             :  * If p is very large (several longs) and x has small coefficients(<<p),
     951             :  * then Brent & Kung algorithm is faster. */
     952             : GEN
     953      388734 : FpX_eval(GEN x,GEN y,GEN p)
     954             : {
     955             :   pari_sp av;
     956             :   GEN p1,r,res;
     957      388734 :   long j, i=lg(x)-1;
     958      388734 :   if (i<=2 || !signe(y))
     959      129948 :     return (i==1)? gen_0: modii(gel(x,2),p);
     960      258786 :   res=cgeti(lgefint(p));
     961      258786 :   av=avma; p1=gel(x,i);
     962             :   /* specific attention to sparse polynomials (see poleval)*/
     963             :   /*You've guessed it! It's a copy-paste(tm)*/
     964     1219663 :   for (i--; i>=2; i=j-1)
     965             :   {
     966     1462845 :     for (j=i; !signe(gel(x,j)); j--)
     967      501968 :       if (j==2)
     968             :       {
     969       35162 :         if (i!=j) y = Fp_powu(y,i-j+1,p);
     970       35162 :         p1=mulii(p1,y);
     971       35162 :         goto fppoleval;/*sorry break(2) no implemented*/
     972             :       }
     973      960877 :     r = (i==j)? y: Fp_powu(y,i-j+1,p);
     974      960877 :     p1 = Fp_addmul(gel(x,j), p1, r, p);
     975      960877 :     if ((i & 7) == 0) { affii(p1, res); p1 = res; set_avma(av); }
     976             :   }
     977             :  fppoleval:
     978      258786 :   modiiz(p1,p,res);
     979      258786 :   set_avma(av); return res;
     980             : }
     981             : 
     982             : /* Tz=Tx*Ty where Tx and Ty coprime
     983             :  * return lift(chinese(Mod(x*Mod(1,p),Tx*Mod(1,p)),Mod(y*Mod(1,p),Ty*Mod(1,p))))
     984             :  * if Tz is NULL it is computed
     985             :  * As we do not return it, and the caller will frequently need it,
     986             :  * it must compute it and pass it.
     987             :  */
     988             : GEN
     989        1085 : FpX_chinese_coprime(GEN x,GEN y,GEN Tx,GEN Ty,GEN Tz,GEN p)
     990             : {
     991        1085 :   pari_sp av = avma;
     992             :   GEN ax,p1;
     993        1085 :   ax = FpX_mul(FpXQ_inv(Tx,Ty,p), Tx,p);
     994        1085 :   p1 = FpX_mul(ax, FpX_sub(y,x,p),p);
     995        1085 :   p1 = FpX_add(x,p1,p);
     996        1085 :   if (!Tz) Tz=FpX_mul(Tx,Ty,p);
     997        1085 :   p1 = FpX_rem(p1,Tz,p);
     998        1085 :   return gerepileupto(av,p1);
     999             : }
    1000             : 
    1001             : /* Res(A,B) = Res(B,R) * lc(B)^(a-r) * (-1)^(ab), with R=A%B, a=deg(A) ...*/
    1002             : GEN
    1003        6155 : FpX_resultant(GEN a, GEN b, GEN p)
    1004             : {
    1005             :   long da,db,dc;
    1006             :   pari_sp av;
    1007        6155 :   GEN c,lb, res = gen_1;
    1008             : 
    1009        6155 :   if (!signe(a) || !signe(b)) return gen_0;
    1010        6148 :   if (lgefint(p) == 3)
    1011             :   {
    1012        2509 :     pari_sp av = avma;
    1013        2509 :     ulong pp = to_Flx(&a, &b, p);
    1014        2509 :     long r = Flx_resultant(a, b, pp);
    1015        2509 :     set_avma(av);
    1016        2509 :     return utoi(r);
    1017             :   }
    1018             : 
    1019        3639 :   da = degpol(a);
    1020        3639 :   db = degpol(b);
    1021        3639 :   if (db > da)
    1022             :   {
    1023           0 :     swapspec(a,b, da,db);
    1024           0 :     if (both_odd(da,db)) res = subii(p, res);
    1025             :   }
    1026        3639 :   if (!da) return gen_1; /* = res * a[2] ^ db, since 0 <= db <= da = 0 */
    1027        3639 :   av = avma;
    1028       12159 :   while (db)
    1029             :   {
    1030        4881 :     lb = gel(b,db+2);
    1031        4881 :     c = FpX_rem(a,b, p);
    1032        4881 :     a = b; b = c; dc = degpol(c);
    1033        4881 :     if (dc < 0) { set_avma(av); return gen_0; }
    1034             : 
    1035        4881 :     if (both_odd(da,db)) res = subii(p, res);
    1036        4881 :     if (!equali1(lb)) res = Fp_mul(res, Fp_powu(lb, da - dc, p), p);
    1037        4881 :     if (gc_needed(av,2))
    1038             :     {
    1039           0 :       if (DEBUGMEM>1) pari_warn(warnmem,"FpX_resultant (da = %ld)",da);
    1040           0 :       gerepileall(av,3, &a,&b,&res);
    1041             :     }
    1042        4881 :     da = db; /* = degpol(a) */
    1043        4881 :     db = dc; /* = degpol(b) */
    1044             :   }
    1045        3639 :   res = Fp_mul(res, Fp_powu(gel(b,2), da, p), p);
    1046        3639 :   return gerepileuptoint(av, res);
    1047             : }
    1048             : 
    1049             : /* disc P = (-1)^(n(n-1)/2) lc(P)^(n - deg P' - 2) Res(P,P'), n = deg P */
    1050             : GEN
    1051          49 : FpX_disc(GEN P, GEN p)
    1052             : {
    1053          49 :   pari_sp av = avma;
    1054          49 :   GEN L, dP = FpX_deriv(P,p), D = FpX_resultant(P, dP, p);
    1055             :   long dd;
    1056          49 :   if (!signe(D)) return gen_0;
    1057          35 :   dd = degpol(P) - 2 - degpol(dP); /* >= -1; > -1 iff p | deg(P) */
    1058          35 :   L = leading_coeff(P);
    1059          35 :   if (dd && !equali1(L))
    1060           7 :     D = (dd == -1)? Fp_div(D,L,p): Fp_mul(D, Fp_powu(L, dd, p), p);
    1061          35 :   if (degpol(P) & 2) D = Fp_neg(D ,p);
    1062          35 :   return gerepileuptoint(av, D);
    1063             : }
    1064             : 
    1065             : GEN
    1066       31115 : FpXV_prod(GEN V, GEN p)
    1067             : {
    1068             :   struct _FpXQ D;
    1069       31115 :   D.p = p;
    1070       31115 :   return gen_product(V, (void *)&D, &_FpX_mul);
    1071             : }
    1072             : 
    1073             : GEN
    1074        9367 : FpV_roots_to_pol(GEN V, GEN p, long v)
    1075             : {
    1076        9367 :   pari_sp ltop=avma;
    1077             :   long i;
    1078        9367 :   GEN g=cgetg(lg(V),t_VEC);
    1079       58502 :   for(i=1;i<lg(V);i++)
    1080       49135 :     gel(g,i) = deg1pol_shallow(gen_1,modii(negi(gel(V,i)),p),v);
    1081        9367 :   return gerepileupto(ltop,FpXV_prod(g,p));
    1082             : }
    1083             : 
    1084             : /* invert all elements of x mod p using Montgomery's multi-inverse trick.
    1085             :  * Not stack-clean. */
    1086             : GEN
    1087        5299 : FpV_inv(GEN x, GEN p)
    1088             : {
    1089        5299 :   long i, lx = lg(x);
    1090        5299 :   GEN u, y = cgetg(lx, t_VEC);
    1091             : 
    1092        5299 :   gel(y,1) = gel(x,1);
    1093        5299 :   for (i=2; i<lx; i++) gel(y,i) = Fp_mul(gel(y,i-1), gel(x,i), p);
    1094             : 
    1095        5299 :   u = Fp_inv(gel(y,--i), p);
    1096      337385 :   for ( ; i > 1; i--)
    1097             :   {
    1098      332086 :     gel(y,i) = Fp_mul(u, gel(y,i-1), p);
    1099      332086 :     u = Fp_mul(u, gel(x,i), p); /* u = 1 / (x[1] ... x[i-1]) */
    1100             :   }
    1101        5299 :   gel(y,1) = u; return y;
    1102             : }
    1103             : GEN
    1104           0 : FqV_inv(GEN x, GEN T, GEN p)
    1105             : {
    1106           0 :   long i, lx = lg(x);
    1107           0 :   GEN u, y = cgetg(lx, t_VEC);
    1108             : 
    1109           0 :   gel(y,1) = gel(x,1);
    1110           0 :   for (i=2; i<lx; i++) gel(y,i) = Fq_mul(gel(y,i-1), gel(x,i), T,p);
    1111             : 
    1112           0 :   u = Fq_inv(gel(y,--i), T,p);
    1113           0 :   for ( ; i > 1; i--)
    1114             :   {
    1115           0 :     gel(y,i) = Fq_mul(u, gel(y,i-1), T,p);
    1116           0 :     u = Fq_mul(u, gel(x,i), T,p); /* u = 1 / (x[1] ... x[i-1]) */
    1117             :   }
    1118           0 :   gel(y,1) = u; return y;
    1119             : }
    1120             : 
    1121             : /***********************************************************************/
    1122             : /**                                                                   **/
    1123             : /**                      Barrett reduction                            **/
    1124             : /**                                                                   **/
    1125             : /***********************************************************************/
    1126             : 
    1127             : static GEN
    1128        1517 : FpX_invBarrett_basecase(GEN T, GEN p)
    1129             : {
    1130        1517 :   long i, l=lg(T)-1, lr = l-1, k;
    1131        1517 :   GEN r=cgetg(lr, t_POL); r[1]=T[1];
    1132        1517 :   gel(r,2) = gen_1;
    1133       86907 :   for (i=3; i<lr; i++)
    1134             :   {
    1135       85390 :     pari_sp av = avma;
    1136       85390 :     GEN u = gel(T,l-i+2);
    1137     2667242 :     for (k=3; k<i; k++)
    1138     2581852 :       u = addii(u, mulii(gel(T,l-i+k), gel(r,k)));
    1139       85390 :     gel(r,i) = gerepileupto(av, modii(negi(u), p));
    1140             :   }
    1141        1517 :   return FpX_renormalize(r,lr);
    1142             : }
    1143             : 
    1144             : /* Return new lgpol */
    1145             : static long
    1146      210798 : ZX_lgrenormalizespec(GEN x, long lx)
    1147             : {
    1148             :   long i;
    1149      235539 :   for (i = lx-1; i>=0; i--)
    1150      235538 :     if (signe(gel(x,i))) break;
    1151      210798 :   return i+1;
    1152             : }
    1153             : 
    1154             : INLINE GEN
    1155      198878 : FpX_recipspec(GEN x, long l, long n)
    1156             : {
    1157      198878 :   return RgX_recipspec_shallow(x, l, n);
    1158             : }
    1159             : 
    1160             : static GEN
    1161         688 : FpX_invBarrett_Newton(GEN T, GEN p)
    1162             : {
    1163         688 :   pari_sp av = avma;
    1164         688 :   long nold, lx, lz, lq, l = degpol(T), i, lQ;
    1165         688 :   GEN q, y, z, x = cgetg(l+2, t_POL) + 2;
    1166         689 :   ulong mask = quadratic_prec_mask(l-2); /* assume l > 2 */
    1167         690 :   for (i=0;i<l;i++) gel(x,i) = gen_0;
    1168         690 :   q = FpX_recipspec(T+2,l+1,l+1); lQ = lgpol(q); q+=2;
    1169             :   /* We work on _spec_ FpX's, all the l[xzq] below are lgpol's */
    1170             : 
    1171             :   /* initialize */
    1172         689 :   gel(x,0) = Fp_inv(gel(q,0), p);
    1173         689 :   if (lQ>1) gel(q,1) = Fp_red(gel(q,1), p);
    1174         689 :   if (lQ>1 && signe(gel(q,1)))
    1175         743 :   {
    1176         638 :     GEN u = gel(q, 1);
    1177         638 :     if (!equali1(gel(x,0))) u = Fp_mul(u, Fp_sqr(gel(x,0), p), p);
    1178         638 :     gel(x,1) = Fp_neg(u, p); lx = 2;
    1179             :   }
    1180             :   else
    1181          51 :     lx = 1;
    1182         794 :   nold = 1;
    1183        6361 :   for (; mask > 1; )
    1184             :   { /* set x -= x(x*q - 1) + O(t^(nnew + 1)), knowing x*q = 1 + O(t^(nold+1)) */
    1185        4877 :     long i, lnew, nnew = nold << 1;
    1186             : 
    1187        4877 :     if (mask & 1) nnew--;
    1188        4877 :     mask >>= 1;
    1189             : 
    1190        4877 :     lnew = nnew + 1;
    1191        4877 :     lq = ZX_lgrenormalizespec(q, minss(lQ,lnew));
    1192        4881 :     z = FpX_mulspec(x, q, p, lx, lq); /* FIXME: high product */
    1193        4881 :     lz = lgpol(z); if (lz > lnew) lz = lnew;
    1194        4885 :     z += 2;
    1195             :     /* subtract 1 [=>first nold words are 0]: renormalize so that z(0) != 0 */
    1196        4885 :     for (i = nold; i < lz; i++) if (signe(gel(z,i))) break;
    1197        4885 :     nold = nnew;
    1198        4885 :     if (i >= lz) continue; /* z-1 = 0(t^(nnew + 1)) */
    1199             : 
    1200             :     /* z + i represents (x*q - 1) / t^i */
    1201        4694 :     lz = ZX_lgrenormalizespec (z+i, lz-i);
    1202        4692 :     z = FpX_mulspec(x, z+i, p, lx, lz); /* FIXME: low product */
    1203        4692 :     lz = lgpol(z); z += 2;
    1204        4692 :     if (lz > lnew-i) lz = ZX_lgrenormalizespec(z, lnew-i);
    1205             : 
    1206        4692 :     lx = lz+ i;
    1207        4692 :     y  = x + i; /* x -= z * t^i, in place */
    1208        4692 :     for (i = 0; i < lz; i++) gel(y,i) = Fp_neg(gel(z,i), p);
    1209             :   }
    1210         690 :   x -= 2; setlg(x, lx + 2); x[1] = T[1];
    1211         690 :   return gerepilecopy(av, x);
    1212             : }
    1213             : 
    1214             : /* 1/polrecip(T)+O(x^(deg(T)-1)) */
    1215             : GEN
    1216        2256 : FpX_invBarrett(GEN T, GEN p)
    1217             : {
    1218        2256 :   pari_sp ltop = avma;
    1219        2256 :   long l = lg(T);
    1220             :   GEN r;
    1221        2256 :   if (l<5) return pol_0(varn(T));
    1222        2205 :   if (l<=FpX_INVBARRETT_LIMIT)
    1223             :   {
    1224        1517 :     GEN c = gel(T,l-1), ci=gen_1;
    1225        1517 :     if (!equali1(c))
    1226             :     {
    1227           4 :       ci = Fp_inv(c, p);
    1228           4 :       T = FpX_Fp_mul(T, ci, p);
    1229           4 :       r = FpX_invBarrett_basecase(T, p);
    1230           4 :       r = FpX_Fp_mul(r, ci, p);
    1231             :     } else
    1232        1513 :       r = FpX_invBarrett_basecase(T, p);
    1233             :   }
    1234             :   else
    1235         688 :     r = FpX_invBarrett_Newton(T, p);
    1236        2207 :   return gerepileupto(ltop, r);
    1237             : }
    1238             : 
    1239             : GEN
    1240      438110 : FpX_get_red(GEN T, GEN p)
    1241             : {
    1242      438110 :   if (typ(T)==t_POL && lg(T)>FpX_BARRETT_LIMIT)
    1243        1889 :     retmkvec2(FpX_invBarrett(T,p),T);
    1244      436221 :   return T;
    1245             : }
    1246             : 
    1247             : /* Compute x mod T where 2 <= degpol(T) <= l+1 <= 2*(degpol(T)-1)
    1248             :  * and mg is the Barrett inverse of T. */
    1249             : static GEN
    1250       98302 : FpX_divrem_Barrettspec(GEN x, long l, GEN mg, GEN T, GEN p, GEN *pr)
    1251             : {
    1252             :   GEN q, r;
    1253       98302 :   long lt = degpol(T); /*We discard the leading term*/
    1254             :   long ld, lm, lT, lmg;
    1255       98302 :   ld = l-lt;
    1256       98302 :   lm = minss(ld, lgpol(mg));
    1257       98302 :   lT  = ZX_lgrenormalizespec(T+2,lt);
    1258       98302 :   lmg = ZX_lgrenormalizespec(mg+2,lm);
    1259       98302 :   q = FpX_recipspec(x+lt,ld,ld);              /* q = rec(x)     lq<=ld*/
    1260       98303 :   q = FpX_mulspec(q+2,mg+2,p,lgpol(q),lmg);    /* q = rec(x) * mg lq<=ld+lm*/
    1261       98301 :   q = FpX_recipspec(q+2,minss(ld,lgpol(q)),ld);/* q = rec (rec(x) * mg) lq<=ld*/
    1262       98302 :   if (!pr) return q;
    1263       98302 :   r = FpX_mulspec(q+2,T+2,p,lgpol(q),lT);      /* r = q*pol        lr<=ld+lt*/
    1264       98300 :   r = FpX_subspec(x,r+2,p,lt,minss(lt,lgpol(r)));/* r = x - r   lr<=lt */
    1265       98302 :   if (pr == ONLY_REM) return r;
    1266         628 :   *pr = r; return q;
    1267             : }
    1268             : 
    1269             : static GEN
    1270       97910 : FpX_divrem_Barrett(GEN x, GEN mg, GEN T, GEN p, GEN *pr)
    1271             : {
    1272       97910 :   GEN q = NULL, r = FpX_red(x, p);
    1273       97910 :   long l = lgpol(r), lt = degpol(T), lm = 2*lt-1, v = varn(T);
    1274             :   long i;
    1275       97910 :   if (l <= lt)
    1276             :   {
    1277           0 :     if (pr == ONLY_REM) return r;
    1278           0 :     if (pr == ONLY_DIVIDES) return signe(r)? NULL: pol_0(v);
    1279           0 :     if (pr) *pr = r;
    1280           0 :     return pol_0(v);
    1281             :   }
    1282       97910 :   if (lt <= 1)
    1283          51 :     return FpX_divrem_basecase(r,T,p,pr);
    1284       97859 :   if (pr != ONLY_REM && l>lm)
    1285             :   {
    1286         151 :     q = cgetg(l-lt+2, t_POL); q[1] = T[1];
    1287         151 :     for (i=0;i<l-lt;i++) gel(q+2,i) = gen_0;
    1288             :   }
    1289      196162 :   while (l>lm)
    1290             :   {
    1291         444 :     GEN zr, zq = FpX_divrem_Barrettspec(r+2+l-lm,lm,mg,T,p,&zr);
    1292         444 :     long lz = lgpol(zr);
    1293         444 :     if (pr != ONLY_REM)
    1294             :     {
    1295         270 :       long lq = lgpol(zq);
    1296         270 :       for(i=0; i<lq; i++) gel(q+2+l-lm,i) = gel(zq,2+i);
    1297             :     }
    1298         444 :     for(i=0; i<lz; i++) gel(r+2+l-lm,i) = gel(zr,2+i);
    1299         444 :     l = l-lm+lz;
    1300             :   }
    1301       97859 :   if (pr == ONLY_REM)
    1302             :   {
    1303       97674 :     if (l > lt)
    1304       97674 :       r = FpX_divrem_Barrettspec(r+2, l, mg, T, p, ONLY_REM);
    1305             :     else
    1306           0 :       r = FpX_renormalize(r, l+2);
    1307       97674 :     setvarn(r, v); return r;
    1308             :   }
    1309         185 :   if (l > lt)
    1310             :   {
    1311         184 :     GEN zq = FpX_divrem_Barrettspec(r+2,l,mg,T,p, pr? &r: NULL);
    1312         184 :     if (!q) q = zq;
    1313             :     else
    1314             :     {
    1315         150 :       long lq = lgpol(zq);
    1316         150 :       for(i=0; i<lq; i++) gel(q+2,i) = gel(zq,2+i);
    1317             :     }
    1318             :   }
    1319           1 :   else if (pr)
    1320           1 :     r = FpX_renormalize(r, l+2);
    1321         185 :   setvarn(q, v); q = FpX_renormalize(q, lg(q));
    1322         185 :   if (pr == ONLY_DIVIDES) return signe(r)? NULL: q;
    1323         185 :   if (pr) { setvarn(r, v); *pr = r; }
    1324         185 :   return q;
    1325             : }
    1326             : 
    1327             : GEN
    1328     4359047 : FpX_divrem(GEN x, GEN T, GEN p, GEN *pr)
    1329             : {
    1330     4359047 :   GEN B, y = get_FpX_red(T, &B);
    1331     4359054 :   long dy = degpol(y), dx = degpol(x), d = dx-dy;
    1332     4359050 :   if (pr==ONLY_REM) return FpX_rem(x, y, p);
    1333     4359050 :   if (!B && d+3 < FpX_DIVREM_BARRETT_LIMIT)
    1334     4357973 :     return FpX_divrem_basecase(x,y,p,pr);
    1335        1077 :   else if (lgefint(p)==3)
    1336             :   {
    1337         864 :     pari_sp av = avma;
    1338         864 :     ulong pp = to_Flxq(&x, &T, p);
    1339         864 :     GEN z = Flx_divrem(x, T, pp, pr);
    1340         864 :     if (!z) return NULL;
    1341         864 :     if (!pr || pr == ONLY_DIVIDES)
    1342           8 :       return Flx_to_ZX_inplace(gerepileuptoleaf(av, z));
    1343         856 :     z = Flx_to_ZX(z);
    1344         856 :     *pr = Flx_to_ZX(*pr);
    1345         856 :     gerepileall(av, 2, &z, pr);
    1346         856 :     return z;
    1347             :   } else
    1348             :   {
    1349         213 :     pari_sp av=avma;
    1350         213 :     GEN mg = B? B: FpX_invBarrett(y, p);
    1351         213 :     GEN q1 = FpX_divrem_Barrett(x,mg,y,p,pr);
    1352         213 :     if (!q1) return gc_NULL(av);
    1353         213 :     if (!pr || pr==ONLY_DIVIDES) return gerepilecopy(av, q1);
    1354         213 :     gerepileall(av,2,&q1,pr);
    1355         213 :     return q1;
    1356             :   }
    1357             : }
    1358             : 
    1359             : GEN
    1360    71841769 : FpX_rem(GEN x, GEN T, GEN p)
    1361             : {
    1362    71841769 :   GEN B, y = get_FpX_red(T, &B);
    1363    71841774 :   long dy = degpol(y), dx = degpol(x), d = dx-dy;
    1364    71841924 :   if (d < 0) return FpX_red(x,p);
    1365    61664112 :   if (!B && d+3 < FpX_REM_BARRETT_LIMIT)
    1366    61552935 :     return FpX_divrem_basecase(x,y,p,ONLY_REM);
    1367      111177 :   else if (lgefint(p)==3)
    1368             :   {
    1369       13479 :     pari_sp av = avma;
    1370       13479 :     ulong pp = to_Flxq(&x, &T, p);
    1371       13482 :     return Flx_to_ZX_inplace(gerepileuptoleaf(av, Flx_rem(x, T, pp)));
    1372             :   } else
    1373             :   {
    1374       97698 :     pari_sp av = avma;
    1375       97698 :     GEN mg = B? B: FpX_invBarrett(y, p);
    1376       97698 :     return gerepileupto(av, FpX_divrem_Barrett(x, mg, y, p, ONLY_REM));
    1377             :   }
    1378             : }
    1379             : 
    1380             : static GEN
    1381        2872 : FpV_producttree(GEN xa, GEN s, GEN p, long vs)
    1382             : {
    1383        2872 :   long n = lg(xa)-1;
    1384        2872 :   long m = n==1 ? 1: expu(n-1)+1;
    1385        2872 :   long i, j, k, ls = lg(s);
    1386        2872 :   GEN T = cgetg(m+1, t_VEC);
    1387        2872 :   GEN t = cgetg(ls, t_VEC);
    1388       19216 :   for (j=1, k=1; j<ls; k+=s[j++])
    1389       32688 :     gel(t, j) = s[j] == 1 ?
    1390       28005 :              deg1pol(gen_1, Fp_neg(gel(xa,k), p), vs):
    1391       46644 :              deg2pol_shallow(gen_1,
    1392       23322 :                Fp_neg(Fp_add(gel(xa,k), gel(xa,k+1), p), p),
    1393       23322 :                Fp_mul(gel(xa,k), gel(xa,k+1), p), vs);
    1394        2872 :   gel(T,1) = t;
    1395        8125 :   for (i=2; i<=m; i++)
    1396             :   {
    1397        5253 :     GEN u = gel(T, i-1);
    1398        5253 :     long n = lg(u)-1;
    1399        5253 :     GEN t = cgetg(((n+1)>>1)+1, t_VEC);
    1400       18725 :     for (j=1, k=1; k<n; j++, k+=2)
    1401       13472 :       gel(t, j) = FpX_mul(gel(u, k), gel(u, k+1), p);
    1402        5253 :     gel(T, i) = t;
    1403             :   }
    1404        2872 :   return T;
    1405             : }
    1406             : 
    1407             : static GEN
    1408        2872 : FpX_FpV_multieval_tree(GEN P, GEN xa, GEN T, GEN p)
    1409             : {
    1410        2872 :   pari_sp av = avma;
    1411             :   long i,j,k;
    1412        2872 :   long m = lg(T)-1;
    1413             :   GEN t;
    1414        2872 :   GEN Tp = cgetg(m+1, t_VEC);
    1415        2872 :   gel(Tp, m) = mkvec(P);
    1416        8125 :   for (i=m-1; i>=1; i--)
    1417             :   {
    1418        5253 :     GEN u = gel(T, i);
    1419        5253 :     GEN v = gel(Tp, i+1);
    1420        5253 :     long n = lg(u)-1;
    1421        5253 :     t = cgetg(n+1, t_VEC);
    1422       18725 :     for (j=1, k=1; k<n; j++, k+=2)
    1423             :     {
    1424       13472 :       gel(t, k)   = FpX_rem(gel(v, j), gel(u, k), p);
    1425       13472 :       gel(t, k+1) = FpX_rem(gel(v, j), gel(u, k+1), p);
    1426             :     }
    1427        5253 :     gel(Tp, i) = t;
    1428             :   }
    1429             :   {
    1430        2872 :     GEN R = cgetg(lg(xa), t_VEC);
    1431        2872 :     GEN u = gel(T, i+1);
    1432        2872 :     GEN v = gel(Tp, i+1);
    1433        2872 :     long n = lg(u)-1;
    1434       19216 :     for (j=1, k=1; j<=n; j++)
    1435             :     {
    1436       16344 :       long c, d = degpol(gel(u,j));
    1437       44349 :       for (c=1; c<=d; c++, k++)
    1438       28005 :         gel(R,k) = FpX_eval(gel(v, j), gel(xa,k), p);
    1439             :     }
    1440        2872 :     return gerepileupto(av, R);
    1441             :   }
    1442             : }
    1443             : 
    1444             : static GEN
    1445           1 : FpVV_polint_tree(GEN T, GEN R, GEN s, GEN xa, GEN ya, GEN p, long vs)
    1446             : {
    1447           1 :   pari_sp av = avma;
    1448           1 :   long m = lg(T)-1;
    1449           1 :   long i, j, k, ls = lg(s);
    1450           1 :   GEN Tp = cgetg(m+1, t_VEC);
    1451           1 :   GEN t = cgetg(ls, t_VEC);
    1452           3 :   for (j=1, k=1; j<ls; k+=s[j++])
    1453           2 :     if (s[j]==2)
    1454             :     {
    1455           2 :       GEN a = Fp_mul(gel(ya,k), gel(R,k), p);
    1456           2 :       GEN b = Fp_mul(gel(ya,k+1), gel(R,k+1), p);
    1457           6 :       gel(t, j) = deg1pol(Fp_add(a, b, p),
    1458           2 :               Fp_neg(Fp_add(Fp_mul(gel(xa,k), b, p ),
    1459           2 :               Fp_mul(gel(xa,k+1), a, p), p), p), vs);
    1460             :     }
    1461             :     else
    1462           0 :       gel(t, j) = scalarpol(Fp_mul(gel(ya,k), gel(R,k), p), vs);
    1463           1 :   gel(Tp, 1) = t;
    1464           2 :   for (i=2; i<=m; i++)
    1465             :   {
    1466           1 :     GEN u = gel(T, i-1);
    1467           1 :     GEN t = cgetg(lg(gel(T,i)), t_VEC);
    1468           1 :     GEN v = gel(Tp, i-1);
    1469           1 :     long n = lg(v)-1;
    1470           2 :     for (j=1, k=1; k<n; j++, k+=2)
    1471           3 :       gel(t, j) = FpX_add(ZX_mul(gel(u, k), gel(v, k+1)),
    1472           2 :                           ZX_mul(gel(u, k+1), gel(v, k)), p);
    1473           1 :     gel(Tp, i) = t;
    1474             :   }
    1475           1 :   return gerepilecopy(av, gmael(Tp,m,1));
    1476             : }
    1477             : 
    1478             : GEN
    1479           0 : FpX_FpV_multieval(GEN P, GEN xa, GEN p)
    1480             : {
    1481           0 :   pari_sp av = avma;
    1482           0 :   GEN s = producttree_scheme(lg(xa)-1);
    1483           0 :   GEN T = FpV_producttree(xa, s, p, varn(P));
    1484           0 :   return gerepileupto(av, FpX_FpV_multieval_tree(P, xa, T, p));
    1485             : }
    1486             : 
    1487             : GEN
    1488           8 : FpV_polint(GEN xa, GEN ya, GEN p, long vs)
    1489             : {
    1490           8 :   pari_sp av = avma;
    1491             :   GEN s, T, P, R;
    1492             :   long m;
    1493           8 :   if (lgefint(p) == 3)
    1494             :   {
    1495           7 :     ulong pp = p[2];
    1496           7 :     P = Flv_polint(ZV_to_Flv(xa, pp), ZV_to_Flv(ya, pp), pp, evalvarn(vs));
    1497           7 :     return gerepileupto(av, Flx_to_ZX(P));
    1498             :   }
    1499           1 :   s = producttree_scheme(lg(xa)-1);
    1500           1 :   T = FpV_producttree(xa, s, p, vs);
    1501           1 :   m = lg(T)-1;
    1502           1 :   P = FpX_deriv(gmael(T, m, 1), p);
    1503           1 :   R = FpV_inv(FpX_FpV_multieval_tree(P, xa, T, p), p);
    1504           1 :   return gerepileupto(av, FpVV_polint_tree(T, R, s, xa, ya, p, vs));
    1505             : }
    1506             : 
    1507             : GEN
    1508           0 : FpV_FpM_polint(GEN xa, GEN ya, GEN p, long vs)
    1509             : {
    1510           0 :   pari_sp av = avma;
    1511           0 :   GEN s = producttree_scheme(lg(xa)-1);
    1512           0 :   GEN T = FpV_producttree(xa, s, p, vs);
    1513           0 :   long i, m = lg(T)-1, l = lg(ya)-1;
    1514           0 :   GEN P = FpX_deriv(gmael(T, m, 1), p);
    1515           0 :   GEN R = FpV_inv(FpX_FpV_multieval_tree(P, xa, T, p), p);
    1516           0 :   GEN M = cgetg(l+1, t_VEC);
    1517           0 :   for (i=1; i<=l; i++)
    1518           0 :     gel(M,i) = FpVV_polint_tree(T, R, s, xa, gel(ya,i), p, vs);
    1519           0 :   return gerepileupto(av, M);
    1520             : }
    1521             : 
    1522             : GEN
    1523        2871 : FpV_invVandermonde(GEN L, GEN den, GEN p)
    1524             : {
    1525        2871 :   pari_sp av = avma;
    1526        2871 :   long i, n = lg(L);
    1527             :   GEN M, R;
    1528        2871 :   GEN s = producttree_scheme(n-1);
    1529        2871 :   GEN tree = FpV_producttree(L, s, p, 0);
    1530        2871 :   long m = lg(tree)-1;
    1531        2871 :   GEN T = gmael(tree, m, 1);
    1532        2871 :   R = FpV_inv(FpX_FpV_multieval_tree(FpX_deriv(T, p), L, tree, p), p);
    1533        2871 :   if (den) R = FpC_Fp_mul(R, den, p);
    1534        2871 :   M = cgetg(n, t_MAT);
    1535       30872 :   for (i = 1; i < n; i++)
    1536             :   {
    1537       28001 :     GEN P = FpX_Fp_mul(FpX_div_by_X_x(T, gel(L,i), p, NULL), gel(R,i), p);
    1538       28001 :     gel(M,i) = RgX_to_RgC(P, n-1);
    1539             :   }
    1540        2871 :   return gerepilecopy(av, M);
    1541             : }
    1542             : 
    1543             : /***********************************************************************/
    1544             : /**                                                                   **/
    1545             : /**                              FpXQ                                 **/
    1546             : /**                                                                   **/
    1547             : /***********************************************************************/
    1548             : 
    1549             : /* FpXQ are elements of Fp[X]/(T), represented by FpX*/
    1550             : 
    1551             : GEN
    1552     6796788 : FpXQ_red(GEN x, GEN T, GEN p)
    1553             : {
    1554     6796788 :   GEN z = FpX_red(x,p);
    1555     6797774 :   return FpX_rem(z, T,p);
    1556             : }
    1557             : 
    1558             : GEN
    1559    42768357 : FpXQ_mul(GEN x,GEN y,GEN T,GEN p)
    1560             : {
    1561    42768357 :   GEN z = FpX_mul(x,y,p);
    1562    42768357 :   return FpX_rem(z, T, p);
    1563             : }
    1564             : 
    1565             : GEN
    1566     4116576 : FpXQ_sqr(GEN x, GEN T, GEN p)
    1567             : {
    1568     4116576 :   GEN z = FpX_sqr(x,p);
    1569     4116575 :   return FpX_rem(z, T, p);
    1570             : }
    1571             : 
    1572             : /* Inverse of x in Z/pZ[X]/(pol) or NULL if inverse doesn't exist
    1573             :  * return lift(1 / (x mod (p,pol))) */
    1574             : GEN
    1575      324865 : FpXQ_invsafe(GEN x, GEN y, GEN p)
    1576             : {
    1577      324865 :   GEN V, z = FpX_extgcd(get_FpX_mod(y), x, p, NULL, &V);
    1578      324865 :   if (degpol(z)) return NULL;
    1579      324865 :   z = Fp_invsafe(gel(z,2), p);
    1580      324865 :   if (!z) return NULL;
    1581      324865 :   return FpX_Fp_mul(V, z, p);
    1582             : }
    1583             : 
    1584             : GEN
    1585      324844 : FpXQ_inv(GEN x,GEN T,GEN p)
    1586             : {
    1587      324844 :   pari_sp av = avma;
    1588      324844 :   GEN U = FpXQ_invsafe(x, T, p);
    1589      324844 :   if (!U) pari_err_INV("FpXQ_inv",x);
    1590      324844 :   return gerepileupto(av, U);
    1591             : }
    1592             : 
    1593             : GEN
    1594      230139 : FpXQ_div(GEN x,GEN y,GEN T,GEN p)
    1595             : {
    1596      230139 :   pari_sp av = avma;
    1597      230139 :   return gerepileupto(av, FpXQ_mul(x,FpXQ_inv(y,T,p),T,p));
    1598             : }
    1599             : 
    1600             : static GEN
    1601     1183105 : _FpXQ_add(void *data, GEN x, GEN y)
    1602             : {
    1603             :   (void) data;
    1604     1183105 :   return ZX_add(x, y);
    1605             : }
    1606             : static GEN
    1607       63056 : _FpXQ_sub(void *data, GEN x, GEN y)
    1608             : {
    1609             :   (void) data;
    1610       63056 :   return ZX_sub(x, y);
    1611             : }
    1612             : static GEN
    1613     1324121 : _FpXQ_cmul(void *data, GEN P, long a, GEN x)
    1614             : {
    1615             :   (void) data;
    1616     1324121 :   return ZX_Z_mul(x, gel(P,a+2));
    1617             : }
    1618             : static GEN
    1619     3646309 : _FpXQ_sqr(void *data, GEN x)
    1620             : {
    1621     3646309 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1622     3646309 :   return FpXQ_sqr(x, D->T, D->p);
    1623             : }
    1624             : static GEN
    1625     1075435 : _FpXQ_mul(void *data, GEN x, GEN y)
    1626             : {
    1627     1075435 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1628     1075435 :   return FpXQ_mul(x,y, D->T, D->p);
    1629             : }
    1630             : static GEN
    1631        4137 : _FpXQ_zero(void *data)
    1632             : {
    1633        4137 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1634        4137 :   return pol_0(get_FpX_var(D->T));
    1635             : }
    1636             : static GEN
    1637      342994 : _FpXQ_one(void *data)
    1638             : {
    1639      342994 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1640      342994 :   return pol_1(get_FpX_var(D->T));
    1641             : }
    1642             : static GEN
    1643      395598 : _FpXQ_red(void *data, GEN x)
    1644             : {
    1645      395598 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1646      395598 :   return FpX_red(x,D->p);
    1647             : }
    1648             : 
    1649             : static struct bb_algebra FpXQ_algebra = { _FpXQ_red, _FpXQ_add, _FpXQ_sub,
    1650             :        _FpXQ_mul, _FpXQ_sqr, _FpXQ_one, _FpXQ_zero };
    1651             : 
    1652             : const struct bb_algebra *
    1653       15043 : get_FpXQ_algebra(void **E, GEN T, GEN p)
    1654             : {
    1655       15043 :   GEN z = new_chunk(sizeof(struct _FpXQ));
    1656       15043 :   struct _FpXQ *e = (struct _FpXQ *) z;
    1657       15043 :   e->T = FpX_get_red(T, p);
    1658       15043 :   e->p  = p; *E = (void*)e;
    1659       15043 :   return &FpXQ_algebra;
    1660             : }
    1661             : 
    1662             : static struct bb_algebra FpX_algebra = { _FpXQ_red, _FpXQ_add, _FpXQ_sub,
    1663             :        _FpX_mul, _FpX_sqr, _FpXQ_one, _FpXQ_zero };
    1664             : 
    1665             : const struct bb_algebra *
    1666           0 : get_FpX_algebra(void **E, GEN p, long v)
    1667             : {
    1668           0 :   GEN z = new_chunk(sizeof(struct _FpXQ));
    1669           0 :   struct _FpXQ *e = (struct _FpXQ *) z;
    1670           0 :   e->T = pol_x(v);
    1671           0 :   e->p  = p; *E = (void*)e;
    1672           0 :   return &FpX_algebra;
    1673             : }
    1674             : 
    1675             : /* x,pol in Z[X], p in Z, n in Z, compute lift(x^n mod (p, pol)) */
    1676             : GEN
    1677      558856 : FpXQ_pow(GEN x, GEN n, GEN T, GEN p)
    1678             : {
    1679             :   struct _FpXQ D;
    1680             :   pari_sp av;
    1681      558856 :   long s = signe(n);
    1682             :   GEN y;
    1683      558856 :   if (!s) return pol_1(varn(x));
    1684      556936 :   if (is_pm1(n)) /* +/- 1 */
    1685       10369 :     return (s < 0)? FpXQ_inv(x,T,p): FpXQ_red(x,T,p);
    1686      546566 :   av = avma;
    1687      546566 :   if (!is_bigint(p))
    1688             :   {
    1689      393178 :     ulong pp = to_Flxq(&x, &T, p);
    1690      393170 :     y = Flxq_pow(x, n, T, pp);
    1691      393166 :     return Flx_to_ZX_inplace(gerepileuptoleaf(av, y));
    1692             :   }
    1693      153389 :   if (s < 0) x = FpXQ_inv(x,T,p);
    1694      153389 :   D.p = p; D.T = FpX_get_red(T,p);
    1695      153389 :   y = gen_pow_i(x, n, (void*)&D, &_FpXQ_sqr, &_FpXQ_mul);
    1696      153389 :   return gerepilecopy(av, y);
    1697             : }
    1698             : 
    1699             : GEN /*Assume n is very small*/
    1700       72839 : FpXQ_powu(GEN x, ulong n, GEN T, GEN p)
    1701             : {
    1702             :   struct _FpXQ D;
    1703             :   pari_sp av;
    1704             :   GEN y;
    1705       72839 :   if (!n) return pol_1(varn(x));
    1706       72839 :   if (n==1) return FpXQ_red(x,T,p);
    1707       38430 :   av = avma;
    1708       38430 :   if (!is_bigint(p))
    1709             :   {
    1710       36734 :     ulong pp = to_Flxq(&x, &T, p);
    1711       36734 :     y = Flxq_powu(x, n, T, pp);
    1712       36734 :     return Flx_to_ZX_inplace(gerepileuptoleaf(av, y));
    1713             :   }
    1714        1696 :   D.T = FpX_get_red(T, p); D.p = p;
    1715        1696 :   y = gen_powu_i(x, n, (void*)&D, &_FpXQ_sqr, &_FpXQ_mul);
    1716        1696 :   return gerepilecopy(av, y);
    1717             : }
    1718             : 
    1719             : /* generates the list of powers of x of degree 0,1,2,...,l*/
    1720             : GEN
    1721      189479 : FpXQ_powers(GEN x, long l, GEN T, GEN p)
    1722             : {
    1723             :   struct _FpXQ D;
    1724             :   int use_sqr;
    1725      189479 :   if (l>2 && lgefint(p) == 3) {
    1726      154351 :     pari_sp av = avma;
    1727      154351 :     ulong pp = to_Flxq(&x, &T, p);
    1728      154351 :     GEN z = FlxV_to_ZXV(Flxq_powers(x, l, T, pp));
    1729      154351 :     return gerepileupto(av, z);
    1730             :   }
    1731       35128 :   use_sqr = 2*degpol(x)>=get_FpX_degree(T);
    1732       35128 :   D.T = FpX_get_red(T,p); D.p = p;
    1733       35128 :   return gen_powers(x, l, use_sqr, (void*)&D, &_FpXQ_sqr, &_FpXQ_mul,&_FpXQ_one);
    1734             : }
    1735             : 
    1736             : GEN
    1737        1737 : FpXQ_matrix_pow(GEN y, long n, long m, GEN P, GEN l)
    1738             : {
    1739        1737 :   return RgXV_to_RgM(FpXQ_powers(y,m-1,P,l),n);
    1740             : }
    1741             : 
    1742             : GEN
    1743      211314 : FpX_Frobenius(GEN T, GEN p)
    1744             : {
    1745      211314 :   return FpXQ_pow(pol_x(get_FpX_var(T)), p, T, p);
    1746             : }
    1747             : 
    1748             : GEN
    1749         876 : FpX_matFrobenius(GEN T, GEN p)
    1750             : {
    1751         876 :   long n = get_FpX_degree(T);
    1752         876 :   return FpXQ_matrix_pow(FpX_Frobenius(T, p), n, n, T, p);
    1753             : }
    1754             : 
    1755             : GEN
    1756      137242 : FpX_FpXQV_eval(GEN Q, GEN x, GEN T, GEN p)
    1757             : {
    1758             :   struct _FpXQ D;
    1759      137242 :   D.T = FpX_get_red(T,p); D.p = p;
    1760      137242 :   return gen_bkeval_powers(Q,degpol(Q),x,(void*)&D,&FpXQ_algebra,_FpXQ_cmul);
    1761             : }
    1762             : 
    1763             : GEN
    1764      173827 : FpX_FpXQ_eval(GEN Q, GEN x, GEN T, GEN p)
    1765             : {
    1766             :   struct _FpXQ D;
    1767             :   int use_sqr;
    1768      173827 :   if (lgefint(p) == 3)
    1769             :   {
    1770      169337 :     pari_sp av = avma;
    1771      169337 :     ulong pp = to_Flxq(&x, &T, p);
    1772      169337 :     GEN z = Flx_Flxq_eval(ZX_to_Flx(Q, pp), x, T, pp);
    1773      169337 :     return Flx_to_ZX_inplace(gerepileuptoleaf(av, z));
    1774             :   }
    1775        4490 :   use_sqr = 2*degpol(x) >= get_FpX_degree(T);
    1776        4490 :   D.T = FpX_get_red(T,p); D.p = p;
    1777        4490 :   return gen_bkeval(Q,degpol(Q),x,use_sqr,(void*)&D,&FpXQ_algebra,_FpXQ_cmul);
    1778             : }
    1779             : 
    1780             : GEN
    1781         854 : FpXC_FpXQV_eval(GEN x, GEN v, GEN T, GEN p)
    1782         854 : { pari_APPLY_type(t_COL, FpX_FpXQV_eval(gel(x,i), v, T, p))
    1783             : }
    1784             : 
    1785             : GEN
    1786         308 : FpXM_FpXQV_eval(GEN x, GEN v, GEN T, GEN p)
    1787         308 : { pari_APPLY_same(FpXC_FpXQV_eval(gel(x,i), v, T, p))
    1788             : }
    1789             : 
    1790             : GEN
    1791           0 : FpXC_FpXQ_eval(GEN x, GEN F, GEN T, GEN p)
    1792             : {
    1793           0 :   long d = brent_kung_optpow(degpol(T)-1,lg(x)-1,1);
    1794           0 :   GEN Fp = FpXQ_powers(F, d, T, p);
    1795           0 :   return FpXC_FpXQV_eval(x, Fp, T, p);
    1796             : }
    1797             : 
    1798             : GEN
    1799         903 : FpXQ_autpowers(GEN aut, long f, GEN T, GEN p)
    1800             : {
    1801         903 :   pari_sp av = avma;
    1802         903 :   long n = get_FpX_degree(T);
    1803         903 :   long i, nautpow = brent_kung_optpow(n-1,f-2,1);
    1804         903 :   long v = get_FpX_var(T);
    1805             :   GEN autpow, V;
    1806         903 :   T = FpX_get_red(T, p);
    1807         903 :   autpow = FpXQ_powers(aut, nautpow,T,p);
    1808         903 :   V = cgetg(f + 2, t_VEC);
    1809         903 :   gel(V,1) = pol_x(v); if (f==0) return gerepileupto(av, V);
    1810         903 :   gel(V,2) = gcopy(aut);
    1811        4228 :   for (i = 3; i <= f+1; i++)
    1812        3325 :     gel(V,i) = FpX_FpXQV_eval(gel(V,i-1),autpow,T,p);
    1813         903 :   return gerepileupto(av, V);
    1814             : }
    1815             : 
    1816             : static GEN
    1817         647 : FpXQ_autpow_sqr(void *E, GEN x)
    1818             : {
    1819         647 :   struct _FpXQ *D = (struct _FpXQ*)E;
    1820         647 :   return FpX_FpXQ_eval(x, x, D->T, D->p);
    1821             : }
    1822             : 
    1823             : static GEN
    1824          21 : FpXQ_autpow_mul(void *E, GEN x, GEN y)
    1825             : {
    1826          21 :   struct _FpXQ *D = (struct _FpXQ*)E;
    1827          21 :   return FpX_FpXQ_eval(x, y, D->T, D->p);
    1828             : }
    1829             : 
    1830             : GEN
    1831         626 : FpXQ_autpow(GEN x, ulong n, GEN T, GEN p)
    1832             : {
    1833         626 :   pari_sp av = avma;
    1834             :   struct _FpXQ D;
    1835         626 :   if (n==0) return FpX_rem(pol_x(varn(x)), T, p);
    1836         626 :   if (n==1) return FpX_rem(x, T, p);
    1837         626 :   D.T = FpX_get_red(T, p); D.p = p;
    1838         626 :   x = gen_powu_i(x,n,(void*)&D,FpXQ_autpow_sqr,FpXQ_autpow_mul);
    1839         626 :   return gerepilecopy(av, x);
    1840             : }
    1841             : 
    1842             : static GEN
    1843         273 : FpXQ_auttrace_mul(void *E, GEN x, GEN y)
    1844             : {
    1845         273 :   struct _FpXQ *D = (struct _FpXQ*)E;
    1846         273 :   GEN T = D->T, p = D->p;
    1847         273 :   GEN phi1 = gel(x,1), a1 = gel(x,2);
    1848         273 :   GEN phi2 = gel(y,1), a2 = gel(y,2);
    1849         273 :   ulong d = brent_kung_optpow(maxss(degpol(phi2),degpol(a2)),2,1);
    1850         273 :   GEN V1 = FpXQ_powers(phi1, d, T, p);
    1851         273 :   GEN phi3 = FpX_FpXQV_eval(phi2, V1, T, p);
    1852         273 :   GEN aphi = FpX_FpXQV_eval(a2, V1, T, p);
    1853         273 :   GEN a3 = FpX_add(a1, aphi, p);
    1854         273 :   return mkvec2(phi3, a3);
    1855             : }
    1856             : 
    1857             : static GEN
    1858         245 : FpXQ_auttrace_sqr(void *E, GEN x)
    1859         245 : { return FpXQ_auttrace_mul(E, x, x); }
    1860             : 
    1861             : GEN
    1862         224 : FpXQ_auttrace(GEN x, ulong n, GEN T, GEN p)
    1863             : {
    1864         224 :   pari_sp av = avma;
    1865             :   struct _FpXQ D;
    1866         224 :   D.T = FpX_get_red(T, p); D.p = p;
    1867         224 :   x = gen_powu_i(x,n,(void*)&D,FpXQ_auttrace_sqr,FpXQ_auttrace_mul);
    1868         224 :   return gerepilecopy(av, x);
    1869             : }
    1870             : 
    1871             : static GEN
    1872        2545 : FpXQ_autsum_mul(void *E, GEN x, GEN y)
    1873             : {
    1874        2545 :   struct _FpXQ *D = (struct _FpXQ*)E;
    1875        2545 :   GEN T = D->T, p = D->p;
    1876        2545 :   GEN phi1 = gel(x,1), a1 = gel(x,2);
    1877        2545 :   GEN phi2 = gel(y,1), a2 = gel(y,2);
    1878        2545 :   ulong d = brent_kung_optpow(maxss(degpol(phi2),degpol(a2)),2,1);
    1879        2545 :   GEN V1 = FpXQ_powers(phi1, d, T, p);
    1880        2545 :   GEN phi3 = FpX_FpXQV_eval(phi2, V1, T, p);
    1881        2545 :   GEN aphi = FpX_FpXQV_eval(a2, V1, T, p);
    1882        2545 :   GEN a3 = FpXQ_mul(a1, aphi, T, p);
    1883        2545 :   return mkvec2(phi3, a3);
    1884             : }
    1885             : static GEN
    1886        1355 : FpXQ_autsum_sqr(void *E, GEN x)
    1887        1355 : { return FpXQ_autsum_mul(E, x, x); }
    1888             : 
    1889             : GEN
    1890        1243 : FpXQ_autsum(GEN x, ulong n, GEN T, GEN p)
    1891             : {
    1892        1243 :   pari_sp av = avma;
    1893             :   struct _FpXQ D;
    1894        1243 :   D.T = FpX_get_red(T, p); D.p = p;
    1895        1243 :   x = gen_powu_i(x,n,(void*)&D,FpXQ_autsum_sqr,FpXQ_autsum_mul);
    1896        1243 :   return gerepilecopy(av, x);
    1897             : }
    1898             : 
    1899             : static GEN
    1900         308 : FpXQM_autsum_mul(void *E, GEN x, GEN y)
    1901             : {
    1902         308 :   struct _FpXQ *D = (struct _FpXQ*)E;
    1903         308 :   GEN T = D->T, p = D->p;
    1904         308 :   GEN phi1 = gel(x,1), a1 = gel(x,2);
    1905         308 :   GEN phi2 = gel(y,1), a2 = gel(y,2);
    1906         308 :   long g = lg(a2)-1, dT = get_FpX_degree(T);
    1907         308 :   ulong d = brent_kung_optpow(dT-1, g*g+1, 1);
    1908         308 :   GEN V1 = FpXQ_powers(phi1, d, T, p);
    1909         308 :   GEN phi3 = FpX_FpXQV_eval(phi2, V1, T, p);
    1910         308 :   GEN aphi = FpXM_FpXQV_eval(a2, V1, T, p);
    1911         308 :   GEN a3 = FqM_mul(a1, aphi, T, p);
    1912         308 :   return mkvec2(phi3, a3);
    1913             : }
    1914             : static GEN
    1915         210 : FpXQM_autsum_sqr(void *E, GEN x)
    1916         210 : { return FpXQM_autsum_mul(E, x, x); }
    1917             : 
    1918             : GEN
    1919         140 : FpXQM_autsum(GEN x, ulong n, GEN T, GEN p)
    1920             : {
    1921         140 :   pari_sp av = avma;
    1922             :   struct _FpXQ D;
    1923         140 :   D.T = FpX_get_red(T, p); D.p = p;
    1924         140 :   x = gen_powu_i(x, n, (void*)&D, FpXQM_autsum_sqr, FpXQM_autsum_mul);
    1925         140 :   return gerepilecopy(av, x);
    1926             : }
    1927             : 
    1928             : static long
    1929        5838 : bounded_order(GEN p, GEN b, long k)
    1930             : {
    1931             :   long i;
    1932        5838 :   GEN a=modii(p,b);
    1933       11655 :   for(i=1;i<k;i++)
    1934             :   {
    1935        7574 :     if (equali1(a))
    1936        1757 :       return i;
    1937        5817 :     a = Fp_mul(a,p,b);
    1938             :   }
    1939        4081 :   return 0;
    1940             : }
    1941             : 
    1942             : /*
    1943             :   n = (p^d-a)\b
    1944             :   b = bb*p^vb
    1945             :   p^k = 1 [bb]
    1946             :   d = m*k+r+vb
    1947             :   u = (p^k-1)/bb;
    1948             :   v = (p^(r+vb)-a)/b;
    1949             :   w = (p^(m*k)-1)/(p^k-1)
    1950             :   n = p^r*w*u+v
    1951             :   w*u = p^vb*(p^(m*k)-1)/b
    1952             :   n = p^(r+vb)*(p^(m*k)-1)/b+(p^(r+vb)-a)/b
    1953             : */
    1954             : 
    1955             : static GEN
    1956      138544 : FpXQ_pow_Frobenius(GEN x, GEN n, GEN aut, GEN T, GEN p)
    1957             : {
    1958      138544 :   pari_sp av=avma;
    1959      138544 :   long d = get_FpX_degree(T);
    1960      138544 :   GEN an = absi_shallow(n), z, q;
    1961      138544 :   if (cmpii(an,p)<0 || cmpis(an,d)<=0) return FpXQ_pow(x, n, T, p);
    1962        5859 :   q = powiu(p, d);
    1963        5859 :   if (dvdii(q, n))
    1964             :   {
    1965           0 :     long vn = logint(an,p);
    1966           0 :     GEN autvn = vn==1 ? aut: FpXQ_autpow(aut,vn,T,p);
    1967           0 :     z = FpX_FpXQ_eval(x,autvn,T,p);
    1968             :   } else
    1969             :   {
    1970        5859 :     GEN b = diviiround(q, an), a = subii(q, mulii(an,b));
    1971             :     GEN bb, u, v, autk;
    1972        5859 :     long vb = Z_pvalrem(b,p,&bb);
    1973        5859 :     long m, r, k = is_pm1(bb) ? 1 : bounded_order(p,bb,d);
    1974        5859 :     if (!k || d-vb<k) return FpXQ_pow(x,n, T, p);
    1975        1778 :     m = (d-vb)/k; r = (d-vb)%k;
    1976        1778 :     u = diviiexact(subiu(powiu(p,k),1),bb);
    1977        1778 :     v = diviiexact(subii(powiu(p,r+vb),a),b);
    1978        1778 :     autk = k==1 ? aut: FpXQ_autpow(aut,k,T,p);
    1979        1778 :     if (r)
    1980             :     {
    1981         549 :       GEN autr = r==1 ? aut: FpXQ_autpow(aut,r,T,p);
    1982         549 :       z = FpX_FpXQ_eval(x,autr,T,p);
    1983        1229 :     } else z = x;
    1984        1778 :     if (m > 1) z = gel(FpXQ_autsum(mkvec2(autk, z), m, T, p), 2);
    1985        1778 :     if (!is_pm1(u)) z = FpXQ_pow(z, u, T, p);
    1986        1778 :     if (signe(v)) z = FpXQ_mul(z, FpXQ_pow(x, v, T, p), T, p);
    1987             :   }
    1988        1778 :   return gerepileupto(av,signe(n)>0 ? z : FpXQ_inv(z,T,p));
    1989             : }
    1990             : 
    1991             : /* assume T irreducible mod p */
    1992             : int
    1993        3712 : FpXQ_issquare(GEN x, GEN T, GEN p)
    1994             : {
    1995             :   pari_sp av;
    1996        3712 :   if (lg(x) == 2 || absequalui(2, p)) return 1;
    1997        3698 :   if (lg(x) == 3) return Fq_issquare(gel(x,2), T, p);
    1998        3596 :   av = avma; /* Ng = g^((q-1)/(p-1)) */
    1999        3596 :   return gc_bool(av, kronecker(FpXQ_norm(x,T,p), p) == 1);
    2000             : }
    2001             : int
    2002       92208 : Fp_issquare(GEN x, GEN p)
    2003             : {
    2004       92208 :   if (absequalui(2, p)) return 1;
    2005       92208 :   return kronecker(x, p) == 1;
    2006             : }
    2007             : /* assume T irreducible mod p */
    2008             : int
    2009       92166 : Fq_issquare(GEN x, GEN T, GEN p)
    2010             : {
    2011       92166 :   if (typ(x) != t_INT) return FpXQ_issquare(x, T, p);
    2012       92068 :   return (T && ! odd(get_FpX_degree(T))) || Fp_issquare(x, p);
    2013             : }
    2014             : 
    2015             : long
    2016          63 : Fq_ispower(GEN x, GEN K, GEN T, GEN p)
    2017             : {
    2018          63 :   pari_sp av = avma;
    2019             :   long d;
    2020             :   GEN Q;
    2021          63 :   if (equaliu(K,2)) return Fq_issquare(x, T, p);
    2022           0 :   if (!T) return Fp_ispower(x, K, p);
    2023           0 :   d = get_FpX_degree(T);
    2024           0 :   if (typ(x) == t_INT && !umodui(d, K)) return 1;
    2025           0 :   Q = subiu(powiu(p,d), 1);
    2026           0 :   Q = diviiexact(Q, gcdii(Q, K));
    2027           0 :   d = gequal1(Fq_pow(x, Q, T,p));
    2028           0 :   return gc_long(av, d);
    2029             : }
    2030             : 
    2031             : /* discrete log in FpXQ for a in Fp^*, g in FpXQ^* of order ord */
    2032             : GEN
    2033       18984 : Fp_FpXQ_log(GEN a, GEN g, GEN o, GEN T, GEN p)
    2034             : {
    2035       18984 :   pari_sp av = avma;
    2036             :   GEN q,n_q,ord,ordp, op;
    2037             : 
    2038       18984 :   if (equali1(a)) return gen_0;
    2039             :   /* p > 2 */
    2040             : 
    2041        6227 :   ordp = subiu(p, 1); /* even */
    2042        6227 :   ord  = get_arith_Z(o);
    2043        6199 :   if (!ord) ord = T? subiu(powiu(p, get_FpX_degree(T)), 1): ordp;
    2044        6199 :   if (equalii(a, ordp)) /* -1 */
    2045        4313 :     return gerepileuptoint(av, shifti(ord,-1));
    2046        1886 :   ordp = gcdii(ordp,ord);
    2047        1886 :   op = typ(o)==t_MAT ? famat_Z_gcd(o,ordp) : ordp;
    2048             : 
    2049        1886 :   q = NULL;
    2050        1886 :   if (T)
    2051             :   { /* we want < g > = Fp^* */
    2052        1886 :     if (!equalii(ord,ordp)) {
    2053        1876 :       q = diviiexact(ord,ordp);
    2054        1876 :       g = FpXQ_pow(g,q,T,p);
    2055             :     }
    2056        1886 :     g = constant_coeff(g);
    2057             :   }
    2058        1886 :   n_q = Fp_log(a,g,op,p);
    2059        1886 :   if (lg(n_q)==1) return gerepileuptoleaf(av, n_q);
    2060        1886 :   if (q) n_q = mulii(q, n_q);
    2061        1886 :   return gerepileuptoint(av, n_q);
    2062             : }
    2063             : 
    2064             : static GEN
    2065      137735 : _FpXQ_pow(void *data, GEN x, GEN n)
    2066             : {
    2067      137735 :   struct _FpXQ *D = (struct _FpXQ*)data;
    2068      137735 :   return FpXQ_pow_Frobenius(x,n, D->aut, D->T, D->p);
    2069             : }
    2070             : 
    2071             : static GEN
    2072        3619 : _FpXQ_rand(void *data)
    2073             : {
    2074        3619 :   pari_sp av=avma;
    2075        3619 :   struct _FpXQ *D = (struct _FpXQ*)data;
    2076             :   GEN z;
    2077             :   do
    2078             :   {
    2079        3619 :     set_avma(av);
    2080        3619 :     z=random_FpX(get_FpX_degree(D->T),get_FpX_var(D->T),D->p);
    2081        3619 :   } while (!signe(z));
    2082        3619 :   return z;
    2083             : }
    2084             : 
    2085             : static GEN
    2086        1438 : _FpXQ_easylog(void *E, GEN a, GEN g, GEN ord)
    2087             : {
    2088        1438 :   struct _FpXQ *s=(struct _FpXQ*) E;
    2089        1438 :   if (degpol(a)) return NULL;
    2090        1414 :   return Fp_FpXQ_log(constant_coeff(a),g,ord,s->T,s->p);
    2091             : }
    2092             : 
    2093             : static const struct bb_group FpXQ_star={_FpXQ_mul,_FpXQ_pow,_FpXQ_rand,hash_GEN,ZX_equal,ZX_equal1,_FpXQ_easylog};
    2094             : 
    2095             : const struct bb_group *
    2096        2010 : get_FpXQ_star(void **E, GEN T, GEN p)
    2097             : {
    2098        2010 :   struct _FpXQ *e = (struct _FpXQ *) stack_malloc(sizeof(struct _FpXQ));
    2099        2010 :   e->T = T; e->p  = p; e->aut =  FpX_Frobenius(T, p);
    2100        2010 :   *E = (void*)e; return &FpXQ_star;
    2101             : }
    2102             : 
    2103             : GEN
    2104          29 : FpXQ_order(GEN a, GEN ord, GEN T, GEN p)
    2105             : {
    2106          29 :   if (lgefint(p)==3)
    2107             :   {
    2108           0 :     pari_sp av=avma;
    2109           0 :     ulong pp = to_Flxq(&a, &T, p);
    2110           0 :     GEN z = Flxq_order(a, ord, T, pp);
    2111           0 :     return gerepileuptoint(av,z);
    2112             :   }
    2113             :   else
    2114             :   {
    2115             :     void *E;
    2116          29 :     const struct bb_group *S = get_FpXQ_star(&E,T,p);
    2117          29 :     return gen_order(a,ord,E,S);
    2118             :   }
    2119             : }
    2120             : 
    2121             : GEN
    2122       89428 : FpXQ_log(GEN a, GEN g, GEN ord, GEN T, GEN p)
    2123             : {
    2124       89428 :   pari_sp av=avma;
    2125       89428 :   if (lgefint(p)==3)
    2126             :   {
    2127       89384 :     if (uel(p,2) == 2)
    2128             :     {
    2129       51198 :       GEN z = F2xq_log(ZX_to_F2x(a), ZX_to_F2x(g), ord,
    2130             :                                      ZX_to_F2x(get_FpX_mod(T)));
    2131       51198 :       return gerepileuptoleaf(av, z);
    2132             :     }
    2133             :     else
    2134             :     {
    2135       38186 :       ulong pp = to_Flxq(&a, &T, p);
    2136       38186 :       GEN z = Flxq_log(a, ZX_to_Flx(g, pp), ord, T, pp);
    2137       38186 :       return gerepileuptoleaf(av, z);
    2138             :     }
    2139             :   }
    2140             :   else
    2141             :   {
    2142             :     void *E;
    2143          44 :     const struct bb_group *S = get_FpXQ_star(&E,T,p);
    2144          44 :     GEN z = gen_PH_log(a,g,ord,E,S);
    2145          16 :     return gerepileuptoleaf(av, z);
    2146             :   }
    2147             : }
    2148             : 
    2149             : GEN
    2150      482181 : Fq_log(GEN a, GEN g, GEN ord, GEN T, GEN p)
    2151             : {
    2152      482181 :   if (!T) return Fp_log(a,g,ord,p);
    2153      106954 :   if (typ(g) == t_INT)
    2154             :   {
    2155           0 :     if (typ(a) == t_POL)
    2156             :     {
    2157           0 :       if (degpol(a)) return cgetg(1,t_VEC);
    2158           0 :       a = gel(a,2);
    2159             :     }
    2160           0 :     return Fp_log(a,g,ord,p);
    2161             :   }
    2162      106954 :   return typ(a) == t_INT? Fp_FpXQ_log(a,g,ord,T,p): FpXQ_log(a,g,ord,T,p);
    2163             : }
    2164             : 
    2165             : GEN
    2166       12528 : FpXQ_sqrtn(GEN a, GEN n, GEN T, GEN p, GEN *zeta)
    2167             : {
    2168       12528 :   pari_sp av = avma;
    2169             :   GEN z;
    2170       12528 :   if (!signe(a))
    2171             :   {
    2172        2401 :     long v=varn(a);
    2173        2401 :     if (signe(n) < 0) pari_err_INV("FpXQ_sqrtn",a);
    2174        2394 :     if (zeta) *zeta=pol_1(v);
    2175        2394 :     return pol_0(v);
    2176             :   }
    2177       10127 :   if (lgefint(p)==3)
    2178             :   {
    2179        8190 :     if (uel(p,2) == 2)
    2180             :     {
    2181        2275 :       z = F2xq_sqrtn(ZX_to_F2x(a), n, ZX_to_F2x(get_FpX_mod(T)), zeta);
    2182        2275 :       if (!z) return NULL;
    2183        2275 :       z = F2x_to_ZX(z);
    2184        2275 :       if (!zeta) return gerepileuptoleaf(av, z);
    2185           7 :       *zeta=F2x_to_ZX(*zeta);
    2186             :     } else
    2187             :     {
    2188        5915 :       ulong pp = to_Flxq(&a, &T, p);
    2189        5915 :       z = Flxq_sqrtn(a, n, T, pp, zeta);
    2190        5915 :       if (!z) return NULL;
    2191        3577 :       if (!zeta) return Flx_to_ZX_inplace(gerepileuptoleaf(av, z));
    2192         175 :       z = Flx_to_ZX(z);
    2193         175 :       *zeta=Flx_to_ZX(*zeta);
    2194             :     }
    2195             :   }
    2196             :   else
    2197             :   {
    2198             :     void *E;
    2199        1937 :     const struct bb_group *S = get_FpXQ_star(&E,T,p);
    2200        1937 :     GEN o = subiu(powiu(p,get_FpX_degree(T)),1);
    2201        1937 :     z = gen_Shanks_sqrtn(a,n,o,zeta,E,S);
    2202        3817 :     if (!z) return NULL;
    2203        1895 :     if (!zeta) return gerepileupto(av, z);
    2204             :   }
    2205         239 :   gerepileall(av, 2, &z,zeta);
    2206         239 :   return z;
    2207             : }
    2208             : 
    2209             : GEN
    2210       11915 : FpXQ_sqrt(GEN a, GEN T, GEN p)
    2211             : {
    2212       11915 :   return FpXQ_sqrtn(a, gen_2, T, p, NULL);
    2213             : }
    2214             : 
    2215             : GEN
    2216        3597 : FpXQ_norm(GEN x, GEN TB, GEN p)
    2217             : {
    2218        3597 :   pari_sp av = avma;
    2219        3597 :   GEN T = get_FpX_mod(TB);
    2220        3597 :   GEN y = FpX_resultant(T, x, p);
    2221        3597 :   GEN L = leading_coeff(T);
    2222        3597 :   if (gequal1(L) || signe(x)==0) return y;
    2223           0 :   return gerepileupto(av, Fp_div(y, Fp_pows(L, degpol(x), p), p));
    2224             : }
    2225             : 
    2226             : GEN
    2227       20873 : FpXQ_trace(GEN x, GEN TB, GEN p)
    2228             : {
    2229       20873 :   pari_sp av = avma;
    2230       20873 :   GEN T = get_FpX_mod(TB);
    2231       20873 :   GEN dT = FpX_deriv(T,p);
    2232       20873 :   long n = degpol(dT);
    2233       20873 :   GEN z = FpXQ_mul(x, dT, TB, p);
    2234       20873 :   if (degpol(z)<n) { set_avma(av); return gen_0; }
    2235       19697 :   return gerepileuptoint(av, Fp_div(gel(z,2+n), gel(T,3+n),p));
    2236             : }
    2237             : 
    2238             : GEN
    2239           1 : FpXQ_charpoly(GEN x, GEN T, GEN p)
    2240             : {
    2241           1 :   pari_sp ltop=avma;
    2242           1 :   long vT, v = fetch_var();
    2243             :   GEN R;
    2244           1 :   T = leafcopy(get_FpX_mod(T));
    2245           1 :   vT = varn(T); setvarn(T, v);
    2246           1 :   x = leafcopy(x); setvarn(x, v);
    2247           1 :   R = FpX_FpXY_resultant(T, deg1pol_shallow(gen_1,FpX_neg(x,p),vT),p);
    2248           1 :   (void)delete_var(); return gerepileupto(ltop,R);
    2249             : }
    2250             : 
    2251             : /* Computing minimal polynomial :                         */
    2252             : /* cf Shoup 'Efficient Computation of Minimal Polynomials */
    2253             : /*          in Algebraic Extensions of Finite Fields'     */
    2254             : 
    2255             : /* Let v a linear form, return the linear form z->v(tau*z)
    2256             :    that is, v*(M_tau) */
    2257             : 
    2258             : static GEN
    2259         534 : FpXQ_transmul_init(GEN tau, GEN T, GEN p)
    2260             : {
    2261             :   GEN bht;
    2262         534 :   GEN h, Tp = get_FpX_red(T, &h);
    2263         534 :   long n = degpol(Tp), vT = varn(Tp);
    2264         534 :   GEN ft = FpX_recipspec(Tp+2, n+1, n+1);
    2265         534 :   GEN bt = FpX_recipspec(tau+2, lgpol(tau), n);
    2266         534 :   setvarn(ft, vT); setvarn(bt, vT);
    2267         534 :   if (h)
    2268          14 :     bht = FpXn_mul(bt, h, n-1, p);
    2269             :   else
    2270             :   {
    2271         520 :     GEN bh = FpX_div(RgX_shift_shallow(tau, n-1), T, p);
    2272         520 :     bht = FpX_recipspec(bh+2, lgpol(bh), n-1);
    2273         520 :     setvarn(bht, vT);
    2274             :   }
    2275         534 :   return mkvec3(bt, bht, ft);
    2276             : }
    2277             : 
    2278             : static GEN
    2279        1495 : FpXQ_transmul(GEN tau, GEN a, long n, GEN p)
    2280             : {
    2281        1495 :   pari_sp ltop = avma;
    2282             :   GEN t1, t2, t3, vec;
    2283        1495 :   GEN bt = gel(tau, 1), bht = gel(tau, 2), ft = gel(tau, 3);
    2284        1495 :   if (signe(a)==0) return pol_0(varn(a));
    2285        1460 :   t2 = RgX_shift_shallow(FpX_mul(bt, a, p),1-n);
    2286        1460 :   if (signe(bht)==0) return gerepilecopy(ltop, t2);
    2287        1172 :   t1 = RgX_shift_shallow(FpX_mul(ft, a, p),-n);
    2288        1172 :   t3 = FpXn_mul(t1, bht, n-1, p);
    2289        1172 :   vec = FpX_sub(t2, RgX_shift_shallow(t3, 1), p);
    2290        1172 :   return gerepileupto(ltop, vec);
    2291             : }
    2292             : 
    2293             : GEN
    2294        5001 : FpXQ_minpoly(GEN x, GEN T, GEN p)
    2295             : {
    2296        5001 :   pari_sp ltop = avma;
    2297             :   long vT, n;
    2298             :   GEN v_x, g, tau;
    2299        5001 :   if (lgefint(p)==3)
    2300             :   {
    2301        4734 :     ulong pp = to_Flxq(&x, &T, p);
    2302        4734 :     GEN g = Flxq_minpoly(x, T, pp);
    2303        4734 :     return gerepileupto(ltop, Flx_to_ZX(g));
    2304             :   }
    2305         267 :   vT = get_FpX_var(T);
    2306         267 :   n = get_FpX_degree(T);
    2307         267 :   g = pol_1(vT);
    2308         267 :   tau = pol_1(vT);
    2309         267 :   T = FpX_get_red(T, p);
    2310         267 :   x = FpXQ_red(x, T, p);
    2311         267 :   v_x = FpXQ_powers(x, usqrt(2*n), T, p);
    2312         801 :   while(signe(tau) != 0)
    2313             :   {
    2314             :     long i, j, m, k1;
    2315             :     GEN M, v, tr;
    2316             :     GEN g_prime, c;
    2317         267 :     if (degpol(g) == n) { tau = pol_1(vT); g = pol_1(vT); }
    2318         267 :     v = random_FpX(n, vT, p);
    2319         267 :     tr = FpXQ_transmul_init(tau, T, p);
    2320         267 :     v = FpXQ_transmul(tr, v, n, p);
    2321         267 :     m = 2*(n-degpol(g));
    2322         267 :     k1 = usqrt(m);
    2323         267 :     tr = FpXQ_transmul_init(gel(v_x,k1+1), T, p);
    2324         267 :     c = cgetg(m+2,t_POL);
    2325         267 :     c[1] = evalsigne(1)|evalvarn(vT);
    2326        1495 :     for (i=0; i<m; i+=k1)
    2327             :     {
    2328        1228 :       long mj = minss(m-i, k1);
    2329        6890 :       for (j=0; j<mj; j++)
    2330        5662 :         gel(c,m+1-(i+j)) = FpX_dotproduct(v, gel(v_x,j+1), p);
    2331        1228 :       v = FpXQ_transmul(tr, v, n, p);
    2332             :     }
    2333         267 :     c = FpX_renormalize(c, m+2);
    2334             :     /* now c contains <v,x^i> , i = 0..m-1  */
    2335         267 :     M = FpX_halfgcd(pol_xn(m, vT), c, p);
    2336         267 :     g_prime = gmael(M, 2, 2);
    2337         267 :     if (degpol(g_prime) < 1) continue;
    2338         267 :     g = FpX_mul(g, g_prime, p);
    2339         267 :     tau = FpXQ_mul(tau, FpX_FpXQV_eval(g_prime, v_x, T, p), T, p);
    2340             :   }
    2341         267 :   g = FpX_normalize(g,p);
    2342         267 :   return gerepilecopy(ltop,g);
    2343             : }
    2344             : 
    2345             : GEN
    2346           8 : FpXQ_conjvec(GEN x, GEN T, GEN p)
    2347             : {
    2348           8 :   pari_sp av=avma;
    2349             :   long i;
    2350           8 :   long n = get_FpX_degree(T), v = varn(x);
    2351           8 :   GEN M = FpX_matFrobenius(T, p);
    2352           8 :   GEN z = cgetg(n+1,t_COL);
    2353           8 :   gel(z,1) = RgX_to_RgC(x,n);
    2354           8 :   for (i=2; i<=n; i++) gel(z,i) = FpM_FpC_mul(M,gel(z,i-1),p);
    2355           8 :   gel(z,1) = x;
    2356           8 :   for (i=2; i<=n; i++) gel(z,i) = RgV_to_RgX(gel(z,i),v);
    2357           8 :   return gerepilecopy(av,z);
    2358             : }
    2359             : 
    2360             : /* p prime, p_1 = p-1, q = p^deg T, Lp = cofactors of some prime divisors
    2361             :  * l_p of p-1, Lq = cofactors of some prime divisors l_q of q-1, return a
    2362             :  * g in Fq such that
    2363             :  * - Ng generates all l_p-Sylows of Fp^*
    2364             :  * - g generates all l_q-Sylows of Fq^* */
    2365             : static GEN
    2366        1626 : gener_FpXQ_i(GEN T, GEN p, GEN p_1, GEN Lp, GEN Lq)
    2367             : {
    2368             :   pari_sp av;
    2369        1626 :   long vT = varn(T), f = degpol(T), l = lg(Lq);
    2370        1626 :   GEN F = FpX_Frobenius(T, p);
    2371        1626 :   int p_is_2 = is_pm1(p_1);
    2372        3045 :   for (av = avma;; set_avma(av))
    2373        1419 :   {
    2374        3045 :     GEN t, g = random_FpX(f, vT, p);
    2375             :     long i;
    2376        3045 :     if (degpol(g) < 1) continue;
    2377        2911 :     if (p_is_2)
    2378         413 :       t = g;
    2379             :     else
    2380             :     {
    2381        2498 :       t = FpX_resultant(T, g, p); /* Ng = g^((q-1)/(p-1)), assuming T monic */
    2382        2498 :       if (kronecker(t, p) == 1) continue;
    2383        1398 :       if (lg(Lp) > 1 && !is_gener_Fp(t, p, p_1, Lp)) continue;
    2384        1398 :       t = FpXQ_pow(g, shifti(p_1,-1), T, p);
    2385             :     }
    2386        2435 :     for (i = 1; i < l; i++)
    2387             :     {
    2388         809 :       GEN a = FpXQ_pow_Frobenius(t, gel(Lq,i), F, T, p);
    2389         809 :       if (!degpol(a) && equalii(gel(a,2), p_1)) break;
    2390             :     }
    2391        3437 :     if (i == l) return g;
    2392             :   }
    2393             : }
    2394             : 
    2395             : GEN
    2396       11720 : gener_FpXQ(GEN T, GEN p, GEN *po)
    2397             : {
    2398       11720 :   long i, j, f = get_FpX_degree(T);
    2399             :   GEN g, Lp, Lq, p_1, q_1, N, o;
    2400       11720 :   pari_sp av = avma;
    2401             : 
    2402       11720 :   p_1 = subiu(p,1);
    2403       11720 :   if (f == 1) {
    2404             :     GEN Lp, fa;
    2405           7 :     o = p_1;
    2406           7 :     fa = Z_factor(o);
    2407           7 :     Lp = gel(fa,1);
    2408           7 :     Lp = vecslice(Lp, 2, lg(Lp)-1); /* remove 2 for efficiency */
    2409             : 
    2410           7 :     g = cgetg(3, t_POL);
    2411           7 :     g[1] = evalsigne(1) | evalvarn(get_FpX_var(T));
    2412           7 :     gel(g,2) = pgener_Fp_local(p, Lp);
    2413           7 :     if (po) *po = mkvec2(o, fa);
    2414           7 :     return g;
    2415             :   }
    2416       11713 :   if (lgefint(p) == 3)
    2417             :   {
    2418       11690 :     ulong pp = to_Flxq(NULL, &T, p);
    2419       11690 :     g = gener_Flxq(T, pp, po);
    2420       11690 :     if (!po) return Flx_to_ZX_inplace(gerepileuptoleaf(av, g));
    2421       11690 :     g = Flx_to_ZX(g);
    2422       11690 :     gerepileall(av, 2, &g, po);
    2423       11690 :     return g;
    2424             :   }
    2425             :   /* p now odd */
    2426          23 :   q_1 = subiu(powiu(p,f), 1);
    2427          23 :   N = diviiexact(q_1, p_1);
    2428          23 :   Lp = odd_prime_divisors(p_1);
    2429          23 :   for (i=lg(Lp)-1; i; i--) gel(Lp,i) = diviiexact(p_1, gel(Lp,i));
    2430          23 :   o = factor_pn_1(p,f);
    2431          23 :   Lq = leafcopy( gel(o, 1) );
    2432         199 :   for (i = j = 1; i < lg(Lq); i++)
    2433             :   {
    2434         176 :     if (dvdii(p_1, gel(Lq,i))) continue;
    2435         106 :     gel(Lq,j++) = diviiexact(N, gel(Lq,i));
    2436             :   }
    2437          23 :   setlg(Lq, j);
    2438          23 :   g = gener_FpXQ_i(get_FpX_mod(T), p, p_1, Lp, Lq);
    2439          23 :   if (!po) g = gerepilecopy(av, g);
    2440             :   else {
    2441           7 :     *po = mkvec2(q_1, o);
    2442           7 :     gerepileall(av, 2, &g, po);
    2443             :   }
    2444          23 :   return g;
    2445             : }
    2446             : 
    2447             : GEN
    2448        1603 : gener_FpXQ_local(GEN T, GEN p, GEN L)
    2449             : {
    2450        1603 :   GEN Lp, Lq, p_1 = subiu(p,1), q_1, N, Q;
    2451        1603 :   long f, i, ip, iq, l = lg(L);
    2452        1603 :   T = get_FpX_mod(T);
    2453        1603 :   f = degpol(T);
    2454        1603 :   q_1 = subiu(powiu(p,f), 1);
    2455        1603 :   N = diviiexact(q_1, p_1);
    2456             : 
    2457        1603 :   Q = is_pm1(p_1)? gen_1: shifti(p_1,-1);
    2458        1603 :   Lp = cgetg(l, t_VEC); ip = 1;
    2459        1603 :   Lq = cgetg(l, t_VEC); iq = 1;
    2460        2422 :   for (i=1; i < l; i++)
    2461             :   {
    2462         819 :     GEN a, b, ell = gel(L,i);
    2463         819 :     if (absequaliu(ell,2)) continue;
    2464         539 :     a = dvmdii(Q, ell, &b);
    2465         539 :     if (b == gen_0)
    2466          21 :       gel(Lp,ip++) = a;
    2467             :     else
    2468         518 :       gel(Lq,iq++) = diviiexact(N,ell);
    2469             :   }
    2470        1603 :   setlg(Lp, ip);
    2471        1603 :   setlg(Lq, iq);
    2472        1603 :   return gener_FpXQ_i(T, p, p_1, Lp, Lq);
    2473             : }
    2474             : 
    2475             : /***********************************************************************/
    2476             : /**                                                                   **/
    2477             : /**                              FpXn                                 **/
    2478             : /**                                                                   **/
    2479             : /***********************************************************************/
    2480             : 
    2481             : INLINE GEN
    2482        1358 : FpXn_red(GEN a, long n)
    2483        1358 : { return RgXn_red_shallow(a, n); }
    2484             : 
    2485             : GEN
    2486     2090602 : FpXn_mul(GEN a, GEN b, long n, GEN p)
    2487             : {
    2488     2090602 :   return FpX_red(ZXn_mul(a, b, n), p);
    2489             : }
    2490             : 
    2491             : GEN
    2492         679 : FpXn_sqr(GEN a, long n, GEN p)
    2493             : {
    2494         679 :   return FpX_red(ZXn_sqr(a, n), p);
    2495             : }
    2496             : 
    2497             : GEN
    2498         182 : FpXn_exp(GEN h, long e, GEN p)
    2499             : {
    2500         182 :   pari_sp av = avma, av2;
    2501         182 :   long v = varn(h), n=1;
    2502         182 :   GEN f = pol_1(v), g = pol_1(v);
    2503         182 :   ulong mask = quadratic_prec_mask(e);
    2504         182 :   av2 = avma;
    2505         182 :   if (signe(h)==0 || degpol(h)<1 || !gequal0(gel(h,2)))
    2506           0 :     pari_err_DOMAIN("FpXn_exp","valuation", "<", gen_1, h);
    2507        1043 :   for (;mask>1;)
    2508             :   {
    2509             :     GEN q, w;
    2510         679 :     long n2 = n;
    2511         679 :     n<<=1; if (mask & 1) n--;
    2512         679 :     mask >>= 1;
    2513         679 :     g = FpX_sub(FpX_mulu(g,2,p), FpXn_mul(f, FpXn_sqr(g, n2, p), n2, p), p);
    2514         679 :     q = FpX_deriv(FpXn_red(h,n2), p);
    2515         679 :     w = FpX_add(q, FpXn_mul(g, FpX_sub(FpX_deriv(f, p), FpXn_mul(f,q,n-1, p), p),n-1, p), p);
    2516         679 :     f = FpX_add(f, FpXn_mul(f, FpX_sub(FpXn_red(h, n), FpX_integ(w,p), p), n, p), p);
    2517         679 :     if (gc_needed(av2,2))
    2518             :     {
    2519           0 :       if (DEBUGMEM>1) pari_warn(warnmem,"FpXn_exp, e = %ld", n);
    2520           0 :       gerepileall(av2, 2, &f, &g);
    2521             :     }
    2522             :   }
    2523         182 :   return gerepileupto(av, f);
    2524             : }
    2525             : 
    2526             : /* (f*g) \/ x^n */
    2527             : static GEN
    2528           0 : FpX_mulhigh_i(GEN f, GEN g, long n, GEN p)
    2529             : {
    2530           0 :   return RgX_shift_shallow(FpX_mul(f,g, p),-n);
    2531             : }
    2532             : 
    2533             : static GEN
    2534           0 : FpXn_mulhigh(GEN f, GEN g, long n2, long n, GEN p)
    2535             : {
    2536           0 :   GEN F = RgX_blocks(f, n2, 2), fl = gel(F,1), fh = gel(F,2);
    2537           0 :   return FpX_add(FpX_mulhigh_i(fl, g, n2, p), FpXn_mul(fh, g, n - n2, p), p);
    2538             : }
    2539             : 
    2540             : GEN
    2541           0 : FpXn_inv(GEN f, long e, GEN p)
    2542             : {
    2543           0 :   pari_sp av = avma, av2;
    2544             :   ulong mask;
    2545             :   GEN W, a;
    2546           0 :   long v = varn(f), n = 1;
    2547             : 
    2548           0 :   if (!signe(f)) pari_err_INV("FpXn_inv",f);
    2549           0 :   a = Fp_inv(gel(f,2), p);
    2550           0 :   if (e == 1) return scalarpol(a, v);
    2551           0 :   else if (e == 2)
    2552             :   {
    2553             :     GEN b;
    2554           0 :     if (degpol(f) <= 0) return scalarpol(a, v);
    2555           0 :     b = Fp_neg(gel(f,3),p);
    2556           0 :     if (signe(b)==0) return scalarpol(a, v);
    2557           0 :     if (!is_pm1(a)) b = Fp_mul(b, Fp_sqr(a, p), p);
    2558           0 :     W = deg1pol_shallow(b, a, v);
    2559           0 :     return gerepilecopy(av, W);
    2560             :   }
    2561           0 :   W = scalarpol_shallow(Fp_inv(gel(f,2), p),v);
    2562           0 :   mask = quadratic_prec_mask(e);
    2563           0 :   av2 = avma;
    2564           0 :   for (;mask>1;)
    2565             :   {
    2566             :     GEN u, fr;
    2567           0 :     long n2 = n;
    2568           0 :     n<<=1; if (mask & 1) n--;
    2569           0 :     mask >>= 1;
    2570           0 :     fr = FpXn_red(f, n);
    2571           0 :     u = FpXn_mul(W, FpXn_mulhigh(fr, W, n2, n, p), n-n2, p);
    2572           0 :     W = FpX_sub(W, RgX_shift_shallow(u, n2), p);
    2573           0 :     if (gc_needed(av2,2))
    2574             :     {
    2575           0 :       if(DEBUGMEM>1) pari_warn(warnmem,"FpXn_inv, e = %ld", n);
    2576           0 :       W = gerepileupto(av2, W);
    2577             :     }
    2578             :   }
    2579           0 :   return gerepileupto(av, W);
    2580             : }

Generated by: LCOV version 1.13