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 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.8.0 lcov report (development 19374-ac31b96) Lines: 1228 1328 92.5 %
Date: 2016-08-30 06:11:37 Functions: 147 153 96.1 %
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    67860489 : get_FpX_red(GEN T, GEN *B)
      21             : {
      22    67860489 :   if (typ(T)!=t_VEC) { *B=NULL; return T; }
      23      121346 :   *B = gel(T,1); return gel(T,2);
      24             : }
      25             : 
      26             : GEN
      27      395740 : get_FpX_mod(GEN T) { return typ(T)==t_VEC? gel(T,2): T; }
      28             : 
      29             : long
      30     1441604 : get_FpX_var(GEN T) { return typ(T)==t_VEC? varn(gel(T,2)): varn(T); }
      31             : 
      32             : long
      33     2350337 : get_FpX_degree(GEN T) { return typ(T)==t_VEC? degpol(gel(T,2)): degpol(T); }
      34             : 
      35             : /***********************************************************************/
      36             : /**                                                                   **/
      37             : /**                              FpX                                  **/
      38             : /**                                                                   **/
      39             : /***********************************************************************/
      40             : 
      41             : /* FpX are polynomials over Z/pZ represented as t_POL with
      42             :  * t_INT coefficients.
      43             :  * 1) Coefficients should belong to {0,...,p-1}, though non-reduced
      44             :  * coefficients should work but be slower.
      45             :  *
      46             :  * 2) p is not assumed to be prime, but it is assumed that impossible divisions
      47             :  *    will not happen.
      48             :  * 3) Theses functions let some garbage on the stack, but are gerepileupto
      49             :  * compatible.
      50             :  */
      51             : 
      52             : static ulong
      53    39891676 : to_Flx(GEN *P, GEN *Q, GEN p)
      54             : {
      55    39891676 :   ulong pp = uel(p,2);
      56    39891676 :   *P = ZX_to_Flx(*P, pp);
      57    39891676 :   *Q = ZX_to_Flx(*Q, pp); return pp;
      58             : }
      59             : 
      60             : static ulong
      61      750544 : to_Flxq(GEN *P, GEN *T, GEN p)
      62             : {
      63      750544 :   ulong pp = uel(p,2);
      64      750544 :   if (P) *P = ZX_to_Flx(*P, pp);
      65      750543 :   *T = ZXT_to_FlxT(*T, pp); return pp;
      66             : }
      67             : 
      68             : GEN
      69        1710 : Z_to_FpX(GEN a, GEN p, long v)
      70             : {
      71        1710 :   pari_sp av = avma;
      72        1710 :   GEN z = cgetg(3, t_POL);
      73        1710 :   GEN x = modii(a, p);
      74        1710 :   if (!signe(x)) { avma =av; return pol_0(v); }
      75        1710 :   z[1] = evalsigne(1) | evalvarn(v);
      76        1710 :   gel(z,2) = x; return z;
      77             : }
      78             : 
      79             : /* z in Z[X], return lift(z * Mod(1,p)), normalized*/
      80             : GEN
      81    70084697 : FpX_red(GEN z, GEN p)
      82             : {
      83    70084697 :   long i, l = lg(z);
      84    70084697 :   GEN x = cgetg(l, t_POL);
      85    70144544 :   for (i=2; i<l; i++) gel(x,i) = modii(gel(z,i),p);
      86    70084657 :   x[1] = z[1]; return FpX_renormalize(x,l);
      87             : }
      88             : GEN
      89      288448 : FpXV_red(GEN z, GEN p)
      90             : {
      91      288448 :   long i,l = lg(z);
      92      288448 :   GEN x = cgetg(l, t_VEC);
      93      288448 :   for (i=1; i<l; i++) gel(x,i) = FpX_red(gel(z,i), p);
      94      288448 :   return x;
      95             : }
      96             : 
      97             : GEN
      98     1089112 : FpXT_red(GEN z, GEN p)
      99             : {
     100     1089112 :   if (typ(z) == t_POL)
     101     1006835 :     return FpX_red(z, p);
     102             :   else
     103             :   {
     104       82277 :     long i,l = lg(z);
     105       82277 :     GEN x = cgetg(l, t_VEC);
     106       82277 :     for (i=1; i<l; i++) gel(x,i) = FpXT_red(gel(z,i), p);
     107       82277 :     return x;
     108             :   }
     109             : }
     110             : 
     111             : GEN
     112      244710 : FpX_normalize(GEN z, GEN p)
     113             : {
     114      244710 :   GEN p1 = leading_coeff(z);
     115      244710 :   if (lg(z) == 2 || equali1(p1)) return z;
     116       52705 :   return FpX_Fp_mul_to_monic(z, Fp_inv(p1,p), p);
     117             : }
     118             : 
     119             : GEN
     120      747359 : FpX_center(GEN T, GEN p, GEN pov2)
     121             : {
     122      747359 :   long i, l = lg(T);
     123      747359 :   GEN P = cgetg(l,t_POL);
     124      747359 :   for(i=2; i<l; i++) gel(P,i) = Fp_center(gel(T,i), p, pov2);
     125      747359 :   P[1] = T[1]; return P;
     126             : }
     127             : 
     128             : GEN
     129     8128479 : FpX_add(GEN x,GEN y,GEN p)
     130             : {
     131     8128479 :   long lx = lg(x), ly = lg(y), i;
     132             :   GEN z;
     133     8128479 :   if (lx < ly) swapspec(x,y, lx,ly);
     134     8128479 :   z = cgetg(lx,t_POL); z[1] = x[1];
     135     8128479 :   for (i=2; i<ly; i++) gel(z,i) = Fp_add(gel(x,i),gel(y,i), p);
     136     8128479 :   for (   ; i<lx; i++) gel(z,i) = modii(gel(x,i), p);
     137     8128479 :   z = ZX_renormalize(z, lx);
     138     8128479 :   if (!lgpol(z)) { avma = (pari_sp)(z + lx); return pol_0(varn(x)); }
     139     7822815 :   return z;
     140             : }
     141             : 
     142             : static GEN
     143        6584 : Fp_red_FpX(GEN x, GEN p, long v)
     144             : {
     145             :   GEN z;
     146        6584 :   if (!signe(x)) return pol_0(v);
     147         442 :   z = cgetg(3, t_POL);
     148         442 :   gel(z,2) = Fp_red(x,p);
     149         442 :   z[1] = evalvarn(v);
     150         442 :   return FpX_renormalize(z, 3);
     151             : }
     152             : 
     153             : static GEN
     154          17 : Fp_neg_FpX(GEN x, GEN p, long v)
     155             : {
     156             :   GEN z;
     157          17 :   if (!signe(x)) return pol_0(v);
     158           1 :   z = cgetg(3, t_POL);
     159           1 :   gel(z,2) = Fp_neg(x,p);
     160           1 :   z[1] = evalvarn(v);
     161           1 :   return FpX_renormalize(z, 3);
     162             : }
     163             : 
     164             : GEN
     165      550558 : FpX_Fp_add(GEN y,GEN x,GEN p)
     166             : {
     167      550558 :   long i, lz = lg(y);
     168             :   GEN z;
     169      550558 :   if (lz == 2) return Fp_red_FpX(x,p,varn(y));
     170      543974 :   z = cgetg(lz,t_POL); z[1] = y[1];
     171      543974 :   gel(z,2) = Fp_add(gel(y,2),x, p);
     172      543974 :   if (lz == 3) z = FpX_renormalize(z,lz);
     173             :   else
     174      495968 :     for(i=3;i<lz;i++) gel(z,i) = icopy(gel(y,i));
     175      543974 :   return z;
     176             : }
     177             : GEN
     178           0 : FpX_Fp_add_shallow(GEN y,GEN x,GEN p)
     179             : {
     180           0 :   long i, lz = lg(y);
     181             :   GEN z;
     182           0 :   if (lz == 2) return scalar_ZX_shallow(x,varn(y));
     183           0 :   z = cgetg(lz,t_POL); z[1] = y[1];
     184           0 :   gel(z,2) = Fp_add(gel(y,2),x, p);
     185           0 :   if (lz == 3) z = FpX_renormalize(z,lz);
     186             :   else
     187           0 :     for(i=3;i<lz;i++) gel(z,i) = gel(y,i);
     188           0 :   return z;
     189             : }
     190             : GEN
     191      298471 : FpX_Fp_sub(GEN y,GEN x,GEN p)
     192             : {
     193      298471 :   long i, lz = lg(y);
     194             :   GEN z;
     195      298471 :   if (lz == 2) return Fp_neg_FpX(x,p,varn(y));
     196      298454 :   z = cgetg(lz,t_POL); z[1] = y[1];
     197      298454 :   gel(z,2) = Fp_sub(gel(y,2),x, p);
     198      298454 :   if (lz == 3) z = FpX_renormalize(z,lz);
     199             :   else
     200      121360 :     for(i=3;i<lz;i++) gel(z,i) = icopy(gel(y,i));
     201      298454 :   return z;
     202             : }
     203             : GEN
     204        1565 : FpX_Fp_sub_shallow(GEN y,GEN x,GEN p)
     205             : {
     206        1565 :   long i, lz = lg(y);
     207             :   GEN z;
     208        1565 :   if (lz == 2) return Fp_neg_FpX(x,p,varn(y));
     209        1565 :   z = cgetg(lz,t_POL); z[1] = y[1];
     210        1565 :   gel(z,2) = Fp_sub(gel(y,2),x, p);
     211        1565 :   if (lz == 3) z = FpX_renormalize(z,lz);
     212             :   else
     213        1405 :     for(i=3;i<lz;i++) gel(z,i) = gel(y,i);
     214        1565 :   return z;
     215             : }
     216             : 
     217             : GEN
     218       80001 : FpX_neg(GEN x,GEN p)
     219             : {
     220       80001 :   long i, lx = lg(x);
     221       80001 :   GEN y = cgetg(lx,t_POL);
     222       80001 :   y[1] = x[1];
     223       80001 :   for(i=2; i<lx; i++) gel(y,i) = Fp_neg(gel(x,i), p);
     224       80001 :   return ZX_renormalize(y, lx);
     225             : }
     226             : 
     227             : static GEN
     228     6823457 : FpX_subspec(GEN x,GEN y,GEN p, long nx, long ny)
     229             : {
     230             :   long i, lz;
     231             :   GEN z;
     232     6823457 :   if (nx >= ny)
     233             :   {
     234     5017586 :     lz = nx+2;
     235     5017586 :     z = cgetg(lz,t_POL); z[1] = 0; z += 2;
     236     5019521 :     for (i=0; i<ny; i++) gel(z,i) = Fp_sub(gel(x,i),gel(y,i), p);
     237     5017588 :     for (   ; i<nx; i++) gel(z,i) = modii(gel(x,i), p);
     238             :   }
     239             :   else
     240             :   {
     241     1805871 :     lz = ny+2;
     242     1805871 :     z = cgetg(lz,t_POL); z[1] = 0; z += 2;
     243     1805881 :     for (i=0; i<nx; i++) gel(z,i) = Fp_sub(gel(x,i),gel(y,i), p);
     244     1805871 :     for (   ; i<ny; i++) gel(z,i) = Fp_neg(gel(y,i), p);
     245             :   }
     246     6823455 :   z = FpX_renormalize(z-2, lz);
     247     6823459 :   if (!lgpol(z)) { avma = (pari_sp)(z + lz); return pol_0(0); }
     248     6663479 :   return z;
     249             : }
     250             : 
     251             : GEN
     252     6724993 : FpX_sub(GEN x,GEN y,GEN p)
     253             : {
     254     6724993 :   GEN z = FpX_subspec(x+2,y+2,p,lgpol(x),lgpol(y));
     255     6724994 :   setvarn(z, varn(x));
     256     6724994 :   return z;
     257             : }
     258             : 
     259             : GEN
     260        9215 : Fp_FpX_sub(GEN x, GEN y, GEN p)
     261             : {
     262        9215 :   long ly = lg(y), i;
     263             :   GEN z;
     264        9215 :   if (ly <= 3) {
     265         220 :     z = cgetg(3, t_POL);
     266         220 :     x = (ly == 3)? Fp_sub(x, gel(y,2), p): modii(x, p);
     267         220 :     if (!signe(x)) { avma = (pari_sp)(z + 3); return pol_0(varn(y)); }
     268         188 :     z[1] = evalsigne(1)|y[1]; gel(z,2) = x; return z;
     269             :   }
     270        8995 :   z = cgetg(ly,t_POL);
     271        8995 :   gel(z,2) = Fp_sub(x, gel(y,2), p);
     272        8995 :   for (i = 3; i < ly; i++) gel(z,i) = Fp_neg(gel(y,i), p);
     273        8995 :   z = ZX_renormalize(z, ly);
     274        8995 :   if (!lgpol(z)) { avma = (pari_sp)(z + ly); return pol_0(varn(x)); }
     275        8995 :   z[1] = y[1]; return z;
     276             : }
     277             : 
     278             : GEN
     279    47592831 : FpX_mul(GEN x,GEN y,GEN p) { return FpX_red(ZX_mul(x, y), p); }
     280             : 
     281             : GEN
     282     1662783 : FpX_mulspec(GEN a, GEN b, GEN p, long na, long nb)
     283     1662783 : { return FpX_red(ZX_mulspec(a, b, na, nb), p); }
     284             : 
     285             : GEN
     286     3858679 : FpX_sqr(GEN x,GEN p) { return FpX_red(ZX_sqr(x), p); }
     287             : 
     288             : GEN
     289     1179498 : FpX_mulu(GEN y, ulong x,GEN p)
     290             : {
     291             :   GEN z;
     292             :   long i, l;
     293     1179498 :   x = umodui(x, p);
     294     1179498 :   if (!x) return zeropol(varn(y));
     295     1179197 :   z = cgetg_copy(y, &l); z[1] = y[1];
     296     1179197 :   for(i=2; i<l; i++) gel(z,i) = Fp_mulu(gel(y,i), x, p);
     297     1179197 :   return z;
     298             : }
     299             : 
     300             : GEN
     301     3246535 : FpX_Fp_mulspec(GEN y,GEN x,GEN p,long ly)
     302             : {
     303             :   GEN z;
     304             :   long i;
     305     3246535 :   if (!signe(x)) return pol_0(0);
     306     2988483 :   z = cgetg(ly+2,t_POL); z[1] = evalsigne(1);
     307     2988483 :   for(i=0; i<ly; i++) gel(z,i+2) = Fp_mul(gel(y,i), x, p);
     308     2988483 :   return ZX_renormalize(z, ly+2);
     309             : }
     310             : 
     311             : GEN
     312     3240851 : FpX_Fp_mul(GEN y,GEN x,GEN p)
     313             : {
     314     3240851 :   GEN z = FpX_Fp_mulspec(y+2,x,p,lgpol(y));
     315     3240851 :   setvarn(z, varn(y)); return z;
     316             : }
     317             : 
     318             : GEN
     319       52705 : FpX_Fp_mul_to_monic(GEN y,GEN x,GEN p)
     320             : {
     321             :   GEN z;
     322             :   long i, l;
     323       52705 :   z = cgetg_copy(y, &l); z[1] = y[1];
     324       52705 :   for(i=2; i<l-1; i++) gel(z,i) = Fp_mul(gel(y,i), x, p);
     325       52705 :   gel(z,l-1) = gen_1; return z;
     326             : }
     327             : 
     328             : struct _FpXQ {
     329             :   GEN T, p, aut;
     330             : };
     331             : 
     332             : static GEN
     333       55083 : _FpX_sqr(void *data, GEN x)
     334             : {
     335       55083 :   struct _FpXQ *D = (struct _FpXQ*)data;
     336       55083 :   return FpX_sqr(x, D->p);
     337             : }
     338             : static GEN
     339       68596 : _FpX_mul(void *data, GEN x, GEN y)
     340             : {
     341       68596 :   struct _FpXQ *D = (struct _FpXQ*)data;
     342       68596 :   return FpX_mul(x,y, D->p);
     343             : }
     344             : 
     345             : GEN
     346      258174 : FpX_powu(GEN x, ulong n, GEN p)
     347             : {
     348             :   struct _FpXQ D;
     349      258174 :   if (n==0) return pol_1(varn(x));
     350       24542 :   D.p = p;
     351       24542 :   return gen_powu(x, n, (void *)&D, _FpX_sqr, _FpX_mul);
     352             : }
     353             : 
     354             : GEN
     355        1124 : FpX_halve(GEN y, GEN p)
     356             : {
     357             :   GEN z;
     358             :   long i, l;
     359        1124 :   z = cgetg_copy(y, &l); z[1] = y[1];
     360        1124 :   for(i=2; i<l; i++) gel(z,i) = Fp_halve(gel(y,i), p);
     361        1124 :   return z;
     362             : }
     363             : 
     364             : static GEN
     365    63935670 : FpX_divrem_basecase(GEN x, GEN y, GEN p, GEN *pr)
     366             : {
     367             :   long vx, dx, dy, dy1, dz, i, j, sx, lr;
     368             :   pari_sp av0, av;
     369             :   GEN z,p1,rem,lead;
     370             : 
     371    63935670 :   if (!signe(y)) pari_err_INV("FpX_divrem",y);
     372    63935670 :   vx = varn(x);
     373    63935670 :   dy = degpol(y);
     374    63935670 :   dx = degpol(x);
     375    63935670 :   if (dx < dy)
     376             :   {
     377       54055 :     if (pr)
     378             :     {
     379       54040 :       av0 = avma; x = FpX_red(x, p);
     380       54040 :       if (pr == ONLY_DIVIDES) { avma=av0; return signe(x)? NULL: pol_0(vx); }
     381       53998 :       if (pr == ONLY_REM) return x;
     382       53998 :       *pr = x;
     383             :     }
     384       54013 :     return pol_0(vx);
     385             :   }
     386    63881615 :   lead = leading_coeff(y);
     387    63881615 :   if (!dy) /* y is constant */
     388             :   {
     389      288827 :     if (pr && pr != ONLY_DIVIDES)
     390             :     {
     391      276451 :       if (pr == ONLY_REM) return pol_0(vx);
     392      244465 :       *pr = pol_0(vx);
     393             :     }
     394      256841 :     av0 = avma;
     395      256841 :     if (equali1(lead)) return FpX_red(x, p);
     396      254156 :     else return gerepileupto(av0, FpX_Fp_mul(x, Fp_inv(lead,p), p));
     397             :   }
     398    63592788 :   av0 = avma; dz = dx-dy;
     399    63592788 :   if (lgefint(p) == 3)
     400             :   { /* assume ab != 0 mod p */
     401    39429511 :     ulong pp = to_Flx(&x, &y, p);
     402    39429511 :     z = Flx_divrem(x, y, pp, pr);
     403    39429511 :     avma = av0; /* HACK: assume pr last on stack, then z */
     404    39429511 :     if (!z) return NULL;
     405    39429476 :     z = leafcopy(z);
     406    39429476 :     if (pr && pr != ONLY_DIVIDES && pr != ONLY_REM)
     407             :     {
     408     1811911 :       *pr = leafcopy(*pr);
     409     1811911 :       *pr = Flx_to_ZX_inplace(*pr);
     410             :     }
     411    39429476 :     return Flx_to_ZX_inplace(z);
     412             :   }
     413    24163277 :   lead = equali1(lead)? NULL: gclone(Fp_inv(lead,p));
     414    24163214 :   avma = av0;
     415    24163214 :   z=cgetg(dz+3,t_POL); z[1] = x[1];
     416    24163214 :   x += 2; y += 2; z += 2;
     417    24163214 :   for (dy1=dy-1; dy1>=0 && !signe(gel(y, dy1)); dy1--);
     418             : 
     419    24163214 :   p1 = gel(x,dx); av = avma;
     420    24163214 :   gel(z,dz) = lead? gerepileuptoint(av, Fp_mul(p1,lead, p)): icopy(p1);
     421    53623403 :   for (i=dx-1; i>=dy; i--)
     422             :   {
     423    29460189 :     av=avma; p1=gel(x,i);
     424   316765197 :     for (j=i-dy1; j<=i && j<=dz; j++)
     425   287305008 :       p1 = subii(p1, mulii(gel(z,j),gel(y,i-j)));
     426    29460189 :     if (lead) p1 = mulii(p1,lead);
     427    29460189 :     gel(z,i-dy) = gerepileuptoint(av,modii(p1, p));
     428             :   }
     429    24163214 :   if (!pr) { if (lead) gunclone(lead); return z-2; }
     430             : 
     431    24145461 :   rem = (GEN)avma; av = (pari_sp)new_chunk(dx+3);
     432    24937457 :   for (sx=0; ; i--)
     433             :   {
     434    24937457 :     p1 = gel(x,i);
     435    76595234 :     for (j=maxss(0,i-dy1); j<=i && j<=dz; j++)
     436    51657777 :       p1 = subii(p1, mulii(gel(z,j),gel(y,i-j)));
     437    24937457 :     p1 = modii(p1,p); if (signe(p1)) { sx = 1; break; }
     438      887889 :     if (!i) break;
     439      791996 :     avma=av;
     440      791996 :   }
     441    24145461 :   if (pr == ONLY_DIVIDES)
     442             :   {
     443           0 :     if (lead) gunclone(lead);
     444           0 :     if (sx) { avma=av0; return NULL; }
     445           0 :     avma = (pari_sp)rem; return z-2;
     446             :   }
     447    24145461 :   lr=i+3; rem -= lr;
     448    24145461 :   rem[0] = evaltyp(t_POL) | evallg(lr);
     449    24145461 :   rem[1] = z[-1];
     450    24145461 :   p1 = gerepileuptoint((pari_sp)rem, p1);
     451    24145461 :   rem += 2; gel(rem,i) = p1;
     452    86698122 :   for (i--; i>=0; i--)
     453             :   {
     454    62552661 :     av=avma; p1 = gel(x,i);
     455   427017234 :     for (j=maxss(0,i-dy1); j<=i && j<=dz; j++)
     456   364464573 :       p1 = subii(p1, mulii(gel(z,j),gel(y,i-j)));
     457    62552661 :     gel(rem,i) = gerepileuptoint(av, modii(p1,p));
     458             :   }
     459    24145461 :   rem -= 2;
     460    24145461 :   if (lead) gunclone(lead);
     461    24145461 :   if (!sx) (void)FpX_renormalize(rem, lr);
     462    24145461 :   if (pr == ONLY_REM) return gerepileupto(av0,rem);
     463      531270 :   *pr = rem; return z-2;
     464             : }
     465             : 
     466             : GEN
     467       18977 : FpX_div_by_X_x(GEN a, GEN x, GEN p, GEN *r)
     468             : {
     469       18977 :   long l = lg(a)-1, i;
     470       18977 :   GEN z = cgetg(l, t_POL);
     471       18977 :   z[1] = evalsigne(1) | evalvarn(0);
     472       18977 :   gel(z, l-1) = gel(a,l);
     473      626395 :   for (i=l-2; i>1; i--) /* z[i] = a[i+1] + x*z[i+1] */
     474      607418 :     gel(z, i) = Fp_addmul(gel(a,i+1), x, gel(z,i+1), p);
     475       18977 :   if (r) *r = Fp_addmul(gel(a,2), x, gel(z,2), p);
     476       18977 :   return z;
     477             : }
     478             : 
     479             : static GEN
     480       69706 : _FpX_divrem(void * E, GEN x, GEN y, GEN *r)
     481             : {
     482       69706 :   struct _FpXQ *D = (struct _FpXQ*) E;
     483       69706 :   return FpX_divrem(x, y, D->p, r);
     484             : }
     485             : static GEN
     486       10241 : _FpX_add(void * E, GEN x, GEN y) {
     487       10241 :   struct _FpXQ *D = (struct _FpXQ*) E;
     488       10241 :   return FpX_add(x, y, D->p);
     489             : }
     490             : 
     491             : static struct bb_ring FpX_ring = { _FpX_add,_FpX_mul,_FpX_sqr };
     492             : 
     493             : GEN
     494        5642 : FpX_digits(GEN x, GEN T, GEN p)
     495             : {
     496        5642 :   pari_sp av = avma;
     497             :   struct _FpXQ D;
     498        5642 :   long d = degpol(T), n = (lgpol(x)+d-1)/d;
     499             :   GEN z;
     500        5642 :   D.p = p;
     501        5642 :   z = gen_digits(x,T,n,(void *)&D, &FpX_ring, _FpX_divrem);
     502        5642 :   return gerepileupto(av, z);
     503             : }
     504             : 
     505             : GEN
     506        2296 : FpX_fromdigits(GEN x, GEN T, GEN p)
     507             : {
     508        2296 :   pari_sp av = avma;
     509             :   struct _FpXQ D;
     510             :   GEN z;
     511        2296 :   D.p = p;
     512        2296 :   z = gen_fromdigits(x,T,(void *)&D, &FpX_ring);
     513        2296 :   return gerepileupto(av, z);
     514             : }
     515             : 
     516             : long
     517       23282 : FpX_valrem(GEN x, GEN t, GEN p, GEN *py)
     518             : {
     519       23282 :   pari_sp av=avma;
     520             :   long k;
     521             :   GEN r, y;
     522             : 
     523       67620 :   for (k=0; ; k++)
     524             :   {
     525       67620 :     y = FpX_divrem(x, t, p, &r);
     526       67620 :     if (signe(r)) break;
     527       44338 :     x = y;
     528       44338 :   }
     529       23282 :   *py = gerepilecopy(av,x);
     530       23282 :   return k;
     531             : }
     532             : 
     533             : static GEN
     534         187 : FpX_halfgcd_basecase(GEN a, GEN b, GEN p)
     535             : {
     536         187 :   pari_sp av=avma;
     537             :   GEN u,u1,v,v1;
     538         187 :   long vx = varn(a);
     539         187 :   long n = lgpol(a)>>1;
     540         187 :   u1 = v = pol_0(vx);
     541         187 :   u = v1 = pol_1(vx);
     542        3475 :   while (lgpol(b)>n)
     543             :   {
     544        3101 :     GEN r, q = FpX_divrem(a,b,p, &r);
     545        3101 :     a = b; b = r; swap(u,u1); swap(v,v1);
     546        3101 :     u1 = FpX_sub(u1, FpX_mul(u, q, p), p);
     547        3101 :     v1 = FpX_sub(v1, FpX_mul(v, q ,p), p);
     548        3101 :     if (gc_needed(av,2))
     549             :     {
     550           0 :       if (DEBUGMEM>1) pari_warn(warnmem,"FpX_halfgcd (d = %ld)",degpol(b));
     551           0 :       gerepileall(av,6, &a,&b,&u1,&v1,&u,&v);
     552             :     }
     553             :   }
     554         187 :   return gerepilecopy(av, mkmat2(mkcol2(u,u1), mkcol2(v,v1)));
     555             : }
     556             : static GEN
     557         340 : FpX_addmulmul(GEN u, GEN v, GEN x, GEN y, GEN p)
     558             : {
     559         340 :   return FpX_add(FpX_mul(u, x, p),FpX_mul(v, y, p), p);
     560             : }
     561             : 
     562             : static GEN
     563         166 : FpXM_FpX_mul2(GEN M, GEN x, GEN y, GEN p)
     564             : {
     565         166 :   GEN res = cgetg(3, t_COL);
     566         166 :   gel(res, 1) = FpX_addmulmul(gcoeff(M,1,1), gcoeff(M,1,2), x, y, p);
     567         166 :   gel(res, 2) = FpX_addmulmul(gcoeff(M,2,1), gcoeff(M,2,2), x, y, p);
     568         166 :   return res;
     569             : }
     570             : 
     571             : static GEN
     572         166 : FpXM_mul2(GEN A, GEN B, GEN p)
     573             : {
     574         166 :   GEN A11=gcoeff(A,1,1),A12=gcoeff(A,1,2), B11=gcoeff(B,1,1),B12=gcoeff(B,1,2);
     575         166 :   GEN A21=gcoeff(A,2,1),A22=gcoeff(A,2,2), B21=gcoeff(B,2,1),B22=gcoeff(B,2,2);
     576         166 :   GEN M1 = FpX_mul(FpX_add(A11,A22, p), FpX_add(B11,B22, p), p);
     577         166 :   GEN M2 = FpX_mul(FpX_add(A21,A22, p), B11, p);
     578         166 :   GEN M3 = FpX_mul(A11, FpX_sub(B12,B22, p), p);
     579         166 :   GEN M4 = FpX_mul(A22, FpX_sub(B21,B11, p), p);
     580         166 :   GEN M5 = FpX_mul(FpX_add(A11,A12, p), B22, p);
     581         166 :   GEN M6 = FpX_mul(FpX_sub(A21,A11, p), FpX_add(B11,B12, p), p);
     582         166 :   GEN M7 = FpX_mul(FpX_sub(A12,A22, p), FpX_add(B21,B22, p), p);
     583         166 :   GEN T1 = FpX_add(M1,M4, p), T2 = FpX_sub(M7,M5, p);
     584         166 :   GEN T3 = FpX_sub(M1,M2, p), T4 = FpX_add(M3,M6, p);
     585         166 :   retmkmat2(mkcol2(FpX_add(T1,T2, p), FpX_add(M2,M4, p)),
     586             :             mkcol2(FpX_add(M3,M5, p), FpX_add(T3,T4, p)));
     587             : }
     588             : 
     589             : /* Return [0,1;1,-q]*M */
     590             : static GEN
     591         162 : FpX_FpXM_qmul(GEN q, GEN M, GEN p)
     592             : {
     593         162 :   GEN u, v, res = cgetg(3, t_MAT);
     594         162 :   u = FpX_sub(gcoeff(M,1,1), FpX_mul(gcoeff(M,2,1), q, p), p);
     595         162 :   gel(res,1) = mkcol2(gcoeff(M,2,1), u);
     596         162 :   v = FpX_sub(gcoeff(M,1,2), FpX_mul(gcoeff(M,2,2), q, p), p);
     597         162 :   gel(res,2) = mkcol2(gcoeff(M,2,2), v);
     598         162 :   return res;
     599             : }
     600             : 
     601             : static GEN
     602           4 : matid2_FpXM(long v)
     603             : {
     604           4 :   retmkmat2(mkcol2(pol_1(v),pol_0(v)),
     605             :             mkcol2(pol_0(v),pol_1(v)));
     606             : }
     607             : 
     608             : static GEN
     609         162 : FpX_halfgcd_split(GEN x, GEN y, GEN p)
     610             : {
     611         162 :   pari_sp av=avma;
     612             :   GEN R, S, V;
     613             :   GEN y1, r, q;
     614         162 :   long l = lgpol(x), n = l>>1, k;
     615         162 :   if (lgpol(y)<=n) return matid2_FpXM(varn(x));
     616         162 :   R = FpX_halfgcd(RgX_shift_shallow(x,-n),RgX_shift_shallow(y,-n),p);
     617         162 :   V = FpXM_FpX_mul2(R,x,y,p); y1 = gel(V,2);
     618         162 :   if (lgpol(y1)<=n) return gerepilecopy(av, R);
     619         162 :   q = FpX_divrem(gel(V,1), y1, p, &r);
     620         162 :   k = 2*n-degpol(y1);
     621         162 :   S = FpX_halfgcd(RgX_shift_shallow(y1,-k), RgX_shift_shallow(r,-k),p);
     622         162 :   return gerepileupto(av, FpXM_mul2(S,FpX_FpXM_qmul(q,R,p),p));
     623             : }
     624             : 
     625             : /* Return M in GL_2(Fp[X]) such that:
     626             : if [a',b']~=M*[a,b]~ then degpol(a')>= (lgpol(a)>>1) >degpol(b')
     627             : */
     628             : 
     629             : static GEN
     630         349 : FpX_halfgcd_i(GEN x, GEN y, GEN p)
     631             : {
     632         349 :   if (lg(x)<=FpX_HALFGCD_LIMIT) return FpX_halfgcd_basecase(x,y,p);
     633         162 :   return FpX_halfgcd_split(x,y,p);
     634             : }
     635             : 
     636             : GEN
     637         455 : FpX_halfgcd(GEN x, GEN y, GEN p)
     638             : {
     639         455 :   pari_sp av = avma;
     640             :   GEN M,q,r;
     641         455 :   if (lgefint(p)==3)
     642             :   {
     643         106 :     ulong pp = to_Flx(&x, &y, p);
     644         106 :     M = FlxM_to_ZXM(Flx_halfgcd(x, y, pp));
     645             :   }
     646             :   else
     647             :   {
     648         349 :     if (!signe(x))
     649             :     {
     650           0 :       long v = varn(x);
     651           0 :       retmkmat2(mkcol2(pol_0(v),pol_1(v)),
     652             :                 mkcol2(pol_1(v),pol_0(v)));
     653             :     }
     654         349 :     if (degpol(y)<degpol(x)) return FpX_halfgcd_i(x,y,p);
     655          10 :     q = FpX_divrem(y,x,p,&r);
     656          10 :     M = FpX_halfgcd_i(x,r,p);
     657          10 :     gcoeff(M,1,1) = FpX_sub(gcoeff(M,1,1), FpX_mul(q, gcoeff(M,1,2), p), p);
     658          10 :     gcoeff(M,2,1) = FpX_sub(gcoeff(M,2,1), FpX_mul(q, gcoeff(M,2,2), p), p);
     659             :   }
     660         116 :   return gerepilecopy(av, M);
     661             : }
     662             : 
     663             : static GEN
     664       52052 : FpX_gcd_basecase(GEN a, GEN b, GEN p)
     665             : {
     666       52052 :   pari_sp av = avma, av0=avma;
     667      696846 :   while (signe(b))
     668             :   {
     669             :     GEN c;
     670      592805 :     if (gc_needed(av0,2))
     671             :     {
     672           0 :       if (DEBUGMEM>1) pari_warn(warnmem,"FpX_gcd (d = %ld)",degpol(b));
     673           0 :       gerepileall(av0,2, &a,&b);
     674             :     }
     675      592805 :     av = avma; c = FpX_rem(a,b,p); a=b; b=c;
     676             :   }
     677       51989 :   avma = av; return a;
     678             : }
     679             : 
     680             : GEN
     681      440308 : FpX_gcd(GEN x, GEN y, GEN p)
     682             : {
     683      440308 :   pari_sp av = avma;
     684      440308 :   if (lgefint(p)==3)
     685             :   {
     686             :     ulong pp;
     687      388075 :     (void)new_chunk((lg(x) + lg(y)) << 2); /* scratch space */
     688      388075 :     pp = to_Flx(&x, &y, p);
     689      388075 :     x = Flx_gcd(x, y, pp);
     690      388075 :     avma = av; return Flx_to_ZX(x);
     691             :   }
     692       52233 :   x = FpX_red(x, p);
     693       52233 :   y = FpX_red(y, p);
     694       52233 :   if (!signe(x)) return gerepileupto(av, y);
     695      104104 :   while (lg(y)>FpX_GCD_LIMIT)
     696             :   {
     697             :     GEN c;
     698           0 :     if (lgpol(y)<=(lgpol(x)>>1))
     699             :     {
     700           0 :       GEN r = FpX_rem(x, y, p);
     701           0 :       x = y; y = r;
     702             :     }
     703           0 :     c = FpXM_FpX_mul2(FpX_halfgcd(x,y, p), x, y, p);
     704           0 :     x = gel(c,1); y = gel(c,2);
     705           0 :     gerepileall(av,2,&x,&y);
     706             :   }
     707       52052 :   return gerepileupto(av, FpX_gcd_basecase(x,y,p));
     708             : }
     709             : 
     710             : /*Return 1 if gcd can be computed
     711             :  * else return a factor of p*/
     712             : 
     713             : GEN
     714           0 : FpX_gcd_check(GEN x, GEN y, GEN p)
     715             : {
     716             :   GEN a,b,c;
     717           0 :   pari_sp av=avma;
     718             : 
     719           0 :   a = FpX_red(x, p);
     720           0 :   b = FpX_red(y, p);
     721           0 :   while (signe(b))
     722             :   {
     723           0 :     GEN lead = leading_coeff(b);
     724           0 :     GEN g = gcdii(lead,p);
     725           0 :     if (!equali1(g)) return gerepileuptoint(av,g);
     726           0 :     c = FpX_rem(a,b,p); a=b; b=c;
     727             :   }
     728           0 :   avma = av; return gen_1;
     729             : }
     730             : 
     731             : static GEN
     732      244465 : FpX_extgcd_basecase(GEN a, GEN b, GEN p, GEN *ptu, GEN *ptv)
     733             : {
     734      244465 :   pari_sp av=avma;
     735             :   GEN u,v,d,d1,v1;
     736      244465 :   long vx = varn(a);
     737      244465 :   d = a; d1 = b;
     738      244465 :   v = pol_0(vx); v1 = pol_1(vx);
     739     1073074 :   while (signe(d1))
     740             :   {
     741      584144 :     GEN r, q = FpX_divrem(d,d1,p, &r);
     742      584144 :     v = FpX_sub(v,FpX_mul(q,v1,p),p);
     743      584144 :     u=v; v=v1; v1=u;
     744      584144 :     u=r; d=d1; d1=u;
     745      584144 :     if (gc_needed(av,2))
     746             :     {
     747           0 :       if (DEBUGMEM>1) pari_warn(warnmem,"FpX_extgcd (d = %ld)",degpol(d));
     748           0 :       gerepileall(av,5, &d,&d1,&u,&v,&v1);
     749             :     }
     750             :   }
     751      244465 :   if (ptu) *ptu = FpX_div(FpX_sub(d,FpX_mul(b,v,p),p),a,p);
     752      244465 :   *ptv = v; return d;
     753             : }
     754             : 
     755             : static GEN
     756           4 : FpX_extgcd_halfgcd(GEN x, GEN y, GEN p, GEN *ptu, GEN *ptv)
     757             : {
     758           4 :   pari_sp av=avma;
     759           4 :   GEN u,v,R = matid2_FpXM(varn(x));
     760          12 :   while (lg(y)>FpX_EXTGCD_LIMIT)
     761             :   {
     762             :     GEN M, c;
     763           4 :     if (lgpol(y)<=(lgpol(x)>>1))
     764             :     {
     765           0 :       GEN r, q = FpX_divrem(x, y, p, &r);
     766           0 :       x = y; y = r;
     767           0 :       R = FpX_FpXM_qmul(q, R, p);
     768             :     }
     769           4 :     M = FpX_halfgcd(x,y, p);
     770           4 :     c = FpXM_FpX_mul2(M, x,y, p);
     771           4 :     R = FpXM_mul2(M, R, p);
     772           4 :     x = gel(c,1); y = gel(c,2);
     773           4 :     gerepileall(av,3,&x,&y,&R);
     774             :   }
     775           4 :   y = FpX_extgcd_basecase(x,y,p,&u,&v);
     776           4 :   if (ptu) *ptu = FpX_addmulmul(u,v,gcoeff(R,1,1),gcoeff(R,2,1),p);
     777           4 :   *ptv = FpX_addmulmul(u,v,gcoeff(R,1,2),gcoeff(R,2,2),p);
     778           4 :   return y;
     779             : }
     780             : 
     781             : /* x and y in Z[X], return lift(gcd(x mod p, y mod p)). Set u and v st
     782             :  * ux + vy = gcd (mod p) */
     783             : GEN
     784      318449 : FpX_extgcd(GEN x, GEN y, GEN p, GEN *ptu, GEN *ptv)
     785             : {
     786             :   GEN d;
     787      318449 :   pari_sp ltop=avma;
     788      318449 :   if (lgefint(p)==3)
     789             :   {
     790       73984 :     ulong pp = to_Flx(&x, &y, p);
     791       73984 :     d = Flx_extgcd(x,y, pp, ptu,ptv);
     792       73984 :     d = Flx_to_ZX(d);
     793       73984 :     if (ptu) *ptu=Flx_to_ZX(*ptu);
     794       73984 :     *ptv=Flx_to_ZX(*ptv);
     795             :   }
     796             :   else
     797             :   {
     798      244465 :     x = FpX_red(x, p);
     799      244465 :     y = FpX_red(y, p);
     800      244465 :     if (lg(y)>FpX_EXTGCD_LIMIT)
     801           4 :       d = FpX_extgcd_halfgcd(x, y, p, ptu, ptv);
     802             :     else
     803      244461 :       d = FpX_extgcd_basecase(x, y, p, ptu, ptv);
     804             :   }
     805      318449 :   gerepileall(ltop,ptu?3:2,&d,ptv,ptu);
     806      318449 :   return d;
     807             : }
     808             : 
     809             : GEN
     810       13062 : FpX_rescale(GEN P, GEN h, GEN p)
     811             : {
     812       13062 :   long i, l = lg(P);
     813       13062 :   GEN Q = cgetg(l,t_POL), hi = h;
     814       13062 :   Q[l-1] = P[l-1];
     815       60417 :   for (i=l-2; i>=2; i--)
     816             :   {
     817       60417 :     gel(Q,i) = Fp_mul(gel(P,i), hi, p);
     818       60417 :     if (i == 2) break;
     819       47355 :     hi = Fp_mul(hi,h, p);
     820             :   }
     821       13062 :   Q[1] = P[1]; return Q;
     822             : }
     823             : 
     824             : GEN
     825      632960 : FpX_deriv(GEN x, GEN p) { return FpX_red(ZX_deriv(x), p); }
     826             : 
     827             : int
     828        1981 : FpX_is_squarefree(GEN f, GEN p)
     829             : {
     830        1981 :   pari_sp av = avma;
     831        1981 :   GEN z = FpX_gcd(f,FpX_deriv(f,p),p);
     832        1981 :   avma = av;
     833        1981 :   return degpol(z)==0;
     834             : }
     835             : 
     836             : GEN
     837       47092 : random_FpX(long d1, long v, GEN p)
     838             : {
     839       47092 :   long i, d = d1+2;
     840       47092 :   GEN y = cgetg(d,t_POL); y[1] = evalsigne(1) | evalvarn(v);
     841       47092 :   for (i=2; i<d; i++) gel(y,i) = randomi(p);
     842       47092 :   return FpX_renormalize(y,d);
     843             : }
     844             : 
     845             : GEN
     846         566 : FpX_dotproduct(GEN x, GEN y, GEN p)
     847             : {
     848         566 :   long i, l = minss(lg(x), lg(y));
     849             :   pari_sp av;
     850             :   GEN c;
     851         566 :   if (l == 2) return gen_0;
     852         566 :   av = avma; c = mulii(gel(x,2),gel(y,2));
     853         566 :   for (i=3; i<l; i++) c = addii(c, mulii(gel(x,i),gel(y,i)));
     854         566 :   return gerepileuptoint(av, modii(c,p));
     855             : }
     856             : 
     857             : /* Evaluation in Fp
     858             :  * x a ZX and y an Fp, return x(y) mod p
     859             :  *
     860             :  * If p is very large (several longs) and x has small coefficients(<<p),
     861             :  * then Brent & Kung algorithm is faster. */
     862             : GEN
     863      522720 : FpX_eval(GEN x,GEN y,GEN p)
     864             : {
     865             :   pari_sp av;
     866             :   GEN p1,r,res;
     867      522720 :   long j, i=lg(x)-1;
     868      522720 :   if (i<=2)
     869        6544 :     return (i==2)? modii(gel(x,2),p): gen_0;
     870      516176 :   res=cgeti(lgefint(p));
     871      516176 :   av=avma; p1=gel(x,i);
     872             :   /* specific attention to sparse polynomials (see poleval)*/
     873             :   /*You've guessed it! It's a copy-paste(tm)*/
     874     2051024 :   for (i--; i>=2; i=j-1)
     875             :   {
     876     2067432 :     for (j=i; !signe(gel(x,j)); j--)
     877      532584 :       if (j==2)
     878             :       {
     879       57235 :         if (i!=j) y = Fp_powu(y,i-j+1,p);
     880       57235 :         p1=mulii(p1,y);
     881       57235 :         goto fppoleval;/*sorry break(2) no implemented*/
     882             :       }
     883     1534848 :     r = (i==j)? y: Fp_powu(y,i-j+1,p);
     884     1534848 :     p1 = Fp_addmul(gel(x,j), p1, r, p);
     885     1534848 :     if ((i & 7) == 0) { affii(p1, res); p1 = res; avma = av; }
     886             :   }
     887             :  fppoleval:
     888      516176 :   modiiz(p1,p,res);
     889      516176 :   avma = av; return res;
     890             : }
     891             : 
     892             : /* Tz=Tx*Ty where Tx and Ty coprime
     893             :  * return lift(chinese(Mod(x*Mod(1,p),Tx*Mod(1,p)),Mod(y*Mod(1,p),Ty*Mod(1,p))))
     894             :  * if Tz is NULL it is computed
     895             :  * As we do not return it, and the caller will frequently need it,
     896             :  * it must compute it and pass it.
     897             :  */
     898             : GEN
     899         665 : FpX_chinese_coprime(GEN x,GEN y,GEN Tx,GEN Ty,GEN Tz,GEN p)
     900             : {
     901         665 :   pari_sp av = avma;
     902             :   GEN ax,p1;
     903         665 :   ax = FpX_mul(FpXQ_inv(Tx,Ty,p), Tx,p);
     904         665 :   p1 = FpX_mul(ax, FpX_sub(y,x,p),p);
     905         665 :   p1 = FpX_add(x,p1,p);
     906         665 :   if (!Tz) Tz=FpX_mul(Tx,Ty,p);
     907         665 :   p1 = FpX_rem(p1,Tz,p);
     908         665 :   return gerepileupto(av,p1);
     909             : }
     910             : 
     911             : /* Res(A,B) = Res(B,R) * lc(B)^(a-r) * (-1)^(ab), with R=A%B, a=deg(A) ...*/
     912             : GEN
     913        6152 : FpX_resultant(GEN a, GEN b, GEN p)
     914             : {
     915             :   long da,db,dc;
     916             :   pari_sp av;
     917        6152 :   GEN c,lb, res = gen_1;
     918             : 
     919        6152 :   if (!signe(a) || !signe(b)) return gen_0;
     920        6152 :   da = degpol(a);
     921        6152 :   db = degpol(b);
     922        6152 :   if (db > da)
     923             :   {
     924           0 :     swapspec(a,b, da,db);
     925           0 :     if (both_odd(da,db)) res = subii(p, res);
     926             :   }
     927        6152 :   if (!da) return gen_1; /* = res * a[2] ^ db, since 0 <= db <= da = 0 */
     928        6152 :   av = avma;
     929       34867 :   while (db)
     930             :   {
     931       22570 :     lb = gel(b,db+2);
     932       22570 :     c = FpX_rem(a,b, p);
     933       22570 :     a = b; b = c; dc = degpol(c);
     934       22570 :     if (dc < 0) { avma = av; return NULL; }
     935             : 
     936       22563 :     if (both_odd(da,db)) res = subii(p, res);
     937       22563 :     if (!equali1(lb)) res = Fp_mul(res, Fp_powu(lb, da - dc, p), p);
     938       22563 :     if (gc_needed(av,2))
     939             :     {
     940           0 :       if (DEBUGMEM>1) pari_warn(warnmem,"FpX_resultant (da = %ld)",da);
     941           0 :       gerepileall(av,3, &a,&b,&res);
     942             :     }
     943       22563 :     da = db; /* = degpol(a) */
     944       22563 :     db = dc; /* = degpol(b) */
     945             :   }
     946        6145 :   res = Fp_mul(res, Fp_powu(gel(b,2), da, p), p);
     947        6145 :   return gerepileuptoint(av, res);
     948             : }
     949             : 
     950             : /* disc P = (-1)^(n(n-1)/2) lc(P)^(n - deg P' - 2) Res(P,P'), n = deg P */
     951             : GEN
     952          28 : FpX_disc(GEN P, GEN p)
     953             : {
     954          28 :   pari_sp av = avma;
     955          28 :   GEN L, dP = FpX_deriv(P,p), D = FpX_resultant(P, dP, p);
     956             :   long dd;
     957          28 :   if (!D || !signe(D)) return gen_0;
     958          21 :   dd = degpol(P) - 2 - degpol(dP); /* >= -1; > -1 iff p | deg(P) */
     959          21 :   L = leading_coeff(P);
     960          21 :   if (dd && !equali1(L))
     961           7 :     D = (dd == -1)? Fp_div(D,L,p): Fp_mul(D, Fp_powu(L, dd, p), p);
     962          21 :   if (degpol(P) & 2) D = Fp_neg(D ,p);
     963          21 :   return gerepileuptoint(av, D);
     964             : }
     965             : 
     966             : GEN
     967       19505 : FpXV_prod(GEN V, GEN p)
     968             : {
     969             :   struct _FpXQ D;
     970       19505 :   D.p = p;
     971       19505 :   return gen_product(V, (void *)&D, &_FpX_mul);
     972             : }
     973             : 
     974             : GEN
     975        5138 : FpV_roots_to_pol(GEN V, GEN p, long v)
     976             : {
     977        5138 :   pari_sp ltop=avma;
     978             :   long i;
     979        5138 :   GEN g=cgetg(lg(V),t_VEC);
     980       39760 :   for(i=1;i<lg(V);i++)
     981       34622 :     gel(g,i) = deg1pol_shallow(gen_1,modii(negi(gel(V,i)),p),v);
     982        5138 :   return gerepileupto(ltop,FpXV_prod(g,p));
     983             : }
     984             : 
     985             : /* invert all elements of x mod p using Montgomery's multi-inverse trick.
     986             :  * Not stack-clean. */
     987             : GEN
     988        4069 : FpV_inv(GEN x, GEN p)
     989             : {
     990        4069 :   long i, lx = lg(x);
     991        4069 :   GEN u, y = cgetg(lx, t_VEC);
     992             : 
     993        4069 :   gel(y,1) = gel(x,1);
     994        4069 :   for (i=2; i<lx; i++) gel(y,i) = Fp_mul(gel(y,i-1), gel(x,i), p);
     995             : 
     996        4069 :   u = Fp_inv(gel(y,--i), p);
     997      327749 :   for ( ; i > 1; i--)
     998             :   {
     999      323680 :     gel(y,i) = Fp_mul(u, gel(y,i-1), p);
    1000      323680 :     u = Fp_mul(u, gel(x,i), p); /* u = 1 / (x[1] ... x[i-1]) */
    1001             :   }
    1002        4069 :   gel(y,1) = u; return y;
    1003             : }
    1004             : GEN
    1005           0 : FqV_inv(GEN x, GEN T, GEN p)
    1006             : {
    1007           0 :   long i, lx = lg(x);
    1008           0 :   GEN u, y = cgetg(lx, t_VEC);
    1009             : 
    1010           0 :   gel(y,1) = gel(x,1);
    1011           0 :   for (i=2; i<lx; i++) gel(y,i) = Fq_mul(gel(y,i-1), gel(x,i), T,p);
    1012             : 
    1013           0 :   u = Fq_inv(gel(y,--i), T,p);
    1014           0 :   for ( ; i > 1; i--)
    1015             :   {
    1016           0 :     gel(y,i) = Fq_mul(u, gel(y,i-1), T,p);
    1017           0 :     u = Fq_mul(u, gel(x,i), T,p); /* u = 1 / (x[1] ... x[i-1]) */
    1018             :   }
    1019           0 :   gel(y,1) = u; return y;
    1020             : }
    1021             : 
    1022             : /***********************************************************************/
    1023             : /**                                                                   **/
    1024             : /**                      Barrett reduction                            **/
    1025             : /**                                                                   **/
    1026             : /***********************************************************************/
    1027             : 
    1028             : static GEN
    1029        1421 : FpX_invBarrett_basecase(GEN T, GEN p)
    1030             : {
    1031        1421 :   long i, l=lg(T)-1, lr = l-1, k;
    1032        1421 :   GEN r=cgetg(lr, t_POL); r[1]=T[1];
    1033        1421 :   gel(r,2) = gen_1;
    1034       82733 :   for (i=3; i<lr; i++)
    1035             :   {
    1036       81312 :     pari_sp av = avma;
    1037       81312 :     GEN u = gel(T,l-i+2);
    1038     2636787 :     for (k=3; k<i; k++)
    1039     2555475 :       u = addii(u, mulii(gel(T,l-i+k), gel(r,k)));
    1040       81312 :     gel(r,i) = gerepileupto(av, modii(negi(u), p));
    1041             :   }
    1042        1421 :   return FpX_renormalize(r,lr);
    1043             : }
    1044             : 
    1045             : /* Return new lgpol */
    1046             : static long
    1047      208306 : ZX_lgrenormalizespec(GEN x, long lx)
    1048             : {
    1049             :   long i;
    1050      255223 :   for (i = lx-1; i>=0; i--)
    1051      255220 :     if (signe(gel(x,i))) break;
    1052      208306 :   return i+1;
    1053             : }
    1054             : 
    1055             : INLINE GEN
    1056      197577 : FpX_recipspec(GEN x, long l, long n)
    1057             : {
    1058      197577 :   return RgX_recipspec_shallow(x, l, n);
    1059             : }
    1060             : 
    1061             : static GEN
    1062         558 : FpX_invBarrett_Newton(GEN T, GEN p)
    1063             : {
    1064         558 :   pari_sp av = avma;
    1065         558 :   long nold, lx, lz, lq, l = degpol(T), i, lQ;
    1066         558 :   GEN q, y, z, x = cgetg(l+2, t_POL) + 2;
    1067         557 :   ulong mask = quadratic_prec_mask(l-2); /* assume l > 2 */
    1068         558 :   for (i=0;i<l;i++) gel(x,i) = gen_0;
    1069         558 :   q = FpX_recipspec(T+2,l+1,l+1); lQ = lgpol(q); q+=2;
    1070             :   /* We work on _spec_ FpX's, all the l[xzq] below are lgpol's */
    1071             : 
    1072             :   /* initialize */
    1073         558 :   gel(x,0) = Fp_inv(gel(q,0), p);
    1074         558 :   if (lQ>1) gel(q,1) = Fp_red(gel(q,1), p);
    1075         558 :   if (lQ>1 && signe(gel(q,1)))
    1076         543 :   {
    1077         507 :     GEN u = gel(q, 1);
    1078         507 :     if (!equali1(gel(x,0))) u = Fp_mul(u, Fp_sqr(gel(x,0), p), p);
    1079         507 :     gel(x,1) = Fp_neg(u, p); lx = 2;
    1080             :   }
    1081             :   else
    1082          51 :     lx = 1;
    1083         594 :   nold = 1;
    1084        5125 :   for (; mask > 1; )
    1085             :   { /* set x -= x(x*q - 1) + O(t^(nnew + 1)), knowing x*q = 1 + O(t^(nold+1)) */
    1086        3974 :     long i, lnew, nnew = nold << 1;
    1087             : 
    1088        3974 :     if (mask & 1) nnew--;
    1089        3974 :     mask >>= 1;
    1090             : 
    1091        3974 :     lnew = nnew + 1;
    1092        3974 :     lq = ZX_lgrenormalizespec(q, minss(lQ,lnew));
    1093        3979 :     z = FpX_mulspec(x, q, p, lx, lq); /* FIXME: high product */
    1094        3979 :     lz = lgpol(z); if (lz > lnew) lz = lnew;
    1095        3980 :     z += 2;
    1096             :     /* subtract 1 [=>first nold words are 0]: renormalize so that z(0) != 0 */
    1097        3980 :     for (i = nold; i < lz; i++) if (signe(gel(z,i))) break;
    1098        3980 :     nold = nnew;
    1099        3980 :     if (i >= lz) continue; /* z-1 = 0(t^(nnew + 1)) */
    1100             : 
    1101             :     /* z + i represents (x*q - 1) / t^i */
    1102        3725 :     lz = ZX_lgrenormalizespec (z+i, lz-i);
    1103        3725 :     z = FpX_mulspec(x, z+i, p, lx, lz); /* FIXME: low product */
    1104        3723 :     lz = lgpol(z); z += 2;
    1105        3722 :     if (lz > lnew-i) lz = ZX_lgrenormalizespec(z, lnew-i);
    1106             : 
    1107        3724 :     lx = lz+ i;
    1108        3724 :     y  = x + i; /* x -= z * t^i, in place */
    1109        3724 :     for (i = 0; i < lz; i++) gel(y,i) = Fp_neg(gel(z,i), p);
    1110             :   }
    1111         557 :   x -= 2; setlg(x, lx + 2); x[1] = T[1];
    1112         557 :   return gerepilecopy(av, x);
    1113             : }
    1114             : 
    1115             : /* 1/polrecip(T)+O(x^(deg(T)-1)) */
    1116             : GEN
    1117        1983 : FpX_invBarrett(GEN T, GEN p)
    1118             : {
    1119        1983 :   pari_sp ltop = avma;
    1120        1983 :   long l = lg(T);
    1121             :   GEN r;
    1122        1983 :   if (l<5) return pol_0(varn(T));
    1123        1979 :   if (l<=FpX_INVBARRETT_LIMIT)
    1124             :   {
    1125        1421 :     GEN c = gel(T,l-1), ci=gen_1;
    1126        1421 :     if (!equali1(c))
    1127             :     {
    1128           0 :       ci = Fp_inv(c, p);
    1129           0 :       T = FpX_Fp_mul(T, ci, p);
    1130           0 :       r = FpX_invBarrett_basecase(T, p);
    1131           0 :       r = FpX_Fp_mul(r, ci, p);
    1132             :     } else
    1133        1421 :       r = FpX_invBarrett_basecase(T, p);
    1134             :   }
    1135             :   else
    1136         558 :     r = FpX_invBarrett_Newton(T, p);
    1137        1979 :   return gerepileupto(ltop, r);
    1138             : }
    1139             : 
    1140             : GEN
    1141     1121213 : FpX_get_red(GEN T, GEN p)
    1142             : {
    1143     1121213 :   if (typ(T)==t_POL && lg(T)>FpX_BARRETT_LIMIT)
    1144        1689 :     retmkvec2(FpX_invBarrett(T,p),T);
    1145     1119524 :   return T;
    1146             : }
    1147             : 
    1148             : /* Compute x mod T where 2 <= degpol(T) <= l+1 <= 2*(degpol(T)-1)
    1149             :  * and mg is the Barrett inverse of T. */
    1150             : static GEN
    1151       98465 : FpX_divrem_Barrettspec(GEN x, long l, GEN mg, GEN T, GEN p, GEN *pr)
    1152             : {
    1153             :   GEN q, r;
    1154       98465 :   long lt = degpol(T); /*We discard the leading term*/
    1155             :   long ld, lm, lT, lmg;
    1156       98465 :   ld = l-lt;
    1157       98465 :   lm = minss(ld, lgpol(mg));
    1158       98465 :   lT  = ZX_lgrenormalizespec(T+2,lt);
    1159       98465 :   lmg = ZX_lgrenormalizespec(mg+2,lm);
    1160       98465 :   q = FpX_recipspec(x+lt,ld,ld);              /* q = rec(x)     lq<=ld*/
    1161       98465 :   q = FpX_mulspec(q+2,mg+2,p,lgpol(q),lmg);    /* q = rec(x) * mg lq<=ld+lm*/
    1162       98465 :   q = FpX_recipspec(q+2,minss(ld,lgpol(q)),ld);/* q = rec (rec(x) * mg) lq<=ld*/
    1163       98464 :   if (!pr) return q;
    1164       98464 :   r = FpX_mulspec(q+2,T+2,p,lgpol(q),lT);      /* r = q*pol        lr<=ld+lt*/
    1165       98465 :   r = FpX_subspec(x,r+2,p,lt,minss(lt,lgpol(r)));/* r = x - r   lr<=lt */
    1166       98463 :   if (pr == ONLY_REM) return r;
    1167         590 :   *pr = r; return q;
    1168             : }
    1169             : 
    1170             : static GEN
    1171       98039 : FpX_divrem_Barrett_noGC(GEN x, GEN mg, GEN T, GEN p, GEN *pr)
    1172             : {
    1173       98039 :   GEN q = NULL, r = FpX_red(x, p);
    1174       98041 :   long l = lgpol(r), lt = degpol(T), lm = 2*lt-1;
    1175             :   long i;
    1176       98041 :   if (l <= lt)
    1177             :   {
    1178           0 :     if (pr == ONLY_REM) return r;
    1179           0 :     if (pr == ONLY_DIVIDES) return signe(x)? NULL: pol_0(varn(x));
    1180           0 :     if (pr) *pr = r;
    1181           0 :     return pol_0(varn(T));
    1182             :   }
    1183       98041 :   if (lt <= 1)
    1184           4 :     return FpX_divrem_basecase(r,T,p,pr);
    1185       98037 :   if (pr != ONLY_REM && l>lm)
    1186             :   {
    1187         128 :     q = cgetg(l-lt+2, t_POL);
    1188         128 :     for (i=0;i<l-lt;i++) gel(q+2,i) = gen_0;
    1189             :   }
    1190      196502 :   while (l>lm)
    1191             :   {
    1192         428 :     GEN zr, zq = FpX_divrem_Barrettspec(r+2+l-lm,lm,mg,T,p,&zr);
    1193         428 :     long lz = lgpol(zr);
    1194         428 :     if (pr != ONLY_REM)
    1195             :     {
    1196         268 :       long lq = lgpol(zq);
    1197         268 :       for(i=0; i<lq; i++) gel(q+2+l-lm,i) = gel(zq,2+i);
    1198             :     }
    1199         428 :     for(i=0; i<lz; i++) gel(r+2+l-lm,i) = gel(zr,2+i);
    1200         428 :     l = l-lm+lz;
    1201             :   }
    1202       98037 :   if (pr != ONLY_REM)
    1203             :   {
    1204         162 :     if (l > lt)
    1205             :     {
    1206         162 :       GEN zq = FpX_divrem_Barrettspec(r+2,l,mg,T,p,&r);
    1207         162 :       if (!q) q = zq;
    1208             :       else
    1209             :       {
    1210         128 :         long lq = lgpol(zq);
    1211         128 :         for(i=0; i<lq; i++) gel(q+2,i) = gel(zq,2+i);
    1212             :       }
    1213             :     }
    1214             :     else
    1215           0 :       r = FpX_renormalize(r, l+2);
    1216             :   }
    1217             :   else
    1218             :   {
    1219       97875 :     if (l > lt)
    1220       97875 :       r = FpX_divrem_Barrettspec(r+2, l, mg, T, p, ONLY_REM);
    1221             :     else
    1222           0 :       r = FpX_renormalize(r, l+2);
    1223       97873 :     r[1] = x[1]; return FpX_renormalize(r, lg(r));
    1224             :   }
    1225         162 :   if (pr) { r[1] = x[1]; r = FpX_renormalize(r, lg(r)); }
    1226         162 :   q[1] = x[1]; q = FpX_renormalize(q, lg(q));
    1227         162 :   if (pr == ONLY_DIVIDES) return signe(r)? NULL: q;
    1228         162 :   if (pr) *pr = r;
    1229         162 :   return q;
    1230             : }
    1231             : 
    1232             : GEN
    1233     3359519 : FpX_divrem(GEN x, GEN T, GEN p, GEN *pr)
    1234             : {
    1235     3359519 :   GEN B, y = get_FpX_red(T, &B);
    1236     3359519 :   long dy = degpol(y), dx = degpol(x), d = dx-dy;
    1237     3359519 :   if (pr==ONLY_REM) return FpX_rem(x, y, p);
    1238     3359519 :   if (!B && d+3 < FpX_DIVREM_BARRETT_LIMIT)
    1239     3358650 :     return FpX_divrem_basecase(x,y,p,pr);
    1240         869 :   else if (lgefint(p)==3)
    1241             :   {
    1242         707 :     pari_sp av = avma;
    1243         707 :     ulong pp = to_Flxq(&x, &T, p);
    1244         707 :     GEN z = Flx_divrem(x, T, pp, pr);
    1245         707 :     if (!z) return NULL;
    1246         707 :     if (!pr || pr == ONLY_DIVIDES)
    1247           0 :       return Flx_to_ZX_inplace(gerepileuptoleaf(av, z));
    1248         707 :     z = Flx_to_ZX(z);
    1249         707 :     *pr = Flx_to_ZX(*pr);
    1250         707 :     gerepileall(av, 2, &z, pr);
    1251         707 :     return z;
    1252             :   } else
    1253             :   {
    1254         162 :     pari_sp av=avma;
    1255         162 :     GEN mg = B? B: FpX_invBarrett(y, p);
    1256         162 :     GEN q1 = FpX_divrem_Barrett_noGC(x,mg,y,p,pr);
    1257         162 :     if (!q1) {avma=av; return NULL;}
    1258         162 :     if (!pr || pr==ONLY_DIVIDES) return gerepilecopy(av, q1);
    1259         162 :     gerepileall(av,2,&q1,pr);
    1260         162 :     return q1;
    1261             :   }
    1262             : }
    1263             : 
    1264             : GEN
    1265    64500938 : FpX_rem(GEN x, GEN T, GEN p)
    1266             : {
    1267    64500938 :   GEN B, y = get_FpX_red(T, &B);
    1268    64500939 :   long dy = degpol(y), dx = degpol(x), d = dx-dy;
    1269    64500940 :   if (d < 0) return FpX_red(x,p);
    1270    60685811 :   if (!B && d+3 < FpX_REM_BARRETT_LIMIT)
    1271    60577016 :     return FpX_divrem_basecase(x,y,p,ONLY_REM);
    1272      108795 :   else if (lgefint(p)==3)
    1273             :   {
    1274       10918 :     pari_sp av = avma;
    1275       10918 :     ulong pp = to_Flxq(&x, &T, p);
    1276       10918 :     return Flx_to_ZX_inplace(gerepileuptoleaf(av, Flx_rem(x, T, pp)));
    1277             :   } else
    1278             :   {
    1279       97877 :     pari_sp av = avma;
    1280       97877 :     GEN mg = B? B: FpX_invBarrett(y, p);
    1281       97877 :     return gerepileupto(av, FpX_divrem_Barrett_noGC(x, mg, y, p, ONLY_REM));
    1282             :   }
    1283             : }
    1284             : 
    1285             : static GEN
    1286        9207 : deg2pol_shallow(GEN x2, GEN x1, GEN x0, long v)
    1287             : {
    1288        9207 :   GEN x = cgetg(5,t_POL);
    1289        9207 :   x[1] = evalsigne(1) | evalvarn(v);
    1290        9207 :   gel(x,2) = x0;
    1291        9207 :   gel(x,3) = x1;
    1292        9207 :   gel(x,4) = x2;
    1293        9207 :   return normalizepol_lg(x,5);
    1294             : }
    1295             : 
    1296             : static GEN
    1297        1646 : FpV_producttree(GEN xa, GEN p, long vs)
    1298             : {
    1299        1646 :   long n = lg(xa)-1;
    1300        1646 :   long m = n==1 ? 1: expu(n-1)+1;
    1301        1646 :   GEN T = cgetg(m+1, t_VEC), t;
    1302             :   long i, j, k;
    1303        1646 :   t = cgetg(((n+1)>>1)+1, t_VEC);
    1304       10853 :   for (j=1, k=1; k<n; j++, k+=2)
    1305       46035 :     gel(t, j) = deg2pol_shallow(gen_1,
    1306       18414 :                 Fp_neg(Fp_add(gel(xa,k), gel(xa,k+1), p), p),
    1307       18414 :                 Fp_mul(gel(xa,k), gel(xa,k+1), p), vs);
    1308        1646 :   if (k==n) gel(t, j) = deg1pol(gen_1, Fp_neg(gel(xa,k), p), vs);
    1309        1646 :   gel(T,1) = t;
    1310        4636 :   for (i=2; i<=m; i++)
    1311             :   {
    1312        2990 :     GEN u = gel(T, i-1);
    1313        2990 :     long n = lg(u)-1;
    1314        2990 :     t = cgetg(((n+1)>>1)+1, t_VEC);
    1315       10838 :     for (j=1, k=1; k<n; j++, k+=2)
    1316        7848 :       gel(t, j) = FpX_mul(gel(u, k), gel(u, k+1), p);
    1317        2990 :     if (k==n) gel(t, j) = gel(u, k);
    1318        2990 :     gel(T, i) = t;
    1319             :   }
    1320        1646 :   return T;
    1321             : }
    1322             : 
    1323             : static GEN
    1324        1646 : FpX_FpV_multieval_tree(GEN P, GEN xa, GEN T, GEN p)
    1325             : {
    1326        1646 :   pari_sp av = avma;
    1327             :   long i,j,k;
    1328        1646 :   long m = lg(T)-1, n = lg(xa)-1;
    1329             :   GEN t;
    1330        1646 :   GEN Tp = cgetg(m+1, t_VEC);
    1331        1646 :   gel(Tp, m) = mkvec(P);
    1332        4636 :   for (i=m-1; i>=1; i--)
    1333             :   {
    1334        2990 :     GEN u = gel(T, i);
    1335        2990 :     GEN v = gel(Tp, i+1);
    1336        2990 :     long n = lg(u)-1;
    1337        2990 :     t = cgetg(n+1, t_VEC);
    1338       10838 :     for (j=1, k=1; k<n; j++, k+=2)
    1339             :     {
    1340        7848 :       gel(t, k)   = FpX_rem(gel(v, j), gel(u, k), p);
    1341        7848 :       gel(t, k+1) = FpX_rem(gel(v, j), gel(u, k+1), p);
    1342             :     }
    1343        2990 :     if (k==n) gel(t, k) = gel(v, j);
    1344        2990 :     gel(Tp, i) = t;
    1345             :   }
    1346             :   {
    1347        1646 :     GEN R = cgetg(n+1, t_VEC);
    1348        1646 :     GEN u = gel(T, i+1);
    1349        1646 :     GEN v = gel(Tp, i+1);
    1350        1646 :     long l = lg(u)-1;
    1351       11140 :     for (j=1, k=1; j<=l; j++)
    1352             :     {
    1353        9494 :       long c, d = degpol(gel(u,j));
    1354       28195 :       for (c=1; c<=d; c++, k++)
    1355       18701 :         gel(R,k) = FpX_eval(gel(v, j), gel(xa,k), p);
    1356             :     }
    1357        1646 :     return gerepileupto(av, R);
    1358             :   }
    1359             : }
    1360             : 
    1361             : static GEN
    1362           1 : FpVV_polint_tree(GEN T, GEN R, GEN xa, GEN ya, GEN p, long vs)
    1363             : {
    1364           1 :   pari_sp av = avma;
    1365           1 :   long m = lg(T)-1, n = lg(ya)-1;
    1366             :   long i,j,k;
    1367           1 :   GEN Tp = cgetg(m+1, t_VEC);
    1368           1 :   GEN t = cgetg(lg(gel(T,1)), t_VEC);
    1369           3 :   for (j=1, k=1; k<n; j++, k+=2)
    1370             :   {
    1371           2 :     GEN a = Fp_mul(gel(ya,k), gel(R,k), p), b = Fp_mul(gel(ya,k+1), gel(R,k+1), p);
    1372           6 :     gel(t, j) = deg1pol(Fp_add(a, b, p),
    1373           2 :                 Fp_neg(Fp_add(Fp_mul(gel(xa,k), b, p ),
    1374           2 :                               Fp_mul(gel(xa,k+1), a, p), p), p), vs);
    1375             :   }
    1376           1 :   if (k==n) gel(t, j) = scalarpol(Fp_mul(gel(ya,k), gel(R,k), p), vs);
    1377           1 :   gel(Tp, 1) = t;
    1378           2 :   for (i=2; i<=m; i++)
    1379             :   {
    1380           1 :     GEN u = gel(T, i-1);
    1381           1 :     GEN t = cgetg(lg(gel(T,i)), t_VEC);
    1382           1 :     GEN v = gel(Tp, i-1);
    1383           1 :     long l = lg(v)-1;
    1384           2 :     for (j=1, k=1; k<l; j++, k+=2)
    1385           3 :       gel(t, j) = FpX_add(ZX_mul(gel(u, k), gel(v, k+1)),
    1386           2 :                           ZX_mul(gel(u, k+1), gel(v, k)), p);
    1387           1 :     if (k==l) gel(t, j) = gel(v, k);
    1388           1 :     gel(Tp, i) = t;
    1389             :   }
    1390           1 :   return gerepilecopy(av, gmael(Tp,m,1));
    1391             : }
    1392             : 
    1393             : GEN
    1394           0 : FpX_FpV_multieval(GEN P, GEN xa, GEN p)
    1395             : {
    1396           0 :   pari_sp av = avma;
    1397           0 :   GEN T = FpV_producttree(xa, p, P[1]);
    1398           0 :   return gerepileupto(av, FpX_FpV_multieval_tree(P, xa, T, p));
    1399             : }
    1400             : 
    1401             : GEN
    1402           1 : FpV_polint(GEN xa, GEN ya, GEN p, long vs)
    1403             : {
    1404           1 :   pari_sp av = avma;
    1405           1 :   GEN T = FpV_producttree(xa, p, vs);
    1406           1 :   long m = lg(T)-1;
    1407           1 :   GEN P = FpX_deriv(gmael(T, m, 1), p);
    1408           1 :   GEN R = FpV_inv(FpX_FpV_multieval_tree(P, xa, T, p), p);
    1409           1 :   return gerepileupto(av, FpVV_polint_tree(T, R, xa, ya, p, vs));
    1410             : }
    1411             : 
    1412             : GEN
    1413           0 : FpV_FpM_polint(GEN xa, GEN ya, GEN p, long vs)
    1414             : {
    1415           0 :   pari_sp av = avma;
    1416           0 :   GEN T = FpV_producttree(xa, p, vs);
    1417           0 :   long m = lg(T)-1, l = lg(ya)-1;
    1418             :   long i;
    1419           0 :   GEN P = FpX_deriv(gmael(T, m, 1), p);
    1420           0 :   GEN R = FpV_inv(FpX_FpV_multieval_tree(P, xa, T, p), p);
    1421           0 :   GEN M = cgetg(l+1, t_VEC);
    1422           0 :   for (i=1; i<=l; i++)
    1423           0 :     gel(M,i) = FpVV_polint_tree(T, R, xa, gel(ya,i), p, vs);
    1424           0 :   return gerepileupto(av, M);
    1425             : }
    1426             : 
    1427             : GEN
    1428        1645 : FpV_invVandermonde(GEN L, GEN den, GEN p)
    1429             : {
    1430        1645 :   pari_sp av = avma;
    1431        1645 :   long i, n = lg(L);
    1432             :   GEN M, R;
    1433        1645 :   GEN tree = FpV_producttree(L, p, 0);
    1434        1645 :   long m = lg(tree)-1;
    1435        1645 :   GEN T = gmael(tree, m, 1);
    1436        1645 :   R = FpV_inv(FpX_FpV_multieval_tree(FpX_deriv(T, p), L, tree, p), p);
    1437        1645 :   if (den) R = FpC_Fp_mul(R, den, p);
    1438        1645 :   M = cgetg(n, t_MAT);
    1439       20342 :   for (i = 1; i < n; i++)
    1440             :   {
    1441       18697 :     GEN P = FpX_Fp_mul(FpX_div_by_X_x(T, gel(L,i), p, NULL), gel(R,i), p);
    1442       18697 :     gel(M,i) = RgX_to_RgC(P, n-1);
    1443             :   }
    1444        1645 :   return gerepilecopy(av, M);
    1445             : }
    1446             : 
    1447             : /***********************************************************************/
    1448             : /**                                                                   **/
    1449             : /**                              FpXQ                                 **/
    1450             : /**                                                                   **/
    1451             : /***********************************************************************/
    1452             : 
    1453             : /* FpXQ are elements of Fp[X]/(T), represented by FpX*/
    1454             : 
    1455             : GEN
    1456     3669432 : FpXQ_red(GEN x, GEN T, GEN p)
    1457             : {
    1458     3669432 :   GEN z = FpX_red(x,p);
    1459     3669432 :   return FpX_rem(z, T,p);
    1460             : }
    1461             : 
    1462             : GEN
    1463    41423458 : FpXQ_mul(GEN x,GEN y,GEN T,GEN p)
    1464             : {
    1465    41423458 :   GEN z = FpX_mul(x,y,p);
    1466    41423458 :   return FpX_rem(z, T, p);
    1467             : }
    1468             : 
    1469             : GEN
    1470     3769169 : FpXQ_sqr(GEN x, GEN T, GEN p)
    1471             : {
    1472     3769169 :   GEN z = FpX_sqr(x,p);
    1473     3769168 :   return FpX_rem(z, T, p);
    1474             : }
    1475             : 
    1476             : /* Inverse of x in Z/pZ[X]/(pol) or NULL if inverse doesn't exist
    1477             :  * return lift(1 / (x mod (p,pol))) */
    1478             : GEN
    1479      285309 : FpXQ_invsafe(GEN x, GEN y, GEN p)
    1480             : {
    1481      285309 :   GEN V, z = FpX_extgcd(get_FpX_mod(y), x, p, NULL, &V);
    1482      285309 :   if (degpol(z)) return NULL;
    1483      285309 :   z = Fp_invsafe(gel(z,2), p);
    1484      285309 :   if (!z) return NULL;
    1485      285309 :   return FpX_Fp_mul(V, z, p);
    1486             : }
    1487             : 
    1488             : GEN
    1489      285295 : FpXQ_inv(GEN x,GEN T,GEN p)
    1490             : {
    1491      285295 :   pari_sp av = avma;
    1492      285295 :   GEN U = FpXQ_invsafe(x, T, p);
    1493      285295 :   if (!U) pari_err_INV("FpXQ_inv",x);
    1494      285295 :   return gerepileupto(av, U);
    1495             : }
    1496             : 
    1497             : GEN
    1498      226448 : FpXQ_div(GEN x,GEN y,GEN T,GEN p)
    1499             : {
    1500      226448 :   pari_sp av = avma;
    1501      226448 :   return gerepileupto(av, FpXQ_mul(x,FpXQ_inv(y,T,p),T,p));
    1502             : }
    1503             : 
    1504             : static GEN
    1505     1024942 : _FpXQ_add(void *data, GEN x, GEN y)
    1506             : {
    1507             :   (void) data;
    1508     1024942 :   return ZX_add(x, y);
    1509             : }
    1510             : static GEN
    1511       57554 : _FpXQ_sub(void *data, GEN x, GEN y)
    1512             : {
    1513             :   (void) data;
    1514       57554 :   return ZX_sub(x, y);
    1515             : }
    1516             : static GEN
    1517     1152100 : _FpXQ_cmul(void *data, GEN P, long a, GEN x)
    1518             : {
    1519             :   (void) data;
    1520     1152100 :   return ZX_Z_mul(x, gel(P,a+2));
    1521             : }
    1522             : static GEN
    1523     3302888 : _FpXQ_sqr(void *data, GEN x)
    1524             : {
    1525     3302888 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1526     3302888 :   return FpXQ_sqr(x, D->T, D->p);
    1527             : }
    1528             : static GEN
    1529     2273909 : _FpXQ_mul(void *data, GEN x, GEN y)
    1530             : {
    1531     2273909 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1532     2273909 :   return FpXQ_mul(x,y, D->T, D->p);
    1533             : }
    1534             : static GEN
    1535        3537 : _FpXQ_zero(void *data)
    1536             : {
    1537        3537 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1538        3537 :   return pol_0(get_FpX_var(D->T));
    1539             : }
    1540             : static GEN
    1541      303559 : _FpXQ_one(void *data)
    1542             : {
    1543      303559 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1544      303559 :   return pol_1(get_FpX_var(D->T));
    1545             : }
    1546             : static GEN
    1547      346072 : _FpXQ_red(void *data, GEN x)
    1548             : {
    1549      346072 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1550      346072 :   return FpX_red(x,D->p);
    1551             : }
    1552             : 
    1553             : static struct bb_algebra FpXQ_algebra = { _FpXQ_red, _FpXQ_add, _FpXQ_sub,
    1554             :        _FpXQ_mul, _FpXQ_sqr, _FpXQ_one, _FpXQ_zero };
    1555             : 
    1556             : const struct bb_algebra *
    1557       13559 : get_FpXQ_algebra(void **E, GEN T, GEN p)
    1558             : {
    1559       13559 :   GEN z = new_chunk(sizeof(struct _FpXQ));
    1560       13559 :   struct _FpXQ *e = (struct _FpXQ *) z;
    1561       13559 :   e->T = FpX_get_red(T, p);
    1562       13559 :   e->p  = p; *E = (void*)e;
    1563       13559 :   return &FpXQ_algebra;
    1564             : }
    1565             : 
    1566             : static struct bb_algebra FpX_algebra = { _FpXQ_red, _FpXQ_add, _FpXQ_sub,
    1567             :        _FpX_mul, _FpX_sqr, _FpXQ_one, _FpXQ_zero };
    1568             : 
    1569             : const struct bb_algebra *
    1570           0 : get_FpX_algebra(void **E, GEN p, long v)
    1571             : {
    1572           0 :   GEN z = new_chunk(sizeof(struct _FpXQ));
    1573           0 :   struct _FpXQ *e = (struct _FpXQ *) z;
    1574           0 :   e->T = pol_x(v);
    1575           0 :   e->p  = p; *E = (void*)e;
    1576           0 :   return &FpX_algebra;
    1577             : }
    1578             : 
    1579             : /* x,pol in Z[X], p in Z, n in Z, compute lift(x^n mod (p, pol)) */
    1580             : GEN
    1581     1212181 : FpXQ_pow(GEN x, GEN n, GEN T, GEN p)
    1582             : {
    1583             :   struct _FpXQ D;
    1584             :   pari_sp av;
    1585     1212181 :   long s = signe(n);
    1586             :   GEN y;
    1587     1212181 :   if (!s) return pol_1(varn(x));
    1588     1210487 :   if (is_pm1(n)) /* +/- 1 */
    1589        8608 :     return (s < 0)? FpXQ_inv(x,T,p): FpXQ_red(x,T,p);
    1590     1201879 :   av = avma;
    1591     1201879 :   if (!is_bigint(p))
    1592             :   {
    1593      337316 :     ulong pp = to_Flxq(&x, &T, p);
    1594      337316 :     y = Flxq_pow(x, n, T, pp);
    1595      337316 :     return Flx_to_ZX_inplace(gerepileuptoleaf(av, y));
    1596             :   }
    1597      864563 :   if (s < 0) x = FpXQ_inv(x,T,p);
    1598      864563 :   D.p = p; D.T = FpX_get_red(T,p);
    1599      864563 :   y = gen_pow(x, n, (void*)&D, &_FpXQ_sqr, &_FpXQ_mul);
    1600      864563 :   return gerepileupto(av, y);
    1601             : }
    1602             : 
    1603             : GEN /*Assume n is very small*/
    1604       61873 : FpXQ_powu(GEN x, ulong n, GEN T, GEN p)
    1605             : {
    1606             :   struct _FpXQ D;
    1607             :   pari_sp av;
    1608             :   GEN y;
    1609       61873 :   if (!n) return pol_1(varn(x));
    1610       61873 :   if (n==1) return FpXQ_red(x,T,p);
    1611       32312 :   av = avma;
    1612       32312 :   if (!is_bigint(p))
    1613             :   {
    1614       32068 :     ulong pp = to_Flxq(&x, &T, p);
    1615       32068 :     y = Flxq_powu(x, n, T, pp);
    1616       32068 :     return Flx_to_ZX_inplace(gerepileuptoleaf(av, y));
    1617             :   }
    1618         244 :   D.T = FpX_get_red(T, p); D.p = p;
    1619         244 :   y = gen_powu(x, n, (void*)&D, &_FpXQ_sqr, &_FpXQ_mul);
    1620         244 :   return gerepileupto(av, y);
    1621             : }
    1622             : 
    1623             : /* generates the list of powers of x of degree 0,1,2,...,l*/
    1624             : GEN
    1625      184144 : FpXQ_powers(GEN x, long l, GEN T, GEN p)
    1626             : {
    1627             :   struct _FpXQ D;
    1628             :   int use_sqr;
    1629      184144 :   if (l>2 && lgefint(p) == 3) {
    1630      152146 :     pari_sp av = avma;
    1631      152146 :     ulong pp = to_Flxq(&x, &T, p);
    1632      152146 :     GEN z = FlxV_to_ZXV(Flxq_powers(x, l, T, pp));
    1633      152146 :     return gerepileupto(av, z);
    1634             :   }
    1635       31998 :   use_sqr = 2*degpol(x)>=get_FpX_degree(T);
    1636       31998 :   D.T = FpX_get_red(T,p); D.p = p;
    1637       31998 :   return gen_powers(x, l, use_sqr, (void*)&D, &_FpXQ_sqr, &_FpXQ_mul,&_FpXQ_one);
    1638             : }
    1639             : 
    1640             : GEN
    1641        3878 : FpXQ_matrix_pow(GEN y, long n, long m, GEN P, GEN l)
    1642             : {
    1643        3878 :   return RgXV_to_RgM(FpXQ_powers(y,m-1,P,l),n);
    1644             : }
    1645             : 
    1646             : GEN
    1647      156244 : FpX_Frobenius(GEN T, GEN p)
    1648             : {
    1649      156244 :   return FpXQ_pow(pol_x(get_FpX_var(T)), p, T, p);
    1650             : }
    1651             : 
    1652             : GEN
    1653        2583 : FpX_matFrobenius(GEN T, GEN p)
    1654             : {
    1655        2583 :   long n = get_FpX_degree(T);
    1656        2583 :   return FpXQ_matrix_pow(FpX_Frobenius(T, p), n, n, T, p);
    1657             : }
    1658             : 
    1659             : GEN
    1660      124340 : FpX_FpXQV_eval(GEN Q, GEN x, GEN T, GEN p)
    1661             : {
    1662             :   struct _FpXQ D;
    1663      124340 :   D.T = FpX_get_red(T,p); D.p = p;
    1664      124340 :   return gen_bkeval_powers(Q,degpol(Q),x,(void*)&D,&FpXQ_algebra,_FpXQ_cmul);
    1665             : }
    1666             : 
    1667             : GEN
    1668      173849 : FpX_FpXQ_eval(GEN Q, GEN x, GEN T, GEN p)
    1669             : {
    1670             :   struct _FpXQ D;
    1671             :   int use_sqr;
    1672      173849 :   if (lgefint(p) == 3)
    1673             :   {
    1674      170623 :     pari_sp av = avma;
    1675      170623 :     ulong pp = to_Flxq(&x, &T, p);
    1676      170623 :     GEN z = Flx_Flxq_eval(ZX_to_Flx(Q, pp), x, T, pp);
    1677      170623 :     return Flx_to_ZX_inplace(gerepileuptoleaf(av, z));
    1678             :   }
    1679        3226 :   use_sqr = 2*degpol(x) >= get_FpX_degree(T);
    1680        3226 :   D.T = FpX_get_red(T,p); D.p = p;
    1681        3226 :   return gen_bkeval(Q,degpol(Q),x,use_sqr,(void*)&D,&FpXQ_algebra,_FpXQ_cmul);
    1682             : }
    1683             : 
    1684             : GEN
    1685         854 : FpXC_FpXQV_eval(GEN P, GEN x, GEN T, GEN p)
    1686             : {
    1687         854 :   long i, l = lg(P);
    1688         854 :   GEN res = cgetg(l, t_COL);
    1689        3682 :   for (i=1; i<l; i++)
    1690        2828 :     gel(res,i) = FpX_FpXQV_eval(gel(P,i), x, T, p);
    1691         854 :   return res;
    1692             : }
    1693             : 
    1694             : GEN
    1695         308 : FpXM_FpXQV_eval(GEN Q, GEN x, GEN T, GEN p)
    1696             : {
    1697         308 :   long i, l = lg(Q);
    1698         308 :   GEN y = cgetg(l, t_MAT);
    1699        1162 :   for (i=1; i<l; i++)
    1700         854 :     gel(y,i) = FpXC_FpXQV_eval(gel(Q,i), x, T, p);
    1701         308 :   return y;
    1702             : }
    1703             : 
    1704             : GEN
    1705         595 : FpXQ_autpowers(GEN aut, long f, GEN T, GEN p)
    1706             : {
    1707         595 :   pari_sp av = avma;
    1708         595 :   long n = get_FpX_degree(T);
    1709         595 :   long i, nautpow = brent_kung_optpow(n-1,f-2,1);
    1710         595 :   long v = get_FpX_var(T);
    1711             :   GEN autpow, V;
    1712         595 :   T = FpX_get_red(T, p);
    1713         595 :   autpow = FpXQ_powers(aut, nautpow,T,p);
    1714         595 :   V = cgetg(f + 2, t_VEC);
    1715         595 :   gel(V,1) = pol_x(v); if (f==0) return gerepileupto(av, V);
    1716         595 :   gel(V,2) = gcopy(aut);
    1717        3101 :   for (i = 3; i <= f+1; i++)
    1718        2506 :     gel(V,i) = FpX_FpXQV_eval(gel(V,i-1),autpow,T,p);
    1719         595 :   return gerepileupto(av, V);
    1720             : }
    1721             : 
    1722             : static GEN
    1723         542 : FpXQ_autpow_sqr(void *E, GEN x)
    1724             : {
    1725         542 :   struct _FpXQ *D = (struct _FpXQ*)E;
    1726         542 :   return FpX_FpXQ_eval(x, x, D->T, D->p);
    1727             : }
    1728             : 
    1729             : static GEN
    1730           7 : FpXQ_autpow_mul(void *E, GEN x, GEN y)
    1731             : {
    1732           7 :   struct _FpXQ *D = (struct _FpXQ*)E;
    1733           7 :   return FpX_FpXQ_eval(x, y, D->T, D->p);
    1734             : }
    1735             : 
    1736             : GEN
    1737         514 : FpXQ_autpow(GEN x, ulong n, GEN T, GEN p)
    1738             : {
    1739             :   struct _FpXQ D;
    1740         514 :   D.T = FpX_get_red(T, p); D.p = p;
    1741         514 :   if (n==0) return pol_x(varn(x));
    1742         514 :   if (n==1) return ZX_copy(x);
    1743         514 :   return gen_powu(x,n,(void*)&D,FpXQ_autpow_sqr,FpXQ_autpow_mul);
    1744             : }
    1745             : 
    1746             : static GEN
    1747           7 : FpXQ_auttrace_mul(void *E, GEN x, GEN y)
    1748             : {
    1749           7 :   struct _FpXQ *D = (struct _FpXQ*)E;
    1750           7 :   GEN T = D->T, p = D->p;
    1751           7 :   GEN phi1 = gel(x,1), a1 = gel(x,2);
    1752           7 :   GEN phi2 = gel(y,1), a2 = gel(y,2);
    1753           7 :   ulong d = brent_kung_optpow(maxss(degpol(phi2),degpol(a2)),2,1);
    1754           7 :   GEN V1 = FpXQ_powers(phi1, d, T, p);
    1755           7 :   GEN phi3 = FpX_FpXQV_eval(phi2, V1, T, p);
    1756           7 :   GEN aphi = FpX_FpXQV_eval(a2, V1, T, p);
    1757           7 :   GEN a3 = FpX_add(a1, aphi, p);
    1758           7 :   return mkvec2(phi3, a3);
    1759             : }
    1760             : 
    1761             : static GEN
    1762           7 : FpXQ_auttrace_sqr(void *E, GEN x)
    1763           7 : { return FpXQ_auttrace_mul(E, x, x); }
    1764             : 
    1765             : GEN
    1766          14 : FpXQ_auttrace(GEN x, ulong n, GEN T, GEN p)
    1767             : {
    1768             :   struct _FpXQ D;
    1769          14 :   D.T = FpX_get_red(T, p); D.p = p;
    1770          14 :   return gen_powu(x,n,(void*)&D,FpXQ_auttrace_sqr,FpXQ_auttrace_mul);
    1771             : }
    1772             : 
    1773             : static GEN
    1774        1866 : FpXQ_autsum_mul(void *E, GEN x, GEN y)
    1775             : {
    1776        1866 :   struct _FpXQ *D = (struct _FpXQ*)E;
    1777        1866 :   GEN T = D->T, p = D->p;
    1778        1866 :   GEN phi1 = gel(x,1), a1 = gel(x,2);
    1779        1866 :   GEN phi2 = gel(y,1), a2 = gel(y,2);
    1780        1866 :   ulong d = brent_kung_optpow(maxss(degpol(phi2),degpol(a2)),2,1);
    1781        1866 :   GEN V1 = FpXQ_powers(phi1, d, T, p);
    1782        1866 :   GEN phi3 = FpX_FpXQV_eval(phi2, V1, T, p);
    1783        1866 :   GEN aphi = FpX_FpXQV_eval(a2, V1, T, p);
    1784        1866 :   GEN a3 = FpXQ_mul(a1, aphi, T, p);
    1785        1866 :   return mkvec2(phi3, a3);
    1786             : }
    1787             : static GEN
    1788         970 : FpXQ_autsum_sqr(void *E, GEN x)
    1789         970 : { return FpXQ_autsum_mul(E, x, x); }
    1790             : 
    1791             : GEN
    1792         956 : FpXQ_autsum(GEN x, ulong n, GEN T, GEN p)
    1793             : {
    1794             :   struct _FpXQ D;
    1795         956 :   D.T = FpX_get_red(T, p); D.p = p;
    1796         956 :   return gen_powu(x,n,(void*)&D,FpXQ_autsum_sqr,FpXQ_autsum_mul);
    1797             : }
    1798             : 
    1799             : static GEN
    1800         308 : FpXQM_autsum_mul(void *E, GEN x, GEN y)
    1801             : {
    1802         308 :   struct _FpXQ *D = (struct _FpXQ*)E;
    1803         308 :   GEN T = D->T, p = D->p;
    1804         308 :   GEN phi1 = gel(x,1), a1 = gel(x,2);
    1805         308 :   GEN phi2 = gel(y,1), a2 = gel(y,2);
    1806         308 :   long g = lg(a2)-1, dT = get_FpX_degree(T);
    1807         308 :   ulong d = brent_kung_optpow(dT-1, g*g+1, 1);
    1808         308 :   GEN V1 = FpXQ_powers(phi1, d, T, p);
    1809         308 :   GEN phi3 = FpX_FpXQV_eval(phi2, V1, T, p);
    1810         308 :   GEN aphi = FpXM_FpXQV_eval(a2, V1, T, p);
    1811         308 :   GEN a3 = FqM_mul(a1, aphi, T, p);
    1812         308 :   return mkvec2(phi3, a3);
    1813             : }
    1814             : static GEN
    1815         210 : FpXQM_autsum_sqr(void *E, GEN x)
    1816         210 : { return FpXQM_autsum_mul(E, x, x); }
    1817             : 
    1818             : GEN
    1819         140 : FpXQM_autsum(GEN x, ulong n, GEN T, GEN p)
    1820             : {
    1821             :   struct _FpXQ D;
    1822         140 :   D.T = FpX_get_red(T, p); D.p = p;
    1823         140 :   return gen_powu(x, n, (void*)&D, FpXQM_autsum_sqr, FpXQM_autsum_mul);
    1824             : }
    1825             : 
    1826             : static long
    1827        5442 : bounded_order(GEN p, GEN b, long k)
    1828             : {
    1829             :   long i;
    1830        5442 :   GEN a=modii(p,b);
    1831       10828 :   for(i=1;i<k;i++)
    1832             :   {
    1833        6772 :     if (equali1(a))
    1834        1386 :       return i;
    1835        5386 :     a = Fp_mul(a,p,b);
    1836             :   }
    1837        4056 :   return 0;
    1838             : }
    1839             : 
    1840             : /*
    1841             :   n = (p^d-a)\b
    1842             :   b = bb*p^vb
    1843             :   p^k = 1 [bb]
    1844             :   d = m*k+r+vb
    1845             :   u = (p^k-1)/bb;
    1846             :   v = (p^(r+vb)-a)/b;
    1847             :   w = (p^(m*k)-1)/(p^k-1)
    1848             :   n = p^r*w*u+v
    1849             :   w*u = p^vb*(p^(m*k)-1)/b
    1850             :   n = p^(r+vb)*(p^(m*k)-1)/b+(p^(r+vb)-a)/b
    1851             : */
    1852             : 
    1853             : static GEN
    1854      853200 : FpXQ_pow_Frobenius(GEN x, GEN n, GEN aut, GEN T, GEN p)
    1855             : {
    1856      853200 :   pari_sp av=avma;
    1857      853200 :   long d = get_FpX_degree(T);
    1858      853200 :   GEN an = absi(n), z, q;
    1859      853200 :   if (cmpii(an,p)<0 || cmpis(an,d)<=0)
    1860      847758 :     return FpXQ_pow(x, n, T, p);
    1861        5442 :   q = powiu(p, d);
    1862        5442 :   if (dvdii(q, n))
    1863             :   {
    1864           0 :     long vn = logint(an,p);
    1865           0 :     GEN autvn = vn==1 ? aut: FpXQ_autpow(aut,vn,T,p);
    1866           0 :     z = FpX_FpXQ_eval(x,autvn,T,p);
    1867             :   } else
    1868             :   {
    1869        5442 :     GEN b = diviiround(q, an), a = subii(q, mulii(an,b));
    1870             :     GEN bb, u, v, autk;
    1871        5442 :     long vb = Z_pvalrem(b,p,&bb);
    1872        5442 :     long m, r, k = is_pm1(bb) ? 1 : bounded_order(p,bb,d);
    1873        5442 :     if (!k || d-vb<k) return FpXQ_pow(x,n, T, p);
    1874        1386 :     m = (d-vb)/k; r = (d-vb)%k;
    1875        1386 :     u = diviiexact(subis(powiu(p,k),1),bb);
    1876        1386 :     v = diviiexact(subii(powiu(p,r+vb),a),b);
    1877        1386 :     autk = k==1 ? aut: FpXQ_autpow(aut,k,T,p);
    1878        1386 :     if (r)
    1879             :     {
    1880         437 :       GEN autr = r==1 ? aut: FpXQ_autpow(aut,r,T,p);
    1881         437 :       z = FpX_FpXQ_eval(x,autr,T,p);
    1882         949 :     } else z = x;
    1883        1386 :     if (m > 1) z = gel(FpXQ_autsum(mkvec2(autk, z), m, T, p), 2);
    1884        1386 :     if (!is_pm1(u)) z = FpXQ_pow(z, u, T, p);
    1885        1386 :     if (signe(v)) z = FpXQ_mul(z, FpXQ_pow(x, v, T, p), T, p);
    1886             :   }
    1887        1386 :   return gerepileupto(av,signe(n)>0 ? z : FpXQ_inv(z,T,p));
    1888             : }
    1889             : 
    1890             : /* assume T irreducible mod p */
    1891             : int
    1892        3470 : FpXQ_issquare(GEN x, GEN T, GEN p)
    1893             : {
    1894             :   pari_sp av;
    1895             :   long res;
    1896        3470 :   if (lg(x) == 2 || absequalui(2, p)) return 1;
    1897        3470 :   if (lg(x) == 3) return Fq_issquare(gel(x,2), T, p);
    1898             :   /* Ng = g^((q-1)/(p-1)) */
    1899        3421 :   av = avma; res = kronecker(FpXQ_norm(x,T,p), p) == 1;
    1900        3421 :   avma = av; return res;
    1901             : }
    1902             : int
    1903       90755 : Fp_issquare(GEN x, GEN p)
    1904             : {
    1905       90755 :   if (absequalui(2, p)) return 1;
    1906       90755 :   return kronecker(x, p) == 1;
    1907             : }
    1908             : /* assume T irreducible mod p */
    1909             : int
    1910       90580 : Fq_issquare(GEN x, GEN T, GEN p)
    1911             : {
    1912       90580 :   if (typ(x) != t_INT) return FpXQ_issquare(x, T, p);
    1913       90545 :   return (T && ! odd(get_FpX_degree(T))) || Fp_issquare(x, p);
    1914             : }
    1915             : 
    1916             : long
    1917         112 : Fq_ispower(GEN x, GEN K, GEN T, GEN p)
    1918             : {
    1919         112 :   pari_sp av = avma;
    1920             :   long d;
    1921             :   GEN Q;
    1922         112 :   if (!T) return Fp_ispower(x,K,p);
    1923         112 :   d = get_FpX_degree(T);
    1924         112 :   if (!umodui(d, K)) return 1;
    1925          63 :   Q = subiu(powiu(p,d), 1);
    1926          63 :   Q = diviiexact(Q, gcdii(Q, K));
    1927          63 :   d = gequal1(Fq_pow(x, Q, T,p));
    1928          63 :   avma = av; return d;
    1929             : }
    1930             : 
    1931             : /* discrete log in FpXQ for a in Fp^*, g in FpXQ^* of order ord */
    1932             : GEN
    1933       38600 : Fp_FpXQ_log(GEN a, GEN g, GEN o, GEN T, GEN p)
    1934             : {
    1935       38600 :   pari_sp av = avma;
    1936             :   GEN q,n_q,ord,ordp, op;
    1937             : 
    1938       38600 :   if (equali1(a)) return gen_0;
    1939             :   /* p > 2 */
    1940             : 
    1941        3591 :   ordp = subis(p, 1); /* even */
    1942        3591 :   ord  = get_arith_Z(o);
    1943        3563 :   if (!ord) ord = T? subis(powiu(p, get_FpX_degree(T)), 1): ordp;
    1944        3563 :   if (equalii(a, ordp)) /* -1 */
    1945        2041 :     return gerepileuptoint(av, shifti(ord,-1));
    1946        1522 :   ordp = gcdii(ordp,ord);
    1947        1522 :   op = typ(o)==t_MAT ? famat_Z_gcd(o,ordp) : ordp;
    1948             : 
    1949        1522 :   q = NULL;
    1950        1522 :   if (T)
    1951             :   { /* we want < g > = Fp^* */
    1952        1522 :     if (!equalii(ord,ordp)) {
    1953        1512 :       q = diviiexact(ord,ordp);
    1954        1512 :       g = FpXQ_pow(g,q,T,p);
    1955             :     }
    1956        1522 :     g = constant_coeff(g);
    1957             :   }
    1958        1522 :   n_q = Fp_log(a,g,op,p);
    1959        1522 :   if (lg(n_q)==1) return gerepileuptoleaf(av, n_q);
    1960        1522 :   if (q) n_q = mulii(q, n_q);
    1961        1522 :   return gerepileuptoint(av, n_q);
    1962             : }
    1963             : 
    1964             : static GEN
    1965      852492 : _FpXQ_pow(void *data, GEN x, GEN n)
    1966             : {
    1967      852492 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1968      852492 :   return FpXQ_pow_Frobenius(x,n, D->aut, D->T, D->p);
    1969             : }
    1970             : 
    1971             : static GEN
    1972        3462 : _FpXQ_rand(void *data)
    1973             : {
    1974        3462 :   pari_sp av=avma;
    1975        3462 :   struct _FpXQ *D = (struct _FpXQ*)data;
    1976             :   GEN z;
    1977             :   do
    1978             :   {
    1979        3462 :     avma=av;
    1980        3462 :     z=random_FpX(get_FpX_degree(D->T),get_FpX_var(D->T),D->p);
    1981        3462 :   } while (!signe(z));
    1982        3462 :   return z;
    1983             : }
    1984             : 
    1985             : static GEN
    1986        1348 : _FpXQ_easylog(void *E, GEN a, GEN g, GEN ord)
    1987             : {
    1988        1348 :   struct _FpXQ *s=(struct _FpXQ*) E;
    1989        1348 :   if (degpol(a)) return NULL;
    1990        1332 :   return Fp_FpXQ_log(constant_coeff(a),g,ord,s->T,s->p);
    1991             : }
    1992             : 
    1993             : static const struct bb_group FpXQ_star={_FpXQ_mul,_FpXQ_pow,_FpXQ_rand,hash_GEN,ZX_equal,ZX_equal1,_FpXQ_easylog};
    1994             : 
    1995             : const struct bb_group *
    1996        1771 : get_FpXQ_star(void **E, GEN T, GEN p)
    1997             : {
    1998        1771 :   struct _FpXQ *e = (struct _FpXQ *) stack_malloc(sizeof(struct _FpXQ));
    1999        1771 :   e->T = T; e->p  = p; e->aut =  FpX_Frobenius(T, p);
    2000        1771 :   *E = (void*)e; return &FpXQ_star;
    2001             : }
    2002             : 
    2003             : GEN
    2004          29 : FpXQ_order(GEN a, GEN ord, GEN T, GEN p)
    2005             : {
    2006          29 :   if (lgefint(p)==3)
    2007             :   {
    2008           0 :     pari_sp av=avma;
    2009           0 :     ulong pp = to_Flxq(&a, &T, p);
    2010           0 :     GEN z = Flxq_order(a, ord, T, pp);
    2011           0 :     return gerepileuptoint(av,z);
    2012             :   }
    2013             :   else
    2014             :   {
    2015             :     void *E;
    2016          29 :     const struct bb_group *S = get_FpXQ_star(&E,T,p);
    2017          29 :     return gen_order(a,ord,E,S);
    2018             :   }
    2019             : }
    2020             : 
    2021             : GEN
    2022       84241 : FpXQ_log(GEN a, GEN g, GEN ord, GEN T, GEN p)
    2023             : {
    2024       84241 :   pari_sp av=avma;
    2025       84241 :   if (lgefint(p)==3)
    2026             :   {
    2027       84197 :     if (uel(p,2) == 2)
    2028             :     {
    2029       53032 :       GEN z = F2xq_log(ZX_to_F2x(a), ZX_to_F2x(g), ord,
    2030             :                                      ZX_to_F2x(get_FpX_mod(T)));
    2031       53032 :       return gerepileuptoleaf(av, z);
    2032             :     }
    2033             :     else
    2034             :     {
    2035       31165 :       ulong pp = to_Flxq(&a, &T, p);
    2036       31165 :       GEN z = Flxq_log(a, ZX_to_Flx(g, pp), ord, T, pp);
    2037       31165 :       return gerepileuptoleaf(av, z);
    2038             :     }
    2039             :   }
    2040             :   else
    2041             :   {
    2042             :     void *E;
    2043          44 :     const struct bb_group *S = get_FpXQ_star(&E,T,p);
    2044          44 :     GEN z = gen_PH_log(a,g,ord,E,S);
    2045          16 :     return gerepileuptoleaf(av, z);
    2046             :   }
    2047             : }
    2048             : 
    2049             : GEN
    2050        1876 : Fq_log(GEN a, GEN g, GEN ord, GEN T, GEN p)
    2051             : {
    2052        1876 :   if (!T) return Fp_log(a,g,ord,p);
    2053         784 :   if (typ(g) == t_INT)
    2054             :   {
    2055           0 :     if (typ(a) == t_POL)
    2056             :     {
    2057           0 :       if (degpol(a)) return cgetg(1,t_VEC);
    2058           0 :       a = gel(a,2);
    2059             :     }
    2060           0 :     return Fp_log(a,g,ord,p);
    2061             :   }
    2062         784 :   return typ(a) == t_INT? Fp_FpXQ_log(a,g,ord,T,p): FpXQ_log(a,g,ord,T,p);
    2063             : }
    2064             : 
    2065             : GEN
    2066        8431 : FpXQ_sqrtn(GEN a, GEN n, GEN T, GEN p, GEN *zeta)
    2067             : {
    2068        8431 :   pari_sp av = avma;
    2069             :   GEN z;
    2070        8431 :   if (!signe(a))
    2071             :   {
    2072          70 :     long v=varn(a);
    2073          70 :     if (signe(n) < 0) pari_err_INV("FpXQ_sqrtn",a);
    2074          63 :     if (zeta) *zeta=pol_1(v);
    2075          63 :     return pol_0(v);
    2076             :   }
    2077        8361 :   if (lgefint(p)==3)
    2078             :   {
    2079        6663 :     if (uel(p,2) == 2)
    2080             :     {
    2081         798 :       z = F2xq_sqrtn(ZX_to_F2x(a), n, ZX_to_F2x(get_Flx_mod(T)), zeta);
    2082         798 :       if (!z) return NULL;
    2083         798 :       z = F2x_to_ZX(z);
    2084         798 :       if (!zeta) return gerepileuptoleaf(av, z);
    2085           7 :       *zeta=F2x_to_ZX(*zeta);
    2086             :     } else
    2087             :     {
    2088        5865 :       ulong pp = to_Flxq(&a, &T, p);
    2089        5865 :       z = Flxq_sqrtn(a, n, T, pp, zeta);
    2090        5865 :       if (!z) return NULL;
    2091        3541 :       if (!zeta) return Flx_to_ZX_inplace(gerepileuptoleaf(av, z));
    2092          28 :       z = Flx_to_ZX(z);
    2093          28 :       *zeta=Flx_to_ZX(*zeta);
    2094             :     }
    2095             :   }
    2096             :   else
    2097             :   {
    2098             :     void *E;
    2099        1698 :     const struct bb_group *S = get_FpXQ_star(&E,T,p);
    2100        1698 :     GEN o = addis(powiu(p,get_FpX_degree(T)),-1);
    2101        1698 :     z = gen_Shanks_sqrtn(a,n,o,zeta,E,S);
    2102        3395 :     if (!z) return NULL;
    2103        1649 :     if (!zeta) return gerepileupto(av, z);
    2104             :   }
    2105          36 :   gerepileall(av, 2, &z,zeta);
    2106          36 :   return z;
    2107             : }
    2108             : 
    2109             : GEN
    2110        7377 : FpXQ_sqrt(GEN a, GEN T, GEN p)
    2111             : {
    2112        7377 :   return FpXQ_sqrtn(a, gen_2, T, p, NULL);
    2113             : }
    2114             : 
    2115             : GEN
    2116        3422 : FpXQ_norm(GEN x, GEN TB, GEN p)
    2117             : {
    2118        3422 :   pari_sp av = avma;
    2119        3422 :   GEN T = get_FpX_mod(TB);
    2120        3422 :   GEN y = FpX_resultant(T, x, p);
    2121        3422 :   GEN L = leading_coeff(T);
    2122        3422 :   if (gequal1(L) || signe(x)==0) return y;
    2123           0 :   return gerepileupto(av, Fp_div(y, Fp_pows(L, degpol(x), p), p));
    2124             : }
    2125             : 
    2126             : GEN
    2127       20803 : FpXQ_trace(GEN x, GEN TB, GEN p)
    2128             : {
    2129       20803 :   pari_sp av = avma;
    2130       20803 :   GEN T = get_FpX_mod(TB);
    2131       20803 :   GEN dT = FpX_deriv(T,p);
    2132       20803 :   long n = degpol(dT);
    2133       20803 :   GEN z = FpXQ_mul(x, dT, TB, p);
    2134       20803 :   if (degpol(z)<n) { avma = av; return gen_0; }
    2135       19627 :   return gerepileuptoint(av, Fp_div(gel(z,2+n), gel(T,3+n),p));
    2136             : }
    2137             : 
    2138             : GEN
    2139           1 : FpXQ_charpoly(GEN x, GEN T, GEN p)
    2140             : {
    2141           1 :   pari_sp ltop=avma;
    2142           1 :   long vT, v = fetch_var();
    2143             :   GEN R;
    2144           1 :   T = leafcopy(get_FpX_mod(T));
    2145           1 :   vT = varn(T); setvarn(T, v);
    2146           1 :   x = leafcopy(x); setvarn(x, v);
    2147           1 :   R = FpX_FpXY_resultant(T, deg1pol_shallow(gen_1,FpX_neg(x,p),vT),p);
    2148           1 :   (void)delete_var(); return gerepileupto(ltop,R);
    2149             : }
    2150             : 
    2151             : /* Computing minimal polynomial :                         */
    2152             : /* cf Shoup 'Efficient Computation of Minimal Polynomials */
    2153             : /*          in Algebraic Extensions of Finite Fields'     */
    2154             : 
    2155             : static GEN
    2156         108 : FpXn_mul(GEN a, GEN b, long n, GEN p)
    2157             : {
    2158         108 :   return FpX_red(RgXn_red_shallow(ZX_mul(a, b), n), p);
    2159             : }
    2160             : 
    2161             : /* Let v a linear form, return the linear form z->v(tau*z)
    2162             :    that is, v*(M_tau) */
    2163             : 
    2164             : static GEN
    2165          30 : FpXQ_transmul_init(GEN tau, GEN T, GEN p)
    2166             : {
    2167             :   GEN bht;
    2168          30 :   GEN h, Tp = get_FpX_red(T, &h);
    2169          30 :   long n = degpol(Tp), vT = varn(Tp);
    2170          30 :   GEN ft = FpX_recipspec(Tp+2, n+1, n+1);
    2171          30 :   GEN bt = FpX_recipspec(tau+2, lgpol(tau), n);
    2172          30 :   setvarn(ft, vT); setvarn(bt, vT);
    2173          30 :   if (h)
    2174           0 :     bht = FpXn_mul(bt, h, n-1, p);
    2175             :   else
    2176             :   {
    2177          30 :     GEN bh = FpX_div(RgX_shift_shallow(tau, n-1), T, p);
    2178          30 :     bht = FpX_recipspec(bh+2, lgpol(bh), n-1);
    2179          30 :     setvarn(bht, vT);
    2180             :   }
    2181          30 :   return mkvec3(bt, bht, ft);
    2182             : }
    2183             : 
    2184             : static GEN
    2185         123 : FpXQ_transmul(GEN tau, GEN a, long n, GEN p)
    2186             : {
    2187         123 :   pari_sp ltop = avma;
    2188             :   GEN t1, t2, t3, vec;
    2189         123 :   GEN bt = gel(tau, 1), bht = gel(tau, 2), ft = gel(tau, 3);
    2190         123 :   if (signe(a)==0) return pol_0(varn(a));
    2191         123 :   t2 = RgX_shift_shallow(FpX_mul(bt, a, p),1-n);
    2192         123 :   if (signe(bht)==0) return gerepilecopy(ltop, t2);
    2193         108 :   t1 = RgX_shift_shallow(FpX_mul(ft, a, p),-n);
    2194         108 :   t3 = FpXn_mul(t1, bht, n-1, p);
    2195         108 :   vec = FpX_sub(t2, RgX_shift_shallow(t3, 1), p);
    2196         108 :   return gerepileupto(ltop, vec);
    2197             : }
    2198             : 
    2199             : GEN
    2200        3109 : FpXQ_minpoly(GEN x, GEN T, GEN p)
    2201             : {
    2202        3109 :   pari_sp ltop = avma;
    2203             :   long vT, n;
    2204             :   GEN v_x, g, tau;
    2205        3109 :   if (lgefint(p)==3)
    2206             :   {
    2207        3094 :     ulong pp = to_Flxq(&x, &T, p);
    2208        3094 :     GEN g = Flxq_minpoly(x, T, pp);
    2209        3094 :     return gerepileupto(ltop, Flx_to_ZX(g));
    2210             :   }
    2211          15 :   vT = get_FpX_var(T);
    2212          15 :   n = get_FpX_degree(T);
    2213          15 :   g = pol_1(vT);
    2214          15 :   tau = pol_1(vT);
    2215          15 :   T = FpX_get_red(T, p);
    2216          15 :   x = FpXQ_red(x, T, p);
    2217          15 :   v_x = FpXQ_powers(x, usqrt(2*n), T, p);
    2218          45 :   while(signe(tau) != 0)
    2219             :   {
    2220             :     long i, j, m, k1;
    2221             :     GEN M, v, tr;
    2222             :     GEN g_prime, c;
    2223          15 :     if (degpol(g) == n) { tau = pol_1(vT); g = pol_1(vT); }
    2224          15 :     v = random_FpX(n, vT, p);
    2225          15 :     tr = FpXQ_transmul_init(tau, T, p);
    2226          15 :     v = FpXQ_transmul(tr, v, n, p);
    2227          15 :     m = 2*(n-degpol(g));
    2228          15 :     k1 = usqrt(m);
    2229          15 :     tr = FpXQ_transmul_init(gel(v_x,k1+1), T, p);
    2230          15 :     c = cgetg(m+2,t_POL);
    2231          15 :     c[1] = evalsigne(1)|evalvarn(vT);
    2232         123 :     for (i=0; i<m; i+=k1)
    2233             :     {
    2234         108 :       long mj = minss(m-i, k1);
    2235         674 :       for (j=0; j<mj; j++)
    2236         566 :         gel(c,m+1-(i+j)) = FpX_dotproduct(v, gel(v_x,j+1), p);
    2237         108 :       v = FpXQ_transmul(tr, v, n, p);
    2238             :     }
    2239          15 :     c = FpX_renormalize(c, m+2);
    2240             :     /* now c contains <v,x^i> , i = 0..m-1  */
    2241          15 :     M = FpX_halfgcd(monomial(gen_1, m, vT), c, p);
    2242          15 :     g_prime = gmael(M, 2, 2);
    2243          15 :     if (degpol(g_prime) < 1) continue;
    2244          15 :     g = FpX_mul(g, g_prime, p);
    2245          15 :     tau = FpXQ_mul(tau, FpX_FpXQV_eval(g_prime, v_x, T, p), T, p);
    2246             :   }
    2247          15 :   g = FpX_normalize(g,p);
    2248          15 :   return gerepilecopy(ltop,g);
    2249             : }
    2250             : 
    2251             : GEN
    2252           8 : FpXQ_conjvec(GEN x, GEN T, GEN p)
    2253             : {
    2254           8 :   pari_sp av=avma;
    2255             :   long i;
    2256           8 :   long n = get_FpX_degree(T), v = varn(x);
    2257           8 :   GEN M = FpX_matFrobenius(T, p);
    2258           8 :   GEN z = cgetg(n+1,t_COL);
    2259           8 :   gel(z,1) = RgX_to_RgC(x,n);
    2260           8 :   for (i=2; i<=n; i++) gel(z,i) = FpM_FpC_mul(M,gel(z,i-1),p);
    2261           8 :   gel(z,1) = x;
    2262           8 :   for (i=2; i<=n; i++) gel(z,i) = RgV_to_RgX(gel(z,i),v);
    2263           8 :   return gerepilecopy(av,z);
    2264             : }
    2265             : 
    2266             : /* p prime, p_1 = p-1, q = p^deg T, Lp = cofactors of some prime divisors
    2267             :  * l_p of p-1, Lq = cofactors of some prime divisors l_q of q-1, return a
    2268             :  * g in Fq such that
    2269             :  * - Ng generates all l_p-Sylows of Fp^*
    2270             :  * - g generates all l_q-Sylows of Fq^* */
    2271             : static GEN
    2272        1458 : gener_FpXQ_i(GEN T, GEN p, GEN p_1, GEN Lp, GEN Lq)
    2273             : {
    2274             :   pari_sp av;
    2275        1458 :   long vT = varn(T), f = degpol(T), l = lg(Lq);
    2276        1458 :   GEN F = FpX_Frobenius(T, p);
    2277        1458 :   int p_is_2 = is_pm1(p_1);
    2278        3216 :   for (av = avma;; avma = av)
    2279             :   {
    2280        3216 :     GEN t, g = random_FpX(f, vT, p);
    2281             :     long i;
    2282        3216 :     if (degpol(g) < 1) continue;
    2283        3097 :     if (p_is_2)
    2284         399 :       t = g;
    2285             :     else
    2286             :     {
    2287        2698 :       t = FpX_resultant(T, g, p); /* Ng = g^((q-1)/(p-1)), assuming T monic */
    2288        2698 :       if (kronecker(t, p) == 1) continue;
    2289        1283 :       if (lg(Lp) > 1 && !is_gener_Fp(t, p, p_1, Lp)) continue;
    2290        1283 :       t = FpXQ_pow(g, shifti(p_1,-1), T, p);
    2291             :     }
    2292        2166 :     for (i = 1; i < l; i++)
    2293             :     {
    2294         708 :       GEN a = FpXQ_pow_Frobenius(t, gel(Lq,i), F, T, p);
    2295         708 :       if (!degpol(a) && equalii(gel(a,2), p_1)) break;
    2296             :     }
    2297        3140 :     if (i == l) return g;
    2298        1758 :   }
    2299             : }
    2300             : 
    2301             : GEN
    2302        6673 : gener_FpXQ(GEN T, GEN p, GEN *po)
    2303             : {
    2304        6673 :   long i, j, f = get_FpX_degree(T);
    2305             :   GEN g, Lp, Lq, p_1, q_1, N, o;
    2306        6673 :   pari_sp av = avma;
    2307             : 
    2308        6673 :   p_1 = subiu(p,1);
    2309        6673 :   if (f == 1) {
    2310             :     GEN Lp, fa;
    2311           7 :     o = p_1;
    2312           7 :     fa = Z_factor(o);
    2313           7 :     Lp = gel(fa,1);
    2314           7 :     Lp = vecslice(Lp, 2, lg(Lp)-1); /* remove 2 for efficiency */
    2315             : 
    2316           7 :     g = cgetg(3, t_POL);
    2317           7 :     g[1] = evalsigne(1) | evalvarn(get_FpX_var(T));
    2318           7 :     gel(g,2) = pgener_Fp_local(p, Lp);
    2319           7 :     if (po) *po = mkvec2(o, fa);
    2320           7 :     return g;
    2321             :   }
    2322        6666 :   if (lgefint(p) == 3)
    2323             :   {
    2324        6643 :     ulong pp = to_Flxq(NULL, &T, p);
    2325        6643 :     g = gener_Flxq(T, pp, po);
    2326        6643 :     if (!po) return Flx_to_ZX_inplace(gerepileuptoleaf(av, g));
    2327           0 :     g = Flx_to_ZX(g);
    2328           0 :     gel(*po,2) = Flx_to_ZX(gel(*po,2));
    2329           0 :     gerepileall(av, 2, &g, po);
    2330           0 :     return g;
    2331             :   }
    2332             :   /* p now odd */
    2333          23 :   q_1 = subiu(powiu(p,f), 1);
    2334          23 :   N = diviiexact(q_1, p_1);
    2335          23 :   Lp = odd_prime_divisors(p_1);
    2336          23 :   for (i=lg(Lp)-1; i; i--) gel(Lp,i) = diviiexact(p_1, gel(Lp,i));
    2337          23 :   o = factor_pn_1(p,f);
    2338          23 :   Lq = leafcopy( gel(o, 1) );
    2339         199 :   for (i = j = 1; i < lg(Lq); i++)
    2340             :   {
    2341         176 :     if (remii(p_1, gel(Lq,i)) == gen_0) continue;
    2342         106 :     gel(Lq,j++) = diviiexact(N, gel(Lq,i));
    2343             :   }
    2344          23 :   setlg(Lq, j);
    2345          23 :   g = gener_FpXQ_i(get_FpX_mod(T), p, p_1, Lp, Lq);
    2346          23 :   if (!po) g = gerepilecopy(av, g);
    2347             :   else {
    2348           7 :     *po = mkvec2(q_1, o);
    2349           7 :     gerepileall(av, 2, &g, po);
    2350             :   }
    2351          23 :   return g;
    2352             : }
    2353             : 
    2354             : #if 0 /* generic version: slower */
    2355             : GEN
    2356             : gener_FpXQ2(GEN T, GEN p, GEN *po)
    2357             : {
    2358             :   pari_sp av = avma;
    2359             :   void *E;
    2360             :   long f = get_FpX_degree(T);
    2361             :   GEN g, o = factor_pn_1(p,f);
    2362             :   const struct bb_group *S = get_FpXQ_star(&E,T,p);
    2363             :   g = gen_gener(o,E,S);
    2364             :   if (!po) g = gerepilecopy(av, g);
    2365             :   else {
    2366             :     *po = mkvec2(powiu(p,f), o);
    2367             :     gerepileall(av, 2, &g, po);
    2368             :   }
    2369             :   return g;
    2370             : }
    2371             : #endif
    2372             : 
    2373             : GEN
    2374        1435 : gener_FpXQ_local(GEN T, GEN p, GEN L)
    2375             : {
    2376        1435 :   GEN Lp, Lq, p_1 = subiu(p,1), q_1, N, Q;
    2377        1435 :   long f, i, ip, iq, l = lg(L);
    2378        1435 :   T = get_FpX_mod(T);
    2379        1435 :   f = degpol(T);
    2380        1435 :   q_1 = subiu(powiu(p,f), 1);
    2381        1435 :   N = diviiexact(q_1, p_1);
    2382             : 
    2383        1435 :   Q = is_pm1(p_1)? gen_1: shifti(p_1,-1);
    2384        1435 :   Lp = cgetg(l, t_VEC); ip = 1;
    2385        1435 :   Lq = cgetg(l, t_VEC); iq = 1;
    2386        2100 :   for (i=1; i < l; i++)
    2387             :   {
    2388         665 :     GEN a, b, ell = gel(L,i);
    2389         665 :     if (absequaliu(ell,2)) continue;
    2390         385 :     a = dvmdii(Q, ell, &b);
    2391         385 :     if (b == gen_0)
    2392          21 :       gel(Lp,ip++) = a;
    2393             :     else
    2394         364 :       gel(Lq,iq++) = diviiexact(N,ell);
    2395             :   }
    2396        1435 :   setlg(Lp, ip);
    2397        1435 :   setlg(Lq, iq);
    2398        1435 :   return gener_FpXQ_i(T, p, p_1, Lp, Lq);
    2399             : }

Generated by: LCOV version 1.11