Code coverage tests

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

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

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

LCOV - code coverage report
Current view: top level - basemath - mftrace.c (source / functions) Hit Total Coverage
Test: PARI/GP v2.18.1 lcov report (development 30672-116f3d5b0e) Lines: 7532 7736 97.4 %
Date: 2026-02-06 09:23:51 Functions: 772 778 99.2 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* Copyright (C) 2016  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; either version 2 of the License, or (at your option) any later
       8             : version. It is distributed in the hope that it will be useful, but WITHOUT
       9             : ANY WARRANTY WHATSOEVER.
      10             : 
      11             : Check the License for details. You should have received a copy of it, along
      12             : with the package; see the file 'COPYING'. If not, write to the Free Software
      13             : Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */
      14             : 
      15             : /*************************************************************************/
      16             : /*                                                                       */
      17             : /*              Modular forms package based on trace formulas            */
      18             : /*                                                                       */
      19             : /*************************************************************************/
      20             : #include "pari.h"
      21             : #include "paripriv.h"
      22             : 
      23             : #define DEBUGLEVEL DEBUGLEVEL_mf
      24             : 
      25             : enum {
      26             :   MF_SPLIT = 1,
      27             :   MF_EISENSPACE,
      28             :   MF_FRICKE,
      29             :   MF_MF2INIT,
      30             :   MF_SPLITN
      31             : };
      32             : 
      33             : typedef struct {
      34             :   GEN vnew, vfull, DATA, VCHIP;
      35             :   long n, newHIT, newTOTAL, cuspHIT, cuspTOTAL;
      36             : } cachenew_t;
      37             : 
      38             : static void init_cachenew(cachenew_t *c, long n, long N, GEN f);
      39             : static long mf1cuspdim_i(long N, GEN CHI, GEN TMP, GEN vSP, long *dih);
      40             : static GEN mfinit_i(GEN NK, long space);
      41             : static GEN mfinit_Nkchi(long N, long k, GEN CHI, long space, long flraw);
      42             : static GEN mf2init_Nkchi(long N, long k, GEN CHI, long space, long flraw);
      43             : static GEN mf2basis(long N, long r, GEN CHI, GEN *pCHI1, long space);
      44             : static GEN mfeisensteinbasis(long N, long k, GEN CHI);
      45             : static GEN mfeisensteindec(GEN mf, GEN F);
      46             : static GEN initwt1newtrace(GEN mf);
      47             : static GEN initwt1trace(GEN mf);
      48             : static GEN myfactoru(long N);
      49             : static GEN mydivisorsu(long N);
      50             : static GEN Qab_Czeta(long k, long ord, GEN C, long vt);
      51             : static GEN mfcoefs_i(GEN F, long n, long d);
      52             : static GEN bhnmat_extend(GEN M, long m,long l, GEN S, cachenew_t *cache);
      53             : static GEN initnewtrace(long N, GEN CHI);
      54             : static void dbg_cachenew(cachenew_t *C);
      55             : static GEN hecke_i(long m, long l, GEN V, GEN F, GEN DATA);
      56             : static GEN c_Ek(long n, long d, GEN F);
      57             : static GEN RgV_heckef2(long n, long d, GEN V, GEN F, GEN DATA);
      58             : static GEN mfcusptrace_i(long N, long k, long n, GEN Dn, GEN TDATA);
      59             : static GEN mfnewtracecache(long N, long k, long n, cachenew_t *cache);
      60             : static GEN colnewtrace(long n0, long n, long d, long N, long k, cachenew_t *c);
      61             : static GEN dihan(GEN bnr, GEN w, GEN k0j, long m, ulong n);
      62             : static GEN sigchi(long k, GEN CHI, long n);
      63             : static GEN sigchi2(long k, GEN CHI1, GEN CHI2, long n, long ord);
      64             : static GEN mflineardivtomat(long N, GEN vF, long n);
      65             : static GEN mfdihedralcusp(long N, GEN CHI, GEN vSP);
      66             : static long mfdihedralcuspdim(long N, GEN CHI, GEN vSP);
      67             : static GEN mfdihedralnew(long N, GEN CHI, GEN SP);
      68             : static GEN mfdihedral(long N);
      69             : static GEN mfdihedralall(long N);
      70             : static long mf1cuspdim(long N, GEN CHI, GEN vSP);
      71             : static long mf2dim_Nkchi(long N, long k, GEN CHI, ulong space);
      72             : static long mfdim_Nkchi(long N, long k, GEN CHI, long space);
      73             : static GEN charLFwtk(long N, long k, GEN CHI, long ord, long t);
      74             : static GEN mfeisensteingacx(GEN E,long w,GEN ga,long n,long prec);
      75             : static GEN mfgaexpansion(GEN mf, GEN F, GEN gamma, long n, long prec);
      76             : static GEN mfEHmat(long n, long r);
      77             : static GEN mfEHcoef(long r, long N);
      78             : static GEN mftobasis_i(GEN mf, GEN F);
      79             : 
      80             : static GEN
      81       37863 : mkgNK(GEN N, GEN k, GEN CHI, GEN P) { return mkvec4(N, k, CHI, P); }
      82             : static GEN
      83       15267 : mkNK(long N, long k, GEN CHI) { return mkgNK(stoi(N), stoi(k), CHI, pol_x(1)); }
      84             : GEN
      85        8848 : MF_get_CHI(GEN mf) { return gmael(mf,1,3); }
      86             : GEN
      87       21273 : MF_get_gN(GEN mf) { return gmael(mf,1,1); }
      88             : long
      89       20069 : MF_get_N(GEN mf) { return itou(MF_get_gN(mf)); }
      90             : GEN
      91       15526 : MF_get_gk(GEN mf) { return gmael(mf,1,2); }
      92             : long
      93        7238 : MF_get_k(GEN mf)
      94             : {
      95        7238 :   GEN gk = MF_get_gk(mf);
      96        7238 :   if (typ(gk)!=t_INT) pari_err_IMPL("half-integral weight");
      97        7238 :   return itou(gk);
      98             : }
      99             : long
     100         280 : MF_get_r(GEN mf)
     101             : {
     102         280 :   GEN gk = MF_get_gk(mf);
     103         280 :   if (typ(gk) == t_INT) pari_err_IMPL("integral weight");
     104         280 :   return itou(gel(gk, 1)) >> 1;
     105             : }
     106             : long
     107       15393 : MF_get_space(GEN mf) { return itos(gmael(mf,1,4)); }
     108             : GEN
     109        4487 : MF_get_E(GEN mf) { return gel(mf,2); }
     110             : GEN
     111       21553 : MF_get_S(GEN mf) { return gel(mf,3); }
     112             : GEN
     113        1911 : MF_get_basis(GEN mf) { return shallowconcat(gel(mf,2), gel(mf,3)); }
     114             : long
     115        5642 : MF_get_dim(GEN mf)
     116             : {
     117        5642 :   switch(MF_get_space(mf))
     118             :   {
     119         721 :     case mf_FULL:
     120         721 :       return lg(MF_get_S(mf)) - 1 + lg(MF_get_E(mf))-1;
     121         140 :     case mf_EISEN:
     122         140 :       return lg(MF_get_E(mf))-1;
     123        4781 :     default: /* mf_NEW, mf_CUSP, mf_OLD */
     124        4781 :       return lg(MF_get_S(mf)) - 1;
     125             :   }
     126             : }
     127             : GEN
     128        7343 : MFnew_get_vj(GEN mf) { return gel(mf,4); }
     129             : GEN
     130         686 : MFcusp_get_vMjd(GEN mf) { return gel(mf,4); }
     131             : GEN
     132        6916 : MF_get_M(GEN mf) { return gmael(mf,5,3); }
     133             : GEN
     134        4872 : MF_get_Minv(GEN mf) { return gmael(mf,5,2); }
     135             : GEN
     136       10640 : MF_get_Mindex(GEN mf) { return gmael(mf,5,1); }
     137             : 
     138             : /* ordinary gtocol forgets about initial 0s */
     139             : GEN
     140        2583 : sertocol(GEN S) { return gtocol0(S, -(lg(S) - 2 + valser(S))); }
     141             : /*******************************************************************/
     142             : /*     Linear algebra in cyclotomic fields (TODO: export this)     */
     143             : /*******************************************************************/
     144             : /* return r and split prime p giving projection Q(zeta_n) -> Fp, zeta -> r */
     145             : static ulong
     146        1246 : QabM_init(long n, ulong *p)
     147             : {
     148        1246 :   ulong pinit = 1000000007;
     149             :   forprime_t T;
     150        1246 :   if (n <= 1) { *p = pinit; return 0; }
     151        1225 :   u_forprime_arith_init(&T, pinit, ULONG_MAX, 1, n);
     152        1225 :   *p = u_forprime_next(&T);
     153        1225 :   return Flx_oneroot(ZX_to_Flx(polcyclo(n, 0), *p), *p);
     154             : }
     155             : static ulong
     156     8534960 : Qab_to_Fl(GEN P, ulong r, ulong p)
     157             : {
     158             :   ulong t;
     159             :   GEN den;
     160     8534960 :   P = Q_remove_denom(liftpol_shallow(P), &den);
     161     8534960 :   if (typ(P) == t_POL) { GEN Pp = ZX_to_Flx(P, p); t = Flx_eval(Pp, r, p); }
     162     8399335 :   else t = umodiu(P, p);
     163     8534960 :   if (den) t = Fl_div(t, umodiu(den, p), p);
     164     8534960 :   return t;
     165             : }
     166             : static GEN
     167       38164 : QabC_to_Flc(GEN x, ulong r, ulong p)
     168     8341333 : { pari_APPLY_long( Qab_to_Fl(gel(x,i), r, p)); }
     169             : static GEN
     170         595 : QabM_to_Flm(GEN x, ulong r, ulong p)
     171       38759 : { pari_APPLY_same(QabC_to_Flc(gel(x, i), r, p);) }
     172             : /* A a t_POL */
     173             : static GEN
     174        1484 : QabX_to_Flx(GEN A, ulong r, ulong p)
     175             : {
     176        1484 :   long i, l = lg(A);
     177        1484 :   GEN a = cgetg(l, t_VECSMALL);
     178        1484 :   a[1] = ((ulong)A[1])&VARNBITS;
     179      233023 :   for (i = 2; i < l; i++) uel(a,i) = Qab_to_Fl(gel(A,i), r, p);
     180        1484 :   return Flx_renormalize(a, l);
     181             : }
     182             : 
     183             : /* FIXME: remove */
     184             : static GEN
     185        1106 : ZabM_pseudoinv_i(GEN M, GEN P, long n, GEN *pv, GEN *den, int ratlift)
     186             : {
     187        1106 :   GEN v = ZabM_indexrank(M, P, n);
     188        1106 :   if (pv) *pv = v;
     189        1106 :   M = shallowmatextract(M,gel(v,1),gel(v,2));
     190        1106 :   return ratlift? ZabM_inv_ratlift(M, P, n, den): ZabM_inv(M, P, n, den);
     191             : }
     192             : 
     193             : /* M matrix with coeff in Q(\chi)), where Q(\chi) = Q(X)/(P) for
     194             :  * P = cyclotomic Phi_n. Assume M rational if n <= 2 */
     195             : static GEN
     196        1652 : QabM_ker(GEN M, GEN P, long n)
     197             : {
     198        1652 :   if (n <= 2) return QM_ker(M);
     199         420 :   return ZabM_ker(row_Q_primpart(liftpol_shallow(M)), P, n);
     200             : }
     201             : /* pseudo-inverse of M. FIXME: should replace QabM_pseudoinv */
     202             : static GEN
     203        1358 : QabM_pseudoinv_i(GEN M, GEN P, long n, GEN *pv, GEN *pden)
     204             : {
     205             :   GEN cM, Mi;
     206        1358 :   if (n <= 2)
     207             :   {
     208        1176 :     M = Q_primitive_part(M, &cM);
     209        1176 :     Mi = ZM_pseudoinv(M, pv, pden); /* M^(-1) = Mi / (cM * den) */
     210             :   }
     211             :   else
     212             :   {
     213         182 :     M = Q_primitive_part(liftpol_shallow(M), &cM);
     214         182 :     Mi = ZabM_pseudoinv(M, P, n, pv, pden);
     215             :   }
     216        1358 :   *pden = mul_content(*pden, cM);
     217        1358 :   return Mi;
     218             : }
     219             : /* FIXME: delete */
     220             : static GEN
     221        1092 : QabM_pseudoinv(GEN M, GEN P, long n, GEN *pv, GEN *pden)
     222             : {
     223        1092 :   GEN Mi = QabM_pseudoinv_i(M, P, n, pv, pden);
     224        1092 :   return P? gmodulo(Mi, P): Mi;
     225             : }
     226             : 
     227             : static GEN
     228       10563 : QabM_indexrank(GEN M, GEN P, long n)
     229             : {
     230             :   GEN z;
     231       10563 :   if (n <= 2)
     232             :   {
     233        9366 :     M = vec_Q_primpart(M);
     234        9366 :     z = ZM_indexrank(M); /* M^(-1) = Mi / (cM * den) */
     235             :   }
     236             :   else
     237             :   {
     238        1197 :     M = vec_Q_primpart(liftpol_shallow(M));
     239        1197 :     z = ZabM_indexrank(M, P, n);
     240             :   }
     241       10563 :   return z;
     242             : }
     243             : 
     244             : /*********************************************************************/
     245             : /*                    Simple arithmetic functions                    */
     246             : /*********************************************************************/
     247             : /* TODO: most of these should be exported and used in ifactor1.c */
     248             : /* phi(n) */
     249             : static ulong
     250      110726 : myeulerphiu(ulong n)
     251             : {
     252             :   pari_sp av;
     253      110726 :   if (n == 1) return 1;
     254       91140 :   av = avma; return gc_ulong(av, eulerphiu_fact(myfactoru(n)));
     255             : }
     256             : static long
     257       65709 : mymoebiusu(ulong n)
     258             : {
     259             :   pari_sp av;
     260       65709 :   if (n == 1) return 1;
     261       54194 :   av = avma; return gc_long(av, moebiusu_fact(myfactoru(n)));
     262             : }
     263             : 
     264             : static long
     265        3031 : mynumdivu(long N)
     266             : {
     267             :   pari_sp av;
     268        3031 :   if (N == 1) return 1;
     269        2898 :   av = avma; return gc_long(av, numdivu_fact(myfactoru(N)));
     270             : }
     271             : 
     272             : /* N\prod_{p|N} (1+1/p) */
     273             : static long
     274      401541 : mypsiu(ulong N)
     275             : {
     276             :   pari_sp av;
     277             :   GEN P;
     278             :   long j, l, a;
     279      401541 :   if (N == 1) return 1;
     280      315532 :   av = avma; P = gel(myfactoru(N), 1); l = lg(P);
     281      751247 :   for (a = N, j = 1; j < l; j++) a += a / P[j];
     282      315532 :   return gc_long(av, a);
     283             : }
     284             : /* write n = mf^2. Return m, set f. */
     285             : static ulong
     286          72 : mycore(ulong n, long *pf)
     287             : {
     288          72 :   pari_sp av = avma;
     289          72 :   GEN fa = myfactoru(n), P = gel(fa,1), E = gel(fa,2);
     290          72 :   long i, l = lg(P), m = 1, f = 1;
     291         275 :   for (i = 1; i < l; i++)
     292             :   {
     293         203 :     long j, p = P[i], e = E[i];
     294         203 :     if (e & 1) m *= p;
     295         462 :     for (j = 2; j <= e; j+=2) f *= p;
     296             :   }
     297          72 :   *pf = f; return gc_long(av,m);
     298             : }
     299             : 
     300             : /* fa = factorization of -D > 0, return -D0 > 0 (where D0 is fundamental) */
     301             : static long
     302     4761947 : corediscs_fact(GEN fa)
     303             : {
     304     4761947 :   GEN P = gel(fa,1), E = gel(fa,2);
     305     4761947 :   long i, l = lg(P), m = 1;
     306    15737672 :   for (i = 1; i < l; i++)
     307             :   {
     308    10975725 :     long p = P[i], e = E[i];
     309    10975725 :     if (e & 1) m *= p;
     310             :   }
     311     4761947 :   if ((m&3L) != 3) m <<= 2;
     312     4761947 :   return m;
     313             : }
     314             : static long
     315        7098 : mubeta(long n)
     316             : {
     317        7098 :   pari_sp av = avma;
     318        7098 :   GEN E = gel(myfactoru(n), 2);
     319        7098 :   long i, s = 1, l = lg(E);
     320       14735 :   for (i = 1; i < l; i++)
     321             :   {
     322        7637 :     long e = E[i];
     323        7637 :     if (e >= 3) return gc_long(av,0);
     324        7637 :     if (e == 1) s *= -2;
     325             :   }
     326        7098 :   return gc_long(av,s);
     327             : }
     328             : 
     329             : /* n = n1*n2, n1 = ppo(n, m); return mubeta(n1)*moebiusu(n2).
     330             :  * N.B. If n from newt_params we, in fact, never return 0 */
     331             : static long
     332     7850904 : mubeta2(long n, long m)
     333             : {
     334     7850904 :   pari_sp av = avma;
     335     7850904 :   GEN fa = myfactoru(n), P = gel(fa,1), E = gel(fa,2);
     336     7850904 :   long i, s = 1, l = lg(P);
     337    15734522 :   for (i = 1; i < l; i++)
     338             :   {
     339     7883618 :     long p = P[i], e = E[i];
     340     7883618 :     if (m % p)
     341             :     { /* p^e in n1 */
     342     6669223 :       if (e >= 3) return gc_long(av,0);
     343     6669223 :       if (e == 1) s *= -2;
     344             :     }
     345             :     else
     346             :     { /* in n2 */
     347     1214395 :       if (e >= 2) return gc_long(av,0);
     348     1214395 :       s = -s;
     349             :     }
     350             :   }
     351     7850904 :   return gc_long(av,s);
     352             : }
     353             : 
     354             : /* write N = prod p^{ep} and n = df^2, d squarefree.
     355             :  * set g  = ppo(gcd(sqfpart(N), f), FC)
     356             :  *     N2 = prod p^if(e==1 || p|n, ep-1, ep-2) */
     357             : static void
     358     1941594 : newt_params(long N, long n, long FC, long *pg, long *pN2)
     359             : {
     360     1941594 :   GEN fa = myfactoru(N), P = gel(fa,1), E = gel(fa,2);
     361     1941594 :   long i, g = 1, N2 = 1, l = lg(P);
     362     5163782 :   for (i = 1; i < l; i++)
     363             :   {
     364     3222188 :     long p = P[i], e = E[i];
     365     3222188 :     if (e == 1)
     366     2825389 :     { if (FC % p && n % (p*p) == 0) g *= p; }
     367             :     else
     368      396799 :       N2 *= upowuu(p,(n % p)? e-2: e-1);
     369             :   }
     370     1941594 :   *pg = g; *pN2 = N2;
     371     1941594 : }
     372             : /* simplified version of newt_params for n = 1 (newdim) */
     373             : static void
     374       42525 : newd_params(long N, long *pN2)
     375             : {
     376       42525 :   GEN fa = myfactoru(N), P = gel(fa,1), E = gel(fa,2);
     377       42525 :   long i, N2 = 1, l = lg(P);
     378      106092 :   for (i = 1; i < l; i++)
     379             :   {
     380       63567 :     long p = P[i], e = E[i];
     381       63567 :     if (e > 2) N2 *= upowuu(p, e-2);
     382             :   }
     383       42525 :   *pN2 = N2;
     384       42525 : }
     385             : 
     386             : static long
     387          21 : newd_params2(long N)
     388             : {
     389          21 :   GEN fa = myfactoru(N), P = gel(fa,1), E = gel(fa,2);
     390          21 :   long i, N2 = 1, l = lg(P);
     391          56 :   for (i = 1; i < l; i++)
     392             :   {
     393          35 :     long p = P[i], e = E[i];
     394          35 :     if (e >= 2) N2 *= upowuu(p, e);
     395             :   }
     396          21 :   return N2;
     397             : }
     398             : 
     399             : /*******************************************************************/
     400             : /*   Relative trace between cyclotomic fields (TODO: export this)  */
     401             : /*******************************************************************/
     402             : /* g>=1; return g * prod_{p | g, (p,q) = 1} (1-1/p) */
     403             : static long
     404       36869 : phipart(long g, long q)
     405             : {
     406       36869 :   if (g > 1)
     407             :   {
     408       19670 :     GEN P = gel(myfactoru(g), 1);
     409       19670 :     long i, l = lg(P);
     410       40194 :     for (i = 1; i < l; i++) { long p = P[i]; if (q % p) g -= g / p; }
     411             :   }
     412       36869 :   return g;
     413             : }
     414             : /* Set s,v s.t. Trace(zeta_N^k) from Q(zeta_N) to Q(\zeta_N) = s * zeta_M^v
     415             :  * With k > 0, N = M*d and N, M != 2 mod 4 */
     416             : static long
     417       84756 : tracerelz(long *pv, long d, long M, long k)
     418             : {
     419             :   long s, g, q, muq;
     420       84756 :   if (d == 1) { *pv = k; return 1; }
     421       65618 :   *pv = 0; g = ugcd(k, d); q = d / g;
     422       65618 :   muq = mymoebiusu(q); if (!muq) return 0;
     423       47173 :   if (M != 1)
     424             :   {
     425       37828 :     long v = Fl_invsafe(q % M, M);
     426       37828 :     if (!v) return 0;
     427       27524 :     *pv = (v * (k/g)) % M;
     428             :   }
     429       36869 :   s = phipart(g, M*q); if (muq < 0) s = -s;
     430       36869 :   return s;
     431             : }
     432             : /* Pi = polcyclo(i), i = m or n. Let Ki = Q(zeta_i), initialize Tr_{Kn/Km} */
     433             : GEN
     434       34062 : Qab_trace_init(long n, long m, GEN Pn, GEN Pm)
     435             : {
     436             :   long a, i, j, N, M, vt, d, D;
     437             :   GEN T, G;
     438             : 
     439       34062 :   if (m == n || n <= 2) return mkvec(Pm);
     440       16555 :   vt = varn(Pn);
     441       16555 :   d = degpol(Pn);
     442             :   /* if (N != n) zeta_N = zeta_n^2 and zeta_n = - zeta_N^{(N+1)/2} */
     443       16555 :   N = ((n & 3) == 2)? n >> 1: n;
     444       16555 :   M = ((m & 3) == 2)? m >> 1: m; /* M | N | n */
     445       16555 :   a = N / M;
     446       16555 :   T = const_vec(d, NULL);
     447       16555 :   D = d / degpol(Pm); /* relative degree */
     448       16555 :   if (D == 1) G = NULL;
     449             :   else
     450             :   { /* zeta_M = zeta_n^A; s_j(zeta_M) = zeta_M <=> j = 1 (mod J) */
     451       15281 :     long lG, A = (N == n)? a: (a << 1), J = n / ugcd(n, A);
     452       15281 :     G = coprimes_zv(n);
     453      150276 :     for (j = lG = 1; j < n; j += J)
     454      134995 :       if (G[j]) G[lG++] = j;
     455       15281 :     setlg(G, lG); /* Gal(Q(zeta_n) / Q(zeta_m)) */
     456             :   }
     457       16555 :   T = const_vec(d, NULL);
     458       16555 :   gel(T,1) = utoipos(D); /* Tr 1 */
     459      140140 :   for (i = 1; i < d; i++)
     460             :   { /* if n = 2N, zeta_n^i = (-1)^i zeta_N^k */
     461             :     long s, v, k;
     462             :     GEN t;
     463             : 
     464      123585 :     if (gel(T, i+1)) continue;
     465       84756 :     k = (N == n)? i: ((odd(i)? i + N: i) >> 1);
     466       84756 :     if ((s = tracerelz(&v, a, M, k)))
     467             :     {
     468       56007 :       if (m != M) v *= 2;/* Tr = s * zeta_m^v */
     469       56007 :       if (n != N && odd(i)) s = -s;
     470       56007 :       t = Qab_Czeta(v, m, stoi(s), vt);
     471             :     }
     472             :     else
     473       28749 :       t = gen_0;
     474             :     /* t = Tr_{Kn/Km} zeta_n^i; fill using Galois action */
     475       84756 :     if (!G)
     476       19138 :       gel(T, i + 1) = t;
     477             :     else
     478      370874 :       for (j = 1; j <= D; j++)
     479             :       {
     480      305256 :         long z = Fl_mul(i,G[j], n);
     481      305256 :         if (z < d) gel(T, z + 1) = t;
     482             :       }
     483             :   }
     484       16555 :   return mkvec3(Pm, Pn, T);
     485             : }
     486             : /* x a t_POL modulo Phi_n */
     487             : static GEN
     488       80255 : tracerel_i(GEN T, GEN x)
     489             : {
     490       80255 :   long k, l = lg(x);
     491             :   GEN S;
     492       80255 :   if (l == 2) return gen_0;
     493       80255 :   S = gmul(gel(T,1), gel(x,2));
     494      283290 :   for (k = 3; k < l; k++) S = gadd(S, gmul(gel(T,k-1), gel(x,k)));
     495       80255 :   return S;
     496             : }
     497             : static GEN
     498      253855 : tracerel(GEN a, GEN v, GEN z)
     499             : {
     500      253855 :   a = liftpol_shallow(a);
     501      253855 :   a = simplify_shallow(z? gmul(z,a): a);
     502      253855 :   if (typ(a) == t_POL)
     503             :   {
     504       80255 :     GEN T = gel(v,3);
     505       80255 :     long degrel = itou(gel(T,1));
     506       80255 :     a = tracerel_i(T, RgX_rem(a, gel(v,2)));
     507       80255 :     if (degrel != 1) a = gdivgu(a, degrel);
     508       80255 :     if (typ(a) == t_POL) a = RgX_rem(a, gel(v,1));
     509             :   }
     510      253855 :   return a;
     511             : }
     512             : static GEN
     513        6944 : tracerel_z(GEN v, long t)
     514             : {
     515        6944 :   GEN Pn = gel(v,2);
     516        6944 :   return t? pol_xn(t, varn(Pn)): NULL;
     517             : }
     518             : /* v = Qab_trace_init(n,m); x is a t_VEC of polmodulo Phi_n; Kn = Q(zeta_n)
     519             :  * [Kn:Km]^(-1) Tr_{Kn/Km} (zeta_n^t * x); 0 <= t < [Kn:Km] */
     520             : GEN
     521           0 : Qab_tracerel(GEN v, long t, GEN a)
     522             : {
     523           0 :   if (lg(v) != 4) return a; /* => t = 0 */
     524           0 :   return tracerel(a, v, tracerel_z(v, t));
     525             : }
     526             : GEN
     527       16198 : QabV_tracerel(GEN v, long t, GEN x)
     528             : {
     529             :   GEN z;
     530       16198 :   if (lg(v) != 4) return x; /* => t = 0 */
     531        6944 :   z = tracerel_z(v, t);
     532      260799 :   pari_APPLY_same(tracerel(gel(x,i), v, z));
     533             : }
     534             : GEN
     535         154 : QabM_tracerel(GEN v, long t, GEN x)
     536             : {
     537         154 :   if (lg(v) != 4) return x;
     538         105 :   pari_APPLY_same(QabV_tracerel(v, t, gel(x,i)));
     539             : }
     540             : 
     541             : /* C*zeta_o^k mod X^o - 1 */
     542             : static GEN
     543     2247966 : Qab_Czeta(long k, long o, GEN C, long vt)
     544             : {
     545     2247966 :   if (!k) return C;
     546     1485694 :   if (!odd(o))
     547             :   { /* optimization: reduce max degree by a factor 2 for free */
     548     1434587 :     o >>= 1;
     549     1434587 :     if (k >= o) { k -= o; C = gneg(C); if (!k) return C; }
     550             :   }
     551     1137486 :   return monomial(C, k, vt);
     552             : }
     553             : /* zeta_o^k */
     554             : static GEN
     555      200767 : Qab_zeta(long k, long o, long vt) { return Qab_Czeta(k, o, gen_1, vt); }
     556             : 
     557             : /*              Operations on Dirichlet characters                       */
     558             : 
     559             : /* A Dirichlet character can be given in GP in different formats, but in this
     560             :  * package, it will be a vector CHI=[G,chi,ord,pol], where G is the (Z/MZ)^* to
     561             :  * which the character belongs, chi is the character in Conrey format, ord is
     562             :  * the order, and pol is polcyclo(ord,'t). */
     563             : 
     564             : static GEN
     565     3876236 : gmfcharorder(GEN CHI) { return gel(CHI, 3); }
     566             : long
     567     3817786 : mfcharorder(GEN CHI) { return itou(gmfcharorder(CHI)); }
     568             : static long
     569        2709 : mfcharistrivial(GEN CHI) { return !CHI || mfcharorder(CHI) == 1; }
     570             : static GEN
     571     1619786 : gmfcharmodulus(GEN CHI) { return gmael3(CHI, 1, 1, 1); }
     572             : long
     573     1619786 : mfcharmodulus(GEN CHI) { return itou(gmfcharmodulus(CHI)); }
     574             : GEN
     575      599354 : mfcharpol(GEN CHI) { return gel(CHI,4); }
     576             : 
     577             : /* vz[i+1] = image of (zeta_o)^i in Fp */
     578             : static ulong
     579      313040 : Qab_Czeta_Fl(long k, GEN vz, ulong C, ulong p)
     580             : {
     581             :   long o;
     582      313040 :   if (!k) return C;
     583      205982 :   o = lg(vz)-2;
     584      205982 :   if ((k << 1) == o) return Fl_neg(C,p);
     585      179053 :   return Fl_mul(C, vz[k+1], p);
     586             : }
     587             : 
     588             : static long
     589     2556365 : znchareval_i(GEN CHI, long n, GEN ord)
     590     2556365 : { return itos(znchareval(gel(CHI,1), gel(CHI,2), stoi(n), ord)); }
     591             : 
     592             : /* n coprime with the modulus of CHI */
     593             : static GEN
     594       14553 : mfchareval(GEN CHI, long n)
     595             : {
     596       14553 :   GEN Pn, C, go = gmfcharorder(CHI);
     597       14553 :   long k, o = go[2];
     598       14553 :   if (o == 1) return gen_1;
     599        7399 :   k = znchareval_i(CHI, n, go);
     600        7399 :   Pn = mfcharpol(CHI);
     601        7399 :   C = Qab_zeta(k, o, varn(Pn));
     602        7399 :   if (typ(C) != t_POL) return C;
     603        5327 :   return gmodulo(C, Pn);
     604             : }
     605             : /* d a multiple of ord(CHI); n coprime with char modulus;
     606             :  * return x s.t. CHI(n) = \zeta_d^x] */
     607             : static long
     608     3675462 : mfcharevalord(GEN CHI, long n, long d)
     609             : {
     610     3675462 :   if (mfcharorder(CHI) == 1) return 0;
     611     2545270 :   return znchareval_i(CHI, n, utoi(d));
     612             : }
     613             : 
     614             : /* G a znstar, L a Conrey log: return a 'mfchar' */
     615             : static GEN
     616      378812 : mfcharGL(GEN G, GEN L)
     617             : {
     618      378812 :   GEN o = zncharorder(G,L);
     619      378812 :   long ord = itou(o), vt = fetch_user_var("t");
     620      378812 :   return mkvec4(G, L, o, polcyclo(ord,vt));
     621             : }
     622             : static GEN
     623        5859 : mfchartrivial()
     624        5859 : { return mfcharGL(znstar0(gen_1,1), cgetg(1,t_COL)); }
     625             : /* convert a generic character into an 'mfchar' */
     626             : static GEN
     627        4074 : get_mfchar(GEN CHI)
     628             : {
     629             :   GEN G, L;
     630        4074 :   if (typ(CHI) != t_VEC) CHI = znchar(CHI);
     631             :   else
     632             :   {
     633         889 :     long l = lg(CHI);
     634         889 :     if ((l != 3 && l != 5) || !checkznstar_i(gel(CHI,1)))
     635           7 :       pari_err_TYPE("checkNF [chi]", CHI);
     636         882 :     if (l == 5) return CHI;
     637             :   }
     638        4004 :   G = gel(CHI,1);
     639        4004 :   L = gel(CHI,2); if (typ(L) != t_COL) L = znconreylog(G,L);
     640        4004 :   return mfcharGL(G, L);
     641             : }
     642             : 
     643             : /* parse [N], [N,k], [N,k,CHI]. If 'joker' is set, allow wildcard for CHI */
     644             : static GEN
     645        9247 : checkCHI(GEN NK, long N, int joker)
     646             : {
     647             :   GEN CHI;
     648        9247 :   if (lg(NK) == 3)
     649         742 :     CHI = mfchartrivial();
     650             :   else
     651             :   {
     652             :     long i, l;
     653        8505 :     CHI = gel(NK,3); l = lg(CHI);
     654        8505 :     if (isintzero(CHI) && joker)
     655        4116 :       CHI = NULL; /* all character orbits */
     656        4389 :     else if (isintm1(CHI) && joker > 1)
     657        2373 :       CHI = gen_m1; /* sum over all character orbits */
     658        2016 :     else if ((typ(CHI) == t_VEC &&
     659         217 :              (l == 1 || l != 3 || !checkznstar_i(gel(CHI,1)))) && joker)
     660             :     {
     661         133 :       CHI = shallowtrans(CHI); /* list of characters */
     662         952 :       for (i = 1; i < l; i++) gel(CHI,i) = get_mfchar(gel(CHI,i));
     663             :     }
     664             :     else
     665             :     {
     666        1883 :       CHI = get_mfchar(CHI); /* single char */
     667        1883 :       if (N % mfcharmodulus(CHI)) pari_err_TYPE("checkNF [chi]", NK);
     668             :     }
     669             :   }
     670        9233 :   return CHI;
     671             : }
     672             : /* support half-integral weight */
     673             : static void
     674        9254 : checkNK2(GEN NK, long *N, long *nk, long *dk, GEN *CHI, int joker)
     675             : {
     676        9254 :   long l = lg(NK);
     677             :   GEN T;
     678        9254 :   if (typ(NK) != t_VEC || l < 3 || l > 4) pari_err_TYPE("checkNK", NK);
     679        9254 :   T = gel(NK,1); if (typ(T) != t_INT) pari_err_TYPE("checkNF [N]", NK);
     680        9254 :   *N = itos(T); if (*N <= 0) pari_err_TYPE("checkNF [N <= 0]", NK);
     681        9254 :   T = gel(NK,2);
     682        9254 :   switch(typ(T))
     683             :   {
     684        5866 :     case t_INT:  *nk = itos(T); *dk = 1; break;
     685        3381 :     case t_FRAC:
     686        3381 :       *nk = itos(gel(T,1));
     687        3381 :       *dk = itou(gel(T,2)); if (*dk == 2) break;
     688           7 :     default: pari_err_TYPE("checkNF [k]", NK);
     689             :   }
     690        9247 :   *CHI = checkCHI(NK, *N, joker);
     691        9233 : }
     692             : /* don't support half-integral weight */
     693             : static void
     694         133 : checkNK(GEN NK, long *N, long *k, GEN *CHI, int joker)
     695             : {
     696             :   long d;
     697         133 :   checkNK2(NK, N, k, &d, CHI, joker);
     698         133 :   if (d != 1) pari_err_TYPE("checkNF [k]", NK);
     699         133 : }
     700             : 
     701             : static GEN
     702        4872 : mfchargalois(long N, int odd, GEN flagorder)
     703             : {
     704        4872 :   GEN G = znstar0(utoi(N), 1), L = chargalois(G, flagorder);
     705        4872 :   long l = lg(L), i, j;
     706      113526 :   for (i = j = 1; i < l; i++)
     707             :   {
     708      108654 :     GEN chi = znconreyfromchar(G, gel(L,i));
     709      108654 :     if (zncharisodd(G,chi) == odd) gel(L,j++) = mfcharGL(G,chi);
     710             :   }
     711        4872 :   setlg(L, j); return L;
     712             : }
     713             : /* possible characters for nontrivial S_1(N, chi) */
     714             : static GEN
     715        1729 : mf1chars(long N, GEN vCHI)
     716             : {
     717        1729 :   if (vCHI) return vCHI; /*do not filter, user knows best*/
     718             :   /* Tate's theorem */
     719        1659 :   return mfchargalois(N, 1, uisprime(N)? mkvecsmall2(2,4): NULL);
     720             : }
     721             : static GEN
     722        3255 : mfchars(long N, long k, long dk, GEN vCHI)
     723        3255 : { return vCHI? vCHI: mfchargalois(N, (dk == 2)? 0: (k & 1), NULL); }
     724             : 
     725             : /* wrappers from mfchar to znchar */
     726             : static long
     727       68670 : mfcharparity(GEN CHI)
     728             : {
     729       68670 :   if (!CHI) return 1;
     730       68670 :   return zncharisodd(gel(CHI,1), gel(CHI,2)) ? -1 : 1;
     731             : }
     732             : /* if CHI is primitive, return CHI itself, not a copy */
     733             : static GEN
     734       82334 : mfchartoprimitive(GEN CHI, long *pF)
     735             : {
     736             :   pari_sp av;
     737             :   GEN chi, F;
     738       82334 :   if (!CHI) { if (pF) *pF = 1; return mfchartrivial(); }
     739       82334 :   av = avma; F = znconreyconductor(gel(CHI,1), gel(CHI,2), &chi);
     740       82334 :   if (typ(F) == t_INT) set_avma(av);
     741             :   else
     742             :   {
     743        7875 :     CHI = leafcopy(CHI);
     744        7875 :     gel(CHI,1) = znstar0(F, 1);
     745        7875 :     gel(CHI,2) = chi;
     746             :   }
     747       82334 :   if (pF) *pF = mfcharmodulus(CHI);
     748       82334 :   return CHI;
     749             : }
     750             : static long
     751      397950 : mfcharconductor(GEN CHI)
     752             : {
     753      397950 :   pari_sp av = avma;
     754      397950 :   GEN res = znconreyconductor(gel(CHI,1), gel(CHI,2), NULL);
     755      397950 :   if (typ(res) == t_VEC) res = gel(res, 1);
     756      397950 :   return gc_long(av, itos(res));
     757             : }
     758             : 
     759             : /*                      Operations on mf closures                    */
     760             : static GEN
     761       64526 : tagparams(long t, GEN NK) { return mkvec2(mkvecsmall(t), NK); }
     762             : static GEN
     763        1197 : lfuntag(long t, GEN x) { return mkvec2(mkvecsmall(t), x); }
     764             : static GEN
     765          56 : tag0(long t, GEN NK) { retmkvec(tagparams(t,NK)); }
     766             : static GEN
     767       10423 : tag(long t, GEN NK, GEN x) { retmkvec2(tagparams(t,NK), x); }
     768             : static GEN
     769       37499 : tag2(long t, GEN NK, GEN x, GEN y) { retmkvec3(tagparams(t,NK), x,y); }
     770             : static GEN
     771       16415 : tag3(long t, GEN NK, GEN x,GEN y,GEN z) { retmkvec4(tagparams(t,NK), x,y,z); }
     772             : static GEN
     773           0 : tag4(long t, GEN NK, GEN x,GEN y,GEN z,GEN a)
     774           0 : { retmkvec5(tagparams(t,NK), x,y,z,a); }
     775             : /* is F a "modular form" ? */
     776             : int
     777       19565 : checkmf_i(GEN F)
     778       19565 : { return typ(F) == t_VEC
     779       18718 :     && lg(F) > 1 && typ(gel(F,1)) == t_VEC
     780       13902 :     && lg(gel(F,1)) == 3
     781       13741 :     && typ(gmael(F,1,1)) == t_VECSMALL
     782       38283 :     && typ(gmael(F,1,2)) == t_VEC; }
     783      241108 : long mf_get_type(GEN F) { return gmael(F,1,1)[1]; }
     784      191968 : GEN mf_get_gN(GEN F) { return gmael3(F,1,2,1); }
     785      144627 : GEN mf_get_gk(GEN F) { return gmael3(F,1,2,2); }
     786             : /* k - 1/2, assume k in 1/2 + Z */
     787         441 : long mf_get_r(GEN F) { return itou(gel(mf_get_gk(F),1)) >> 1; }
     788      124019 : long mf_get_N(GEN F) { return itou(mf_get_gN(F)); }
     789       73731 : long mf_get_k(GEN F)
     790             : {
     791       73731 :   GEN gk = mf_get_gk(F);
     792       73731 :   if (typ(gk)!=t_INT) pari_err_IMPL("half-integral weight");
     793       73731 :   return itou(gk);
     794             : }
     795       65527 : GEN mf_get_CHI(GEN F) { return gmael3(F,1,2,3); }
     796       25081 : GEN mf_get_field(GEN F) { return gmael3(F,1,2,4); }
     797       19691 : GEN mf_get_NK(GEN F) { return gmael(F,1,2); }
     798             : static void
     799         588 : mf_setfield(GEN f, GEN P)
     800             : {
     801         588 :   gel(f,1) = leafcopy(gel(f,1));
     802         588 :   gmael(f,1,2) = leafcopy(gmael(f,1,2));
     803         588 :   gmael3(f,1,2,4) = P;
     804         588 : }
     805             : 
     806             : /* UTILITY FUNCTIONS */
     807             : GEN
     808        9121 : mftocol(GEN F, long lim, long d)
     809        9121 : { GEN c = mfcoefs_i(F, lim, d); settyp(c,t_COL); return c; }
     810             : GEN
     811        2135 : mfvectomat(GEN vF, long lim, long d)
     812             : {
     813        2135 :   long j, l = lg(vF);
     814        2135 :   GEN M = cgetg(l, t_MAT);
     815       10437 :   for (j = 1; j < l; j++) gel(M,j) = mftocol(gel(vF,j), lim, d);
     816        2135 :   return M;
     817             : }
     818             : 
     819             : static GEN
     820        4893 : RgV_to_ser_full(GEN x) { return RgV_to_ser(x, 0, lg(x)+1); }
     821             : /* TODO: delete */
     822             : static GEN
     823         679 : mfcoefsser(GEN F, long n) { return RgV_to_ser_full(mfcoefs_i(F,n,1)); }
     824             : static GEN
     825         847 : sertovecslice(GEN S, long n)
     826             : {
     827         847 :   GEN v = gtovec0(S, -(lg(S) - 2 + valser(S)));
     828         847 :   long l = lg(v), n2 = n + 2;
     829         847 :   if (l < n2) pari_err_BUG("sertovecslice [n too large]");
     830         847 :   return (l == n2)? v: vecslice(v, 1, n2-1);
     831             : }
     832             : 
     833             : /* a, b two RgV of the same length, multiply as truncated power series */
     834             : static GEN
     835        8869 : RgV_mul_RgXn(GEN a, GEN b)
     836             : {
     837        8869 :   long n = lg(a)-1;
     838             :   GEN c;
     839        8869 :   a = RgV_to_RgX(a,0);
     840        8869 :   b = RgV_to_RgX(b,0); c = RgXn_mul(a, b, n);
     841        8869 :   c = RgX_to_RgC(c,n); settyp(c,t_VEC); return c;
     842             : }
     843             : /* divide as truncated power series */
     844             : static GEN
     845         399 : RgV_div_RgXn(GEN a, GEN b)
     846             : {
     847         399 :   long n = lg(a)-1;
     848             :   GEN c;
     849         399 :   a = RgV_to_RgX(a,0);
     850         399 :   b = RgV_to_RgX(b,0); c = RgXn_div_i(a, b, n);
     851         399 :   c = RgX_to_RgC(c,n); settyp(c,t_VEC); return c;
     852             : }
     853             : /* a^b */
     854             : static GEN
     855         112 : RgV_pows_RgXn(GEN a, long b)
     856             : {
     857         112 :   long n = lg(a)-1;
     858             :   GEN c;
     859         112 :   a = RgV_to_RgX(a,0);
     860         112 :   if (b < 0) { a = RgXn_inv(a, n); b = -b; }
     861         112 :   c = RgXn_powu_i(a,b,n);
     862         112 :   c = RgX_to_RgC(c,n); settyp(c,t_VEC); return c;
     863             : }
     864             : 
     865             : /* assume lg(V) >= n*d + 2 */
     866             : static GEN
     867        8939 : c_deflate(long n, long d, GEN v)
     868             : {
     869        8939 :   long i, id, l = n+2;
     870             :   GEN w;
     871        8939 :   if (d == 1) return lg(v) == l ? v: vecslice(v, 1, l-1);
     872         581 :   w = cgetg(l, typ(v));
     873       11249 :   for (i = id = 1; i < l; i++, id += d) gel(w, i) = gel(v, id);
     874         581 :   return w;
     875             : }
     876             : 
     877             : static void
     878          14 : err_cyclo(void)
     879          14 : { pari_err_IMPL("changing cyclotomic fields in mf"); }
     880             : /* Q(zeta_a) = Q(zeta_b) ? */
     881             : static int
     882         616 : same_cyc(long a, long b)
     883         616 : { return (a == b) || (odd(a) && b == (a<<1)) || (odd(b) && a == (b<<1)); }
     884             : /* need to combine elements in Q(CHI1) and Q(CHI2) with result in Q(CHI),
     885             :  * CHI = CHI1 * CHI2 or CHI / CHI2 times some character of order 2 */
     886             : static GEN
     887        2835 : chicompat(GEN CHI, GEN CHI1, GEN CHI2)
     888             : {
     889        2835 :   long o1 = mfcharorder(CHI1);
     890        2835 :   long o2 = mfcharorder(CHI2), O, o;
     891             :   GEN T1, T2, P, Po;
     892        2835 :   if (o1 <= 2 && o2 <= 2) return NULL;
     893         623 :   o = mfcharorder(CHI);
     894         623 :   Po = mfcharpol(CHI);
     895         623 :   P = mfcharpol(CHI1);
     896         623 :   if (o1 == o2)
     897             :   {
     898          21 :     if (o1 == o) return NULL;
     899          14 :     if (!same_cyc(o1,o)) err_cyclo();
     900           0 :     return mkvec4(P, gen_1,gen_1, Qab_trace_init(o1, o, P, Po));
     901             :   }
     902         602 :   O = ulcm(o1, o2);
     903         602 :   if (!same_cyc(O,o)) err_cyclo();
     904         602 :   if (O != o1) P = (O == o2)? mfcharpol(CHI2): polcyclo(O, varn(P));
     905         602 :   T1 = o1 <= 2? gen_1: utoipos(O / o1);
     906         602 :   T2 = o2 <= 2? gen_1: utoipos(O / o2);
     907         602 :   return mkvec4(P, T1, T2, O == o? gen_1: Qab_trace_init(O, o, P, Po));
     908             : }
     909             : static GEN
     910          49 : inflatemod(GEN f, long o, GEN P)
     911             : {
     912          49 :   f = lift_shallow(f);
     913          49 :   return gmodulo(typ(f)==t_POL? RgX_inflate(f,o): f, P);
     914             : }
     915             : static GEN
     916           7 : RgV_inflatemod(GEN x, long o, GEN P)
     917          56 : { pari_APPLY_same(inflatemod(gel(x,i), o, P)); }
     918             : /* *F a vector of cyclotomic numbers */
     919             : static void
     920         651 : chicompatlift(GEN T, GEN *F, GEN *G)
     921             : {
     922         651 :   long o1 = itou(gel(T,2)), o2 = itou(gel(T,3));
     923         651 :   GEN P = gel(T,1);
     924         651 :   if (o1 != 1) *F = RgV_inflatemod(*F, o1, P);
     925         651 :   if (o2 != 1 && G) *G = RgV_inflatemod(*G, o2, P);
     926         651 : }
     927             : static GEN
     928         651 : chicompatfix(GEN T, GEN F)
     929             : {
     930         651 :   GEN V = gel(T,4);
     931         651 :   if (typ(V) == t_VEC) F = gmodulo(QabV_tracerel(V, 0, F), gel(V,1));
     932         651 :   return F;
     933             : }
     934             : 
     935             : static GEN
     936         637 : c_mul(long n, long d, GEN S)
     937             : {
     938         637 :   pari_sp av = avma;
     939         637 :   long nd = n*d;
     940         637 :   GEN F = gel(S,2), G = gel(S,3);
     941         637 :   F = mfcoefs_i(F, nd, 1);
     942         637 :   G = mfcoefs_i(G, nd, 1);
     943         637 :   if (lg(S) == 5) chicompatlift(gel(S,4),&F,&G);
     944         637 :   F = c_deflate(n, d, RgV_mul_RgXn(F,G));
     945         637 :   if (lg(S) == 5) F = chicompatfix(gel(S,4), F);
     946         637 :   return gc_GEN(av, F);
     947             : }
     948             : static GEN
     949         112 : c_pow(long n, long d, GEN S)
     950             : {
     951         112 :   pari_sp av = avma;
     952         112 :   long nd = n*d;
     953         112 :   GEN F = gel(S,2), a = gel(S,3), f = mfcoefs_i(F,nd,1);
     954         112 :   if (lg(S) == 5) chicompatlift(gel(S,4),&F, NULL);
     955         112 :   f = RgV_pows_RgXn(f, itos(a));
     956         112 :   f = c_deflate(n, d, f);
     957         112 :   if (lg(S) == 5) f = chicompatfix(gel(S,4), f);
     958         112 :   return gc_GEN(av, f);
     959             : }
     960             : 
     961             : /* F * Theta */
     962             : static GEN
     963         448 : mfmultheta(GEN F)
     964             : {
     965         448 :   if (typ(mf_get_gk(F)) == t_FRAC && mf_get_type(F) == t_MF_DIV)
     966             :   {
     967         154 :     GEN T = gel(F,3); /* hopefully mfTheta() */
     968         154 :     if (mf_get_type(T) == t_MF_THETA && mf_get_N(T) == 4) return gel(F,2);
     969             :   }
     970         294 :   return mfmul(F, mfTheta(NULL));
     971             : }
     972             : 
     973             : static GEN
     974          42 : c_bracket(long n, long d, GEN S)
     975             : {
     976          42 :   pari_sp av = avma;
     977          42 :   long i, nd = n*d;
     978          42 :   GEN F = gel(S,2), G = gel(S,3), tF, tG, C, mpow, res, gk, gl;
     979          42 :   GEN VF = mfcoefs_i(F, nd, 1);
     980          42 :   GEN VG = mfcoefs_i(G, nd, 1);
     981          42 :   ulong j, m = itou(gel(S,4));
     982             : 
     983          42 :   if (!n)
     984             :   {
     985          14 :     if (m > 0) { set_avma(av); return mkvec(gen_0); }
     986           7 :     return gc_GEN(av, mkvec(gmul(gel(VF, 1), gel(VG, 1))));
     987             :   }
     988          28 :   tF = cgetg(nd+2, t_VEC);
     989          28 :   tG = cgetg(nd+2, t_VEC);
     990          28 :   res = NULL; gk = mf_get_gk(F); gl = mf_get_gk(G);
     991             :   /* pow[i,j+1] = i^j */
     992          28 :   if (lg(S) == 6) chicompatlift(gel(S,5),&VF,&VG);
     993          28 :   mpow = cgetg(m+2, t_MAT);
     994          28 :   gel(mpow,1) = const_col(nd, gen_1);
     995          56 :   for (j = 1; j <= m; j++)
     996             :   {
     997          28 :     GEN c = cgetg(nd+1, t_COL);
     998          28 :     gel(mpow,j+1) = c;
     999         245 :     for (i = 1; i <= nd; i++) gel(c,i) = muliu(gcoeff(mpow,i,j), i);
    1000             :   }
    1001          28 :   C = binomial(gaddgs(gk, m-1), m);
    1002          28 :   if (odd(m)) C = gneg(C);
    1003          84 :   for (j = 0; j <= m; j++)
    1004             :   { /* C = (-1)^(m-j) binom(m+l-1, j) binom(m+k-1,m-j) */
    1005             :     GEN c;
    1006          56 :     gel(tF,1) = j == 0? gel(VF,1): gen_0;
    1007          56 :     gel(tG,1) = j == m? gel(VG,1): gen_0;
    1008          56 :     gel(tF,2) = gel(VF,2); /* assume nd >= 1 */
    1009          56 :     gel(tG,2) = gel(VG,2);
    1010         518 :     for (i = 2; i <= nd; i++)
    1011             :     {
    1012         462 :       gel(tF, i+1) = gmul(gcoeff(mpow,i,j+1),   gel(VF, i+1));
    1013         462 :       gel(tG, i+1) = gmul(gcoeff(mpow,i,m-j+1), gel(VG, i+1));
    1014             :     }
    1015          56 :     c = gmul(C, c_deflate(n, d, RgV_mul_RgXn(tF, tG)));
    1016          56 :     res = res? gadd(res, c): c;
    1017          56 :     if (j < m)
    1018          56 :       C = gdiv(gmul(C, gmulsg(m-j, gaddgs(gl,m-j-1))),
    1019          28 :                gmulsg(-(j+1), gaddgs(gk,j)));
    1020             :   }
    1021          28 :   if (lg(S) == 6) res = chicompatfix(gel(S,5), res);
    1022          28 :   return gc_upto(av, res);
    1023             : }
    1024             : /* linear combination \sum L[j] vecF[j] */
    1025             : static GEN
    1026        3024 : c_linear(long n, long d, GEN F, GEN L, GEN dL)
    1027             : {
    1028        3024 :   pari_sp av = avma;
    1029        3024 :   long j, l = lg(L);
    1030        3024 :   GEN S = NULL;
    1031       10780 :   for (j = 1; j < l; j++)
    1032             :   {
    1033        7756 :     GEN c = gel(L,j);
    1034        7756 :     if (gequal0(c)) continue;
    1035        7000 :     c = gmul(c, mfcoefs_i(gel(F,j), n, d));
    1036        7000 :     S = S? gadd(S,c): c;
    1037             :   }
    1038        3024 :   if (!S) return zerovec(n+1);
    1039        3024 :   if (!is_pm1(dL)) S = gdiv(S, dL);
    1040        3024 :   return gc_upto(av, S);
    1041             : }
    1042             : 
    1043             : /* B_d(T_j Trace^new) as t_MF_BD(t_MF_HECKE(t_MF_NEWTRACE)) or
    1044             :  * t_MF_HECKE(t_MF_NEWTRACE)
    1045             :  * or t_MF_NEWTRACE in level N. Set d and j, return t_MF_NEWTRACE component*/
    1046             : static GEN
    1047       84665 : bhn_parse(GEN f, long *d, long *j)
    1048             : {
    1049       84665 :   long t = mf_get_type(f);
    1050       84665 :   *d = *j = 1;
    1051       84665 :   if (t == t_MF_BD) { *d = itos(gel(f,3)); f = gel(f,2); t = mf_get_type(f); }
    1052       84665 :   if (t == t_MF_HECKE) { *j = gel(f,2)[1]; f = gel(f,3); }
    1053       84665 :   return f;
    1054             : }
    1055             : /* f as above, return the t_MF_NEWTRACE component */
    1056             : static GEN
    1057       34993 : bhn_newtrace(GEN f)
    1058             : {
    1059       34993 :   long t = mf_get_type(f);
    1060       34993 :   if (t == t_MF_BD) { f = gel(f,2); t = mf_get_type(f); }
    1061       34993 :   if (t == t_MF_HECKE) f = gel(f,3);
    1062       34993 :   return f;
    1063             : }
    1064             : static int
    1065        4144 : ok_bhn_linear(GEN vf)
    1066             : {
    1067        4144 :   long i, N0 = 0, l = lg(vf);
    1068             :   GEN CHI, gk;
    1069        4144 :   if (l == 1) return 1;
    1070        4144 :   gk = mf_get_gk(gel(vf,1));
    1071        4144 :   CHI = mf_get_CHI(gel(vf,1));
    1072       29792 :   for (i = 1; i < l; i++)
    1073             :   {
    1074       28007 :     GEN f = bhn_newtrace(gel(vf,i));
    1075       28007 :     long N = mf_get_N(f);
    1076       28007 :     if (mf_get_type(f) != t_MF_NEWTRACE) return 0;
    1077       25648 :     if (N < N0) return 0; /* largest level must come last */
    1078       25648 :     N0 = N;
    1079       25648 :     if (!gequal(gk,mf_get_gk(f))) return 0; /* same k */
    1080       25648 :     if (!gequal(gel(mf_get_CHI(f),2), gel(CHI,2))) return 0; /* same CHI */
    1081             :   }
    1082        1785 :   return 1;
    1083             : }
    1084             : 
    1085             : /* vF not empty, same hypotheses as bhnmat_extend */
    1086             : static GEN
    1087        7091 : bhnmat_extend_nocache(GEN M, long N, long n, long d, GEN vF)
    1088             : {
    1089             :   cachenew_t cache;
    1090        7091 :   long l = lg(vF);
    1091             :   GEN f;
    1092        7091 :   if (l == 1) return M? M: cgetg(1, t_MAT);
    1093        6986 :   f = bhn_newtrace(gel(vF,1)); /* N.B. mf_get_N(f) divides N */
    1094        6986 :   init_cachenew(&cache, n*d, N, f);
    1095        6986 :   M = bhnmat_extend(M, n, d, vF, &cache);
    1096        6986 :   dbg_cachenew(&cache); return M;
    1097             : }
    1098             : /* c_linear of "bhn" mf closures, same hypotheses as bhnmat_extend */
    1099             : static GEN
    1100        2331 : c_linear_bhn(long n, long d, GEN F)
    1101             : {
    1102             :   pari_sp av;
    1103        2331 :   GEN M, v, vF = gel(F,2), L = gel(F,3), dL = gel(F,4);
    1104        2331 :   if (lg(L) == 1) return zerovec(n+1);
    1105        2331 :   av = avma;
    1106        2331 :   M = bhnmat_extend_nocache(NULL, mf_get_N(F), n, d, vF);
    1107        2331 :   v = RgM_RgC_mul(M,L); settyp(v, t_VEC);
    1108        2331 :   if (!is_pm1(dL)) v = gdiv(v, dL);
    1109        2331 :   return gc_upto(av, v);
    1110             : }
    1111             : 
    1112             : /* c in K, K := Q[X]/(T) vz = vector of consecutive powers of root z of T
    1113             :  * attached to an embedding s: K -> C. Return s(c) in C */
    1114             : static GEN
    1115       84658 : Rg_embed1(GEN c, GEN vz)
    1116             : {
    1117       84658 :   long t = typ(c);
    1118       84658 :   if (t == t_POLMOD) { c = gel(c,2); t = typ(c); }
    1119       84658 :   if (t == t_POL) c = RgX_RgV_eval(c, vz);
    1120       84658 :   return c;
    1121             : }
    1122             : /* return s(x) in C[X] */
    1123             : static GEN
    1124       14203 : RgX_embed1(GEN x, GEN vz)
    1125       42042 : { pari_APPLY_pol(Rg_embed1(gel(x,i), vz)); }
    1126             : /* return s(x) in C^n */
    1127             : static GEN
    1128         798 : vecembed1(GEN x, GEN vz)
    1129       39858 : { pari_APPLY_same(Rg_embed1(gel(x,i), vz)); }
    1130             : /* P in L = K[X]/(U), K = Q[t]/T; s an embedding of K -> C attached
    1131             :  * to a root of T, extended to an embedding of L -> C attached to a root
    1132             :  * of s(U); vT powers of the root of T, vU powers of the root of s(U).
    1133             :  * Return s(P) in C^n */
    1134             : static GEN
    1135       13328 : Rg_embed2(GEN P, long vt, GEN vT, GEN vU)
    1136             : {
    1137       13328 :   P = liftpol_shallow(P);
    1138       13328 :   if (typ(P) != t_POL) return P;
    1139       13300 :   if (varn(P) == vt) return Rg_embed1(P, vT);
    1140       13293 :   return Rg_embed1(RgX_embed1(P, vT), vU); /* varn(P) == vx */
    1141             : }
    1142             : static GEN
    1143          42 : vecembed2(GEN x, long vt, GEN vT, GEN vU)
    1144        1050 : { pari_APPLY_same(Rg_embed2(gel(x,i), vt, vT, vU)); }
    1145             : static GEN
    1146         532 : RgX_embed2(GEN x, long vt, GEN vT, GEN vU)
    1147        3724 : { pari_APPLY_pol(Rg_embed2(gel(x,i), vt, vT, vU)); }
    1148             : /* embed polynomial f in variable 0 [ may be a scalar ], E from getembed */
    1149             : static GEN
    1150        1687 : RgX_embed(GEN f, GEN E)
    1151             : {
    1152             :   GEN vT;
    1153        1687 :   if (typ(f) != t_POL || varn(f) != 0) return mfembed(E, f);
    1154        1645 :   if (lg(E) == 1) return f;
    1155        1407 :   vT = gel(E,2);
    1156        1407 :   if (lg(E) == 3)
    1157         875 :     f = RgX_embed1(f, vT);
    1158             :   else
    1159         532 :     f = RgX_embed2(f, varn(gel(E,1)), vT, gel(E,3));
    1160        1407 :   return f;
    1161             : }
    1162             : /* embed vector, E from getembed */
    1163             : GEN
    1164        1743 : mfvecembed(GEN E, GEN v)
    1165             : {
    1166             :   GEN vT;
    1167        1743 :   if (lg(E) == 1) return v;
    1168         840 :   vT = gel(E,2);
    1169         840 :   if (lg(E) == 3)
    1170         798 :     v = vecembed1(v, vT);
    1171             :   else
    1172          42 :     v = vecembed2(v, varn(gel(E,1)), vT, gel(E,3));
    1173         840 :   return v;
    1174             : }
    1175             : GEN
    1176          70 : mfmatembed(GEN E, GEN x)
    1177             : {
    1178          70 :   if (lg(E) == 1) return x;
    1179         168 :   pari_APPLY_same(mfvecembed(E, gel(x,i)));
    1180             : }
    1181             : /* embed vector of polynomials in var 0 */
    1182             : static GEN
    1183          98 : RgXV_embed(GEN x, GEN E)
    1184             : {
    1185          98 :   if (lg(E) == 1) return x;
    1186        1358 :   pari_APPLY_same(RgX_embed(gel(x,i), E));
    1187             : }
    1188             : 
    1189             : /* embed scalar */
    1190             : GEN
    1191      100845 : mfembed(GEN E, GEN f)
    1192             : {
    1193             :   GEN vT;
    1194      100845 :   if (lg(E) == 1) return f;
    1195       13587 :   vT = gel(E,2);
    1196       13587 :   if (lg(E) == 3)
    1197        4459 :     f = Rg_embed1(f, vT);
    1198             :   else
    1199        9128 :     f = Rg_embed2(f, varn(gel(E,1)), vT, gel(E,3));
    1200       13587 :   return f;
    1201             : }
    1202             : /* vector of the sigma(f), sigma in vE */
    1203             : static GEN
    1204         364 : RgX_embedall(GEN f, GEN vE)
    1205             : {
    1206         364 :   long i, l = lg(vE);
    1207             :   GEN v;
    1208         364 :   if (l == 2) return RgX_embed(f, gel(vE,1));
    1209          35 :   v = cgetg(l, t_VEC);
    1210         105 :   for (i = 1; i < l; i++) gel(v,i) = RgX_embed(f, gel(vE,i));
    1211          35 :   return v;
    1212             : }
    1213             : /* matrix whose colums are the sigma(v), sigma in vE */
    1214             : static GEN
    1215         350 : RgC_embedall(GEN v, GEN vE)
    1216             : {
    1217         350 :   long j, l = lg(vE);
    1218         350 :   GEN M = cgetg(l, t_MAT);
    1219         875 :   for (j = 1; j < l; j++) gel(M,j) = mfvecembed(gel(vE,j), v);
    1220         350 :   return M;
    1221             : }
    1222             : /* vector of the sigma(v), sigma in vE */
    1223             : static GEN
    1224        4907 : Rg_embedall_i(GEN v, GEN vE)
    1225             : {
    1226        4907 :   long j, l = lg(vE);
    1227        4907 :   GEN M = cgetg(l, t_VEC);
    1228       14735 :   for (j = 1; j < l; j++) gel(M,j) = mfembed(gel(vE,j), v);
    1229        4907 :   return M;
    1230             : }
    1231             : /* vector of the sigma(v), sigma in vE; if #vE == 1, return v */
    1232             : static GEN
    1233       95154 : Rg_embedall(GEN v, GEN vE)
    1234       95154 : { return (lg(vE) == 2)? mfembed(gel(vE,1), v): Rg_embedall_i(v, vE); }
    1235             : 
    1236             : static GEN
    1237         847 : c_div_i(long n, GEN S)
    1238             : {
    1239         847 :   GEN F = gel(S,2), G = gel(S,3);
    1240             :   GEN a0, a0i, H;
    1241         847 :   F = mfcoefs_i(F, n, 1);
    1242         847 :   G = mfcoefs_i(G, n, 1);
    1243         847 :   if (lg(S) == 5) chicompatlift(gel(S,4),&F,&G);
    1244         847 :   F = RgV_to_ser_full(F);
    1245         847 :   G = RgV_to_ser_full(G);
    1246         847 :   a0 = polcoef_i(G, 0, -1); /* != 0 */
    1247         847 :   if (gequal1(a0)) a0 = a0i = NULL;
    1248             :   else
    1249             :   {
    1250         602 :     a0i = ginv(a0);
    1251         602 :     G = gmul(ser_unscale(G,a0), a0i);
    1252         602 :     F = gmul(ser_unscale(F,a0), a0i);
    1253             :   }
    1254         847 :   H = gdiv(F, G);
    1255         847 :   if (a0) H = ser_unscale(H,a0i);
    1256         847 :   H = sertovecslice(H, n);
    1257         847 :   if (lg(S) == 5) H = chicompatfix(gel(S,4), H);
    1258         847 :   return H;
    1259             : }
    1260             : static GEN
    1261         847 : c_div(long n, long d, GEN S)
    1262             : {
    1263         847 :   pari_sp av = avma;
    1264         847 :   GEN D = (d==1)? c_div_i(n, S): c_deflate(n, d, c_div_i(n*d, S));
    1265         847 :   return gc_GEN(av, D);
    1266             : }
    1267             : 
    1268             : static GEN
    1269          35 : c_shift(long n, long d, GEN F, GEN gsh)
    1270             : {
    1271          35 :   pari_sp av = avma;
    1272             :   GEN vF;
    1273          35 :   long sh = itos(gsh), n1 = n*d + sh;
    1274          35 :   if (n1 < 0) return zerovec(n+1);
    1275          35 :   vF = mfcoefs_i(F, n1, 1);
    1276          35 :   if (sh < 0) vF = shallowconcat(zerovec(-sh), vF);
    1277          35 :   else vF = vecslice(vF, sh+1, n1+1);
    1278          35 :   return gc_GEN(av, c_deflate(n, d, vF));
    1279             : }
    1280             : 
    1281             : static GEN
    1282         175 : c_deriv(long n, long d, GEN F, GEN gm)
    1283             : {
    1284         175 :   pari_sp av = avma;
    1285         175 :   GEN V = mfcoefs_i(F, n, d), res;
    1286         175 :   long i, m = itos(gm);
    1287         175 :   if (!m) return V;
    1288         175 :   res = cgetg(n+2, t_VEC); gel(res,1) = gen_0;
    1289         175 :   if (m < 0)
    1290          49 :   { for (i=1; i <= n; i++) gel(res, i+1) = gdiv(gel(V, i+1), powuu(i,-m)); }
    1291             :   else
    1292        2457 :   { for (i=1; i <= n; i++) gel(res, i+1) = gmul(gel(V,i+1), powuu(i,m)); }
    1293         175 :   return gc_upto(av, res);
    1294             : }
    1295             : 
    1296             : static GEN
    1297          14 : c_derivE2(long n, long d, GEN F, GEN gm)
    1298             : {
    1299          14 :   pari_sp av = avma;
    1300             :   GEN VF, VE, res, tmp, gk;
    1301          14 :   long i, m = itos(gm), nd;
    1302          14 :   if (m == 0) return mfcoefs_i(F, n, d);
    1303          14 :   nd = n*d;
    1304          14 :   VF = mfcoefs_i(F, nd, 1); VE = mfcoefs_i(mfEk(2), nd, 1);
    1305          14 :   gk = mf_get_gk(F);
    1306          14 :   if (m == 1)
    1307             :   {
    1308           7 :     res = cgetg(n+2, t_VEC);
    1309          56 :     for (i = 0; i <= n; i++) gel(res, i+1) = gmulsg(i, gel(VF, i*d+1));
    1310           7 :     tmp = c_deflate(n, d, RgV_mul_RgXn(VF, VE));
    1311           7 :     return gc_upto(av, gsub(res, gmul(gdivgu(gk, 12), tmp)));
    1312             :   }
    1313             :   else
    1314             :   {
    1315             :     long j;
    1316          35 :     for (j = 1; j <= m; j++)
    1317             :     {
    1318          28 :       tmp = RgV_mul_RgXn(VF, VE);
    1319         140 :       for (i = 0; i <= nd; i++) gel(VF, i+1) = gmulsg(i, gel(VF, i+1));
    1320          28 :       VF = gsub(VF, gmul(gdivgu(gaddgs(gk, 2*(j-1)), 12), tmp));
    1321             :     }
    1322           7 :     return gc_GEN(av, c_deflate(n, d, VF));
    1323             :   }
    1324             : }
    1325             : 
    1326             : /* Twist by the character (D/.) */
    1327             : static GEN
    1328         168 : c_twist(long n, long d, GEN F, GEN D)
    1329             : {
    1330         168 :   pari_sp av = avma;
    1331         168 :   GEN v = mfcoefs_i(F, n, d), z = cgetg(n+2, t_VEC);
    1332             :   long i;
    1333         994 :   for (i = 0; i <= n; i++)
    1334             :   {
    1335             :     long s;
    1336         826 :     GEN a = gel(v, i+1);
    1337         826 :     if (d == 1) s = krois(D, i);
    1338             :     else
    1339             :     {
    1340         266 :       pari_sp av2 = avma;
    1341         266 :       s = kronecker(D, muluu(i, d)); set_avma(av2);
    1342             :     }
    1343         826 :     switch(s)
    1344             :     {
    1345         259 :       case 1: a = gcopy(a); break;
    1346         252 :       case -1: a = gneg(a); break;
    1347         315 :       default: a = gen_0; break;
    1348             :     }
    1349         826 :     gel(z, i+1) = a;
    1350             :   }
    1351         168 :   return gc_upto(av, z);
    1352             : }
    1353             : 
    1354             : /* form F given by closure, compute T(n)(F) as closure */
    1355             : static GEN
    1356        1246 : c_hecke(long m, long l, GEN DATA, GEN F)
    1357             : {
    1358        1246 :   pari_sp av = avma;
    1359        1246 :   return gc_GEN(av, hecke_i(m, l, NULL, F, DATA));
    1360             : }
    1361             : static GEN
    1362         147 : c_const(long n, long d, GEN C)
    1363             : {
    1364         147 :   GEN V = zerovec(n+1);
    1365         147 :   long i, j, l = lg(C);
    1366         147 :   if (l > d*n+2) l = d*n+2;
    1367         196 :   for (i = j = 1; i < l; i+=d, j++) gel(V, j) = gcopy(gel(C,i));
    1368         147 :   return V;
    1369             : }
    1370             : 
    1371             : /* m > 0 */
    1372             : static GEN
    1373         525 : eta3_ZXn(long m)
    1374             : {
    1375         525 :   long l = m+2, n, k;
    1376         525 :   GEN P = cgetg(l,t_POL);
    1377         525 :   P[1] = evalsigne(1)|evalvarn(0);
    1378        7245 :   for (n = 2; n < l; n++) gel(P,n) = gen_0;
    1379         525 :   for (n = k = 0;; n++)
    1380             :   {
    1381        2891 :     if (k + n >= m) { setlg(P, k+3); return P; }
    1382        2366 :     k += n;
    1383             :     /* now k = n(n+1) / 2 */
    1384        2366 :     gel(P, k+2) = odd(n)? utoineg(2*n+1): utoipos(2*n+1);
    1385             :   }
    1386             : }
    1387             : 
    1388             : static GEN
    1389         539 : c_delta(long n, long d)
    1390             : {
    1391         539 :   pari_sp ltop = avma;
    1392         539 :   long N = n*d;
    1393             :   GEN e;
    1394         539 :   if (!N) return mkvec(gen_0);
    1395         525 :   e = eta3_ZXn(N);
    1396         525 :   e = ZXn_sqr(e,N);
    1397         525 :   e = ZXn_sqr(e,N);
    1398         525 :   e = ZXn_sqr(e,N); /* eta(x)^24 */
    1399         525 :   settyp(e, t_VEC);
    1400         525 :   gel(e,1) = gen_0; /* Delta(x) = x*eta(x)^24 as a t_VEC */
    1401         525 :   return gc_GEN(ltop, c_deflate(n, d, e));
    1402             : }
    1403             : 
    1404             : /* return s(d) such that s|f <=> d | f^2 */
    1405             : static long
    1406          56 : mysqrtu(ulong d)
    1407             : {
    1408          56 :   GEN fa = myfactoru(d), P = gel(fa,1), E = gel(fa,2);
    1409          56 :   long l = lg(P), i, s = 1;
    1410         140 :   for (i = 1; i < l; i++) s *= upowuu(P[i], (E[i]+1)>>1);
    1411          56 :   return s;
    1412             : }
    1413             : static GEN
    1414        1946 : c_theta(long n, long d, GEN psi)
    1415             : {
    1416        1946 :   long lim = usqrt(n*d), F = mfcharmodulus(psi), par = mfcharparity(psi);
    1417        1946 :   long f, d2 = d == 1? 1: mysqrtu(d);
    1418        1946 :   GEN V = zerovec(n + 1);
    1419        8722 :   for (f = d2; f <= lim; f += d2)
    1420        6776 :     if (ugcd(F, f) == 1)
    1421             :     {
    1422        6769 :       pari_sp av = avma;
    1423        6769 :       GEN c = mfchareval(psi, f);
    1424        6769 :       gel(V, f*f/d + 1) = gc_upto(av, par < 0? gmulgu(c,2*f): gmul2n(c,1));
    1425             :     }
    1426        1946 :   if (F == 1) gel(V, 1) = gen_1;
    1427        1946 :   return V; /* no GC needed */
    1428             : }
    1429             : 
    1430             : static GEN
    1431         203 : c_etaquo(long n, long d, GEN eta, GEN gs)
    1432             : {
    1433         203 :   pari_sp av = avma;
    1434         203 :   long s = itos(gs), nd = n*d, nds = nd - s + 1;
    1435             :   GEN c;
    1436         203 :   if (nds <= 0) return zerovec(n+1);
    1437         182 :   c = RgX_to_RgC(eta_product_ZXn(eta, nds), nds); settyp(c, t_VEC);
    1438         182 :   if (s > 0) c = shallowconcat(zerovec(s), c);
    1439         182 :   return gc_GEN(av, c_deflate(n, d, c));
    1440             : }
    1441             : 
    1442             : static GEN
    1443          77 : c_ell(long n, long d, GEN E)
    1444             : {
    1445          77 :   pari_sp av = avma;
    1446             :   GEN v;
    1447          77 :   if (d == 1) return gconcat(gen_0, ellan(E, n));
    1448           7 :   v = vec_prepend(ellan(E, n*d), gen_0);
    1449           7 :   return gc_GEN(av, c_deflate(n, d, v));
    1450             : }
    1451             : 
    1452             : static GEN
    1453          21 : c_cusptrace(long n, long d, GEN F)
    1454             : {
    1455          21 :   pari_sp av = avma;
    1456          21 :   GEN D = gel(F,2), res = cgetg(n+2, t_VEC);
    1457          21 :   long i, N = mf_get_N(F), k = mf_get_k(F);
    1458          21 :   gel(res, 1) = gen_0;
    1459         140 :   for (i = 1; i <= n; i++)
    1460         119 :     gel(res, i+1) = mfcusptrace_i(N, k, i*d, mydivisorsu(i*d), D);
    1461          21 :   return gc_GEN(av, res);
    1462             : }
    1463             : 
    1464             : static GEN
    1465        1918 : c_newtrace(long n, long d, GEN F)
    1466             : {
    1467        1918 :   pari_sp av = avma;
    1468             :   cachenew_t cache;
    1469        1918 :   long N = mf_get_N(F);
    1470             :   GEN v;
    1471        1918 :   init_cachenew(&cache, n == 1? 1: n*d, N, F);
    1472        1918 :   v = colnewtrace(0, n, d, N, mf_get_k(F), &cache);
    1473        1918 :   settyp(v, t_VEC); return gc_GEN(av, v);
    1474             : }
    1475             : 
    1476             : static GEN
    1477        7525 : c_Bd(long n, long d, GEN F, GEN A)
    1478             : {
    1479        7525 :   pari_sp av = avma;
    1480        7525 :   long a = itou(A), ad = ugcd(a,d), aad = a/ad, i, j;
    1481        7525 :   GEN w, v = mfcoefs_i(F, n/aad, d/ad);
    1482        7525 :   if (a == 1) return v;
    1483        7525 :   n++; w = zerovec(n);
    1484      213416 :   for (i = j = 1; j <= n; i++, j += aad) gel(w,j) = gcopy(gel(v,i));
    1485        7525 :   return gc_upto(av, w);
    1486             : }
    1487             : 
    1488             : static GEN
    1489        5579 : c_dihedral(long n, long d, GEN F)
    1490             : {
    1491        5579 :   pari_sp av = avma;
    1492        5579 :   GEN CHI = mf_get_CHI(F);
    1493        5579 :   GEN w = gel(F,3), V = dihan(gel(F,2), w, gel(F,4), mfcharorder(CHI), n*d);
    1494        5579 :   GEN Tinit = gel(w,3), Pm = gel(Tinit,1);
    1495        5579 :   GEN A = c_deflate(n, d, V);
    1496        5579 :   if (degpol(Pm) == 1 || RgV_is_ZV(A)) return gc_GEN(av, A);
    1497        1043 :   return gc_upto(av, gmodulo(A, Pm));
    1498             : }
    1499             : 
    1500             : static GEN
    1501         343 : c_mfEH(long n, long d, GEN F)
    1502             : {
    1503         343 :   pari_sp av = avma;
    1504             :   GEN v, M, A;
    1505         343 :   long i, r = mf_get_r(F);
    1506         343 :   if (n == 1)
    1507          14 :     return gc_GEN(av, mkvec2(mfEHcoef(r,0),mfEHcoef(r,d)));
    1508             :   /* speedup mfcoef */
    1509         329 :   if (r == 1)
    1510             :   {
    1511          70 :     v = cgetg(n+2, t_VEC);
    1512          70 :     gel(v,1) = sstoQ(-1,12);
    1513       83258 :     for (i = 1; i <= n; i++)
    1514             :     {
    1515       83188 :       long id = i*d, a = id & 3;
    1516       83188 :       gel(v,i+1) = (a==1 || a==2)? gen_0: uutoQ(hclassno6u(id), 6);
    1517             :     }
    1518          70 :     return v; /* no GC needed */
    1519             :   }
    1520         259 :   M = mfEHmat(n*d+1,r);
    1521         259 :   if (d > 1)
    1522             :   {
    1523          35 :     long l = lg(M);
    1524         119 :     for (i = 1; i < l; i++) gel(M,i) = c_deflate(n, d, gel(M,i));
    1525             :   }
    1526         259 :   A = gel(F,2); /* [num(B), den(B)] */
    1527         259 :   v = RgC_Rg_div(RgM_RgC_mul(M, gel(A,1)), gel(A,2));
    1528         259 :   settyp(v,t_VEC); return gc_upto(av, v);
    1529             : }
    1530             : 
    1531             : static GEN
    1532       11361 : c_mfeisen(long n, long d, GEN F)
    1533             : {
    1534       11361 :   pari_sp av = avma;
    1535       11361 :   GEN v, vchi, E0, P, T, CHI, gk = mf_get_gk(F);
    1536             :   long i, k;
    1537       11361 :   if (typ(gk) != t_INT) return c_mfEH(n, d, F);
    1538       11018 :   k = itou(gk);
    1539       11018 :   vchi = gel(F,2);
    1540       11018 :   E0 = gel(vchi,1);
    1541       11018 :   T = gel(vchi,2);
    1542       11018 :   P = gel(T,1);
    1543       11018 :   CHI = gel(vchi,3);
    1544       11018 :   v = cgetg(n+2, t_VEC);
    1545       11018 :   gel(v, 1) = gcopy(E0); /* E(0) */
    1546       11018 :   if (lg(vchi) == 5)
    1547             :   { /* E_k(chi1,chi2) */
    1548        8925 :     GEN CHI2 = gel(vchi,4), F3 = gel(F,3);
    1549        8925 :     long ord = F3[1], j = F3[2];
    1550      509670 :     for (i = 1; i <= n; i++) gel(v, i+1) = sigchi2(k, CHI, CHI2, i*d, ord);
    1551        8925 :     v = QabV_tracerel(T, j, v);
    1552             :   }
    1553             :   else
    1554             :   { /* E_k(chi) */
    1555       26285 :     for (i = 1; i <= n; i++) gel(v, i+1) = sigchi(k, CHI, i*d);
    1556             :   }
    1557       11018 :   if (degpol(P) != 1 && !RgV_is_QV(v)) return gc_upto(av, gmodulo(v, P));
    1558        8085 :   return gc_GEN(av, v);
    1559             : }
    1560             : 
    1561             : /* N^k * (D * B_k)(x/N), set D = denom(B_k) */
    1562             : static GEN
    1563        2023 : bern_init(long N, long k, GEN *pD)
    1564        2023 : { return ZX_rescale(Q_remove_denom(bernpol(k, 0), pD), utoi(N)); }
    1565             : 
    1566             : /* L(chi_D, 1-k) */
    1567             : static GEN
    1568          28 : lfunquadneg_naive(long D, long k)
    1569             : {
    1570             :   GEN B, dS, S;
    1571          28 :   long r, N = labs(D);
    1572             :   pari_sp av;
    1573          28 :   if (k == 1 && N == 1) return gneg(ghalf);
    1574          28 :   B = bern_init(N, k, &dS);
    1575          28 :   dS = mul_denom(dS, stoi(-N*k));
    1576          28 :   av = avma;
    1577        7175 :   for (r = 0, S = gen_0; r < N; r++)
    1578             :   {
    1579        7147 :     long c = kross(D, r);
    1580        7147 :     if (c)
    1581             :     {
    1582        5152 :       GEN t = ZX_Z_eval(B, utoi(r));
    1583        5152 :       S = c > 0 ? addii(S, t) : subii(S, t);
    1584        5152 :       S = gc_INT(av, S);
    1585             :     }
    1586             :   }
    1587          28 :   return gdiv(S, dS);
    1588             : }
    1589             : 
    1590             : /* Returns vector of coeffs from F[0], F[d], ..., F[d*n] */
    1591             : static GEN
    1592       38563 : mfcoefs_i(GEN F, long n, long d)
    1593             : {
    1594       38563 :   if (n < 0) return gen_0;
    1595       38563 :   switch(mf_get_type(F))
    1596             :   {
    1597         147 :     case t_MF_CONST: return c_const(n, d, gel(F,2));
    1598       11361 :     case t_MF_EISEN: return c_mfeisen(n, d, F);
    1599         882 :     case t_MF_Ek: return c_Ek(n, d, F);
    1600         539 :     case t_MF_DELTA: return c_delta(n, d);
    1601        1680 :     case t_MF_THETA: return c_theta(n, d, gel(F,2));
    1602         203 :     case t_MF_ETAQUO: return c_etaquo(n, d, gel(F,2), gel(F,3));
    1603          77 :     case t_MF_ELL: return c_ell(n, d, gel(F,2));
    1604         637 :     case t_MF_MUL: return c_mul(n, d, F);
    1605         112 :     case t_MF_POW: return c_pow(n, d, F);
    1606          42 :     case t_MF_BRACKET: return c_bracket(n, d, F);
    1607        3024 :     case t_MF_LINEAR: return c_linear(n, d, gel(F,2), gel(F,3), gel(F,4));
    1608        2331 :     case t_MF_LINEAR_BHN: return c_linear_bhn(n, d, F);
    1609         847 :     case t_MF_DIV: return c_div(n, d, F);
    1610          35 :     case t_MF_SHIFT: return c_shift(n, d, gel(F,2), gel(F,3));
    1611         175 :     case t_MF_DERIV: return c_deriv(n, d, gel(F,2), gel(F,3));
    1612          14 :     case t_MF_DERIVE2: return c_derivE2(n, d, gel(F,2), gel(F,3));
    1613         168 :     case t_MF_TWIST: return c_twist(n, d, gel(F,2), gel(F,3));
    1614        1246 :     case t_MF_HECKE: return c_hecke(n, d, gel(F,2), gel(F,3));
    1615        7525 :     case t_MF_BD: return c_Bd(n, d, gel(F,2), gel(F,3));
    1616          21 :     case t_MF_TRACE: return c_cusptrace(n, d, F);
    1617        1918 :     case t_MF_NEWTRACE: return c_newtrace(n, d, F);
    1618        5579 :     case t_MF_DIHEDRAL: return c_dihedral(n, d, F);
    1619             :     default: pari_err_TYPE("mfcoefs",F); return NULL;/*LCOV_EXCL_LINE*/
    1620             :   }
    1621             : }
    1622             : 
    1623             : static GEN
    1624         392 : matdeflate(long n, long d, GEN x)
    1625        1680 : { pari_APPLY_same(c_deflate(n,d,gel(x,i))); }
    1626             : static int
    1627        6104 : space_is_cusp(long space) { return space != mf_FULL && space != mf_EISEN; }
    1628             : /* safe with flraw mf */
    1629             : static GEN
    1630        2632 : mfcoefs_mf(GEN mf, long n, long d)
    1631             : {
    1632        2632 :   GEN MS, ME, E = MF_get_E(mf), S = MF_get_S(mf), M = MF_get_M(mf);
    1633        2632 :   long lE = lg(E), lS = lg(S), l = lE+lS-1;
    1634             : 
    1635        2632 :   if (l == 1) return cgetg(1, t_MAT);
    1636        2520 :   if (typ(M) == t_MAT && lg(M) != 1 && (n+1)*d < nbrows(M))
    1637          21 :     return matdeflate(n, d, M); /*cached; lg = 1 is possible from mfinit */
    1638        2499 :   ME = (lE == 1)? cgetg(1, t_MAT): mfvectomat(E, n, d);
    1639        2499 :   if (lS == 1)
    1640         455 :     MS = cgetg(1, t_MAT);
    1641        2044 :   else if (mf_get_type(gel(S,1)) == t_MF_DIV) /*k 1/2-integer or k=1 (exotic)*/
    1642         371 :     MS = matdeflate(n,d, mflineardivtomat(MF_get_N(mf), S, n*d));
    1643        1673 :   else if (MF_get_k(mf) == 1) /* k = 1 (dihedral) */
    1644             :   {
    1645         308 :     GEN M = mfvectomat(gmael(S,1,2), n, d);
    1646             :     long i;
    1647         308 :     MS = cgetg(lS, t_MAT);
    1648        1589 :     for (i = 1; i < lS; i++)
    1649             :     {
    1650        1281 :       GEN f = gel(S,i), dc = gel(f,4), c = RgM_RgC_mul(M, gel(f,3));
    1651        1281 :       if (!equali1(dc)) c = RgC_Rg_div(c,dc);
    1652        1281 :       gel(MS,i) = c;
    1653             :     }
    1654             :   }
    1655             :   else /* k >= 2 integer */
    1656        1365 :     MS = bhnmat_extend_nocache(NULL, MF_get_N(mf), n, d, S);
    1657        2499 :   return shallowconcat(ME,MS);
    1658             : }
    1659             : GEN
    1660        4144 : mfcoefs(GEN F, long n, long d)
    1661             : {
    1662        4144 :   if (!checkmf_i(F))
    1663             :   {
    1664          42 :     pari_sp av = avma;
    1665          42 :     GEN mf = checkMF_i(F); if (!mf) pari_err_TYPE("mfcoefs", F);
    1666          42 :     return gc_GEN(av, mfcoefs_mf(mf,n,d));
    1667             :   }
    1668        4102 :   if (d <= 0) pari_err_DOMAIN("mfcoefs", "d", "<=", gen_0, stoi(d));
    1669        4102 :   if (n < 0) return cgetg(1, t_VEC);
    1670        4102 :   return mfcoefs_i(F, n, d);
    1671             : }
    1672             : 
    1673             : /* assume k >= 0 */
    1674             : static GEN
    1675         455 : mfak_i(GEN F, long k)
    1676             : {
    1677         455 :   if (!k) return gel(mfcoefs_i(F,0,1), 1);
    1678         294 :   return gel(mfcoefs_i(F,1,k), 2);
    1679             : }
    1680             : GEN
    1681         301 : mfcoef(GEN F, long n)
    1682             : {
    1683         301 :   pari_sp av = avma;
    1684         301 :   if (!checkmf_i(F)) pari_err_TYPE("mfcoef",F);
    1685         301 :   return n < 0? gen_0: gc_GEN(av, mfak_i(F, n));
    1686             : }
    1687             : 
    1688             : static GEN
    1689         133 : paramconst() { return tagparams(t_MF_CONST, mkNK(1,0,mfchartrivial())); }
    1690             : static GEN
    1691          91 : mftrivial(void) { retmkvec2(paramconst(), cgetg(1,t_VEC)); }
    1692             : static GEN
    1693          42 : mf1(void) { retmkvec2(paramconst(), mkvec(gen_1)); }
    1694             : 
    1695             : /* induce mfchar CHI to G */
    1696             : static GEN
    1697      312179 : induce(GEN G, GEN CHI)
    1698             : {
    1699             :   GEN o, chi;
    1700      312179 :   if (typ(CHI) == t_INT) /* Kronecker */
    1701             :   {
    1702      300888 :     chi = znchar_quad(G, CHI);
    1703      300888 :     o = ZV_equal0(chi)? gen_1: gen_2;
    1704      300888 :     CHI = mkvec4(G,chi,o,cgetg(1,t_VEC));
    1705             :   }
    1706             :   else
    1707             :   {
    1708       11291 :     if (mfcharmodulus(CHI) == itos(znstar_get_N(G))) return CHI;
    1709       10640 :     CHI = leafcopy(CHI);
    1710       10640 :     chi = zncharinduce(gel(CHI,1), gel(CHI,2), G);
    1711       10640 :     gel(CHI,1) = G;
    1712       10640 :     gel(CHI,2) = chi;
    1713             :   }
    1714      311528 :   return CHI;
    1715             : }
    1716             : /* induce mfchar CHI to znstar(N) */
    1717             : static GEN
    1718       42476 : induceN(long N, GEN CHI)
    1719             : {
    1720       42476 :   if (mfcharmodulus(CHI) != N) CHI = induce(znstar0(utoipos(N),1), CHI);
    1721       42476 :   return CHI;
    1722             : }
    1723             : /* *pCHI1 and *pCHI2 are mfchar, induce to common modulus */
    1724             : static void
    1725       11305 : char2(GEN *pCHI1, GEN *pCHI2)
    1726             : {
    1727       11305 :   GEN CHI1 = *pCHI1, G1 = gel(CHI1,1), N1 = znstar_get_N(G1);
    1728       11305 :   GEN CHI2 = *pCHI2, G2 = gel(CHI2,1), N2 = znstar_get_N(G2);
    1729       11305 :   if (!equalii(N1,N2))
    1730             :   {
    1731        8855 :     GEN G, d = gcdii(N1,N2);
    1732        8855 :     if      (equalii(N2,d)) *pCHI2 = induce(G1, CHI2);
    1733        1589 :     else if (equalii(N1,d)) *pCHI1 = induce(G2, CHI1);
    1734             :     else
    1735             :     {
    1736         154 :       if (!equali1(d)) N2 = diviiexact(N2,d);
    1737         154 :       G = znstar0(mulii(N1,N2), 1);
    1738         154 :       *pCHI1 = induce(G, CHI1);
    1739         154 :       *pCHI2 = induce(G, CHI2);
    1740             :     }
    1741             :   }
    1742       11305 : }
    1743             : /* mfchar or charinit wrt same modulus; outputs a mfchar */
    1744             : static GEN
    1745      301994 : mfcharmul_i(GEN CHI1, GEN CHI2)
    1746             : {
    1747      301994 :   GEN G = gel(CHI1,1), chi3 = zncharmul(G, gel(CHI1,2), gel(CHI2,2));
    1748      301994 :   return mfcharGL(G, chi3);
    1749             : }
    1750             : /* mfchar or charinit; outputs a mfchar */
    1751             : static GEN
    1752        1127 : mfcharmul(GEN CHI1, GEN CHI2)
    1753             : {
    1754        1127 :   char2(&CHI1, &CHI2); return mfcharmul_i(CHI1,CHI2);
    1755             : }
    1756             : /* mfchar or charinit; outputs a mfchar */
    1757             : static GEN
    1758         161 : mfcharpow(GEN CHI, GEN n)
    1759             : {
    1760             :   GEN G, chi;
    1761         161 :   G = gel(CHI,1); chi = zncharpow(G, gel(CHI,2), n);
    1762         161 :   return mfchartoprimitive(mfcharGL(G, chi), NULL);
    1763             : }
    1764             : /* mfchar or charinit wrt same modulus; outputs a mfchar */
    1765             : static GEN
    1766       10178 : mfchardiv_i(GEN CHI1, GEN CHI2)
    1767             : {
    1768       10178 :   GEN G = gel(CHI1,1), chi3 = znchardiv(G, gel(CHI1,2), gel(CHI2,2));
    1769       10178 :   return mfcharGL(G, chi3);
    1770             : }
    1771             : /* mfchar or charinit; outputs a mfchar */
    1772             : static GEN
    1773       10178 : mfchardiv(GEN CHI1, GEN CHI2)
    1774             : {
    1775       10178 :   char2(&CHI1, &CHI2); return mfchardiv_i(CHI1,CHI2);
    1776             : }
    1777             : static GEN
    1778          56 : mfcharconj(GEN CHI)
    1779             : {
    1780          56 :   CHI = leafcopy(CHI);
    1781          56 :   gel(CHI,2) = zncharconj(gel(CHI,1), gel(CHI,2));
    1782          56 :   return CHI;
    1783             : }
    1784             : 
    1785             : /* CHI mfchar, assume 4 | N. Multiply CHI by \chi_{-4} */
    1786             : static GEN
    1787        1092 : mfchilift(GEN CHI, long N)
    1788             : {
    1789        1092 :   CHI = induceN(N, CHI);
    1790        1092 :   return mfcharmul_i(CHI, induce(gel(CHI,1), stoi(-4)));
    1791             : }
    1792             : /* CHI defined mod N, N4 = N/4;
    1793             :  * if CHI is defined mod N4 return CHI;
    1794             :  * else if CHI' = CHI*(-4,.) is defined mod N4, return CHI' (primitive)
    1795             :  * else error */
    1796             : static GEN
    1797          42 : mfcharchiliftprim(GEN CHI, long N4)
    1798             : {
    1799          42 :   long FC = mfcharconductor(CHI);
    1800             :   GEN CHIP;
    1801          42 :   if (N4 % FC == 0) return CHI;
    1802          14 :   CHIP = mfchartoprimitive(mfchilift(CHI, N4 << 2), &FC);
    1803          14 :   if (N4 % FC) pari_err_TYPE("mfkohnenbasis [incorrect CHI]", CHI);
    1804          14 :   return CHIP;
    1805             : }
    1806             : /* ensure CHI(-1) = (-1)^k [k integer] or 1 [half-integer], by multiplying
    1807             :  * by (-4/.) if needed */
    1808             : static GEN
    1809        2933 : mfchiadjust(GEN CHI, GEN gk, long N)
    1810             : {
    1811        2933 :   long par = mfcharparity(CHI);
    1812        2933 :   if (typ(gk) == t_INT &&  mpodd(gk)) par = -par;
    1813        2933 :   return par == 1 ? CHI : mfchilift(CHI, N);
    1814             : }
    1815             : 
    1816             : static GEN
    1817        4270 : mfsamefield(GEN T, GEN P, GEN Q)
    1818             : {
    1819        4270 :   if (degpol(P) == 1) return Q;
    1820         721 :   if (degpol(Q) == 1) return P;
    1821         630 :   if (!gequal(P,Q)) pari_err_TYPE("mfsamefield [different fields]",mkvec2(P,Q));
    1822         623 :   if (T) err_cyclo();
    1823         623 :   return P;
    1824             : }
    1825             : 
    1826             : GEN
    1827         455 : mfmul(GEN f, GEN g)
    1828             : {
    1829         455 :   pari_sp av = avma;
    1830             :   GEN T, N, K, NK, CHI, CHIf, CHIg;
    1831         455 :   if (!checkmf_i(f)) pari_err_TYPE("mfmul",f);
    1832         455 :   if (!checkmf_i(g)) pari_err_TYPE("mfmul",g);
    1833         455 :   N = lcmii(mf_get_gN(f), mf_get_gN(g));
    1834         455 :   K = gadd(mf_get_gk(f), mf_get_gk(g));
    1835         455 :   CHIf = mf_get_CHI(f);
    1836         455 :   CHIg = mf_get_CHI(g);
    1837         455 :   CHI = mfchiadjust(mfcharmul(CHIf,CHIg), K, itos(N));
    1838         455 :   T = chicompat(CHI, CHIf, CHIg);
    1839         455 :   NK = mkgNK(N, K, CHI, mfsamefield(T, mf_get_field(f), mf_get_field(g)));
    1840         448 :   return gc_GEN(av, T? tag3(t_MF_MUL,NK,f,g,T): tag2(t_MF_MUL,NK,f,g));
    1841             : }
    1842             : GEN
    1843          77 : mfpow(GEN f, long n)
    1844             : {
    1845          77 :   pari_sp av = avma;
    1846             :   GEN T, KK, NK, gn, CHI, CHIf;
    1847          77 :   if (!checkmf_i(f)) pari_err_TYPE("mfpow",f);
    1848          77 :   if (!n) return mf1();
    1849          77 :   if (n == 1) return gcopy(f);
    1850          77 :   KK = gmulsg(n,mf_get_gk(f));
    1851          77 :   gn = stoi(n);
    1852          77 :   CHIf = mf_get_CHI(f);
    1853          77 :   CHI = mfchiadjust(mfcharpow(CHIf,gn), KK, mf_get_N(f));
    1854          77 :   T = chicompat(CHI, CHIf, CHIf);
    1855          70 :   NK = mkgNK(mf_get_gN(f), KK, CHI, mf_get_field(f));
    1856          70 :   return gc_GEN(av, T? tag3(t_MF_POW,NK,f,gn,T): tag2(t_MF_POW,NK,f,gn));
    1857             : }
    1858             : GEN
    1859          28 : mfbracket(GEN f, GEN g, long m)
    1860             : {
    1861          28 :   pari_sp av = avma;
    1862             :   GEN T, N, K, NK, CHI, CHIf, CHIg;
    1863          28 :   if (!checkmf_i(f)) pari_err_TYPE("mfbracket",f);
    1864          28 :   if (!checkmf_i(g)) pari_err_TYPE("mfbracket",g);
    1865          28 :   if (m < 0) pari_err_TYPE("mfbracket [m<0]",stoi(m));
    1866          28 :   K = gaddgs(gadd(mf_get_gk(f), mf_get_gk(g)), 2*m);
    1867          28 :   if (gsigne(K) < 0) pari_err_IMPL("mfbracket for this form");
    1868          28 :   N = lcmii(mf_get_gN(f), mf_get_gN(g));
    1869          28 :   CHIf = mf_get_CHI(f);
    1870          28 :   CHIg = mf_get_CHI(g);
    1871          28 :   CHI = mfcharmul(CHIf, CHIg);
    1872          28 :   CHI = mfchiadjust(CHI, K, itou(N));
    1873          28 :   T = chicompat(CHI, CHIf, CHIg);
    1874          28 :   NK = mkgNK(N, K, CHI, mfsamefield(T, mf_get_field(f), mf_get_field(g)));
    1875          56 :   return gc_GEN(av, T? tag4(t_MF_BRACKET, NK, f, g, utoi(m), T)
    1876          28 :                            : tag3(t_MF_BRACKET, NK, f, g, utoi(m)));
    1877             : }
    1878             : 
    1879             : /* remove 0 entries in L */
    1880             : static int
    1881        1960 : mflinear_strip(GEN *pF, GEN *pL)
    1882             : {
    1883        1960 :   pari_sp av = avma;
    1884        1960 :   GEN F = *pF, L = *pL;
    1885        1960 :   long i, j, l = lg(L);
    1886        1960 :   GEN F2 = cgetg(l, t_VEC), L2 = cgetg(l, t_VEC);
    1887       11753 :   for (i = j = 1; i < l; i++)
    1888             :   {
    1889        9793 :     if (gequal0(gel(L,i))) continue;
    1890        4823 :     gel(F2,j) = gel(F,i);
    1891        4823 :     gel(L2,j) = gel(L,i); j++;
    1892             :   }
    1893        1960 :   if (j == l) set_avma(av);
    1894             :   else
    1895             :   {
    1896         602 :     setlg(F2,j); *pF = F2;
    1897         602 :     setlg(L2,j); *pL = L2;
    1898             :   }
    1899        1960 :   return (j > 1);
    1900             : }
    1901             : static GEN
    1902        7070 : taglinear_i(long t, GEN NK, GEN F, GEN L)
    1903             : {
    1904             :   GEN dL;
    1905        7070 :   L = Q_remove_denom(L, &dL); if (!dL) dL = gen_1;
    1906        7070 :   return tag3(t, NK, F, L, dL);
    1907             : }
    1908             : static GEN
    1909        2926 : taglinear(GEN NK, GEN F, GEN L)
    1910             : {
    1911        2926 :   long t = ok_bhn_linear(F)? t_MF_LINEAR_BHN: t_MF_LINEAR;
    1912        2926 :    return taglinear_i(t, NK, F, L);
    1913             : }
    1914             : /* assume F has parameters NK = [N,K,CHI] */
    1915             : static GEN
    1916         490 : mflinear_i(GEN NK, GEN F, GEN L)
    1917             : {
    1918         490 :   if (!mflinear_strip(&F,&L)) return mftrivial();
    1919         490 :   return taglinear(NK, F,L);
    1920             : }
    1921             : static GEN
    1922         770 : mflinear_bhn(GEN mf, GEN L)
    1923             : {
    1924             :   long i, l;
    1925         770 :   GEN P, NK, F = MF_get_S(mf);
    1926         770 :   if (!mflinear_strip(&F,&L)) return mftrivial();
    1927         763 :   l = lg(L); P = pol_x(1);
    1928        3465 :   for (i = 1; i < l; i++)
    1929             :   {
    1930        2702 :     GEN c = gel(L,i);
    1931        2702 :     if (typ(c) == t_POLMOD && varn(gel(c,1)) == 1)
    1932         665 :       P = mfsamefield(NULL, P, gel(c,1));
    1933             :   }
    1934         763 :   NK = mkgNK(MF_get_gN(mf), MF_get_gk(mf), MF_get_CHI(mf), P);
    1935         763 :   return taglinear_i(t_MF_LINEAR_BHN,  NK, F,L);
    1936             : }
    1937             : 
    1938             : /* F vector of forms with same weight and character but varying level, return
    1939             :  * global [N,k,chi,P] */
    1940             : static GEN
    1941        3339 : vecmfNK(GEN F)
    1942             : {
    1943        3339 :   long i, l = lg(F);
    1944             :   GEN N, f;
    1945        3339 :   if (l == 1) return mkNK(1, 0, mfchartrivial());
    1946        3339 :   f = gel(F,1); N = mf_get_gN(f);
    1947       47474 :   for (i = 2; i < l; i++) N = lcmii(N, mf_get_gN(gel(F,i)));
    1948        3339 :   return mkgNK(N, mf_get_gk(f), mf_get_CHI(f), mf_get_field(f));
    1949             : }
    1950             : /* do not use mflinear: mflineardivtomat rely on F being constant across the
    1951             :  * basis where mflinear strips the ones matched by 0 coeffs. Assume k and CHI
    1952             :  * constant, N is allowed to vary. */
    1953             : static GEN
    1954        1218 : vecmflinear(GEN F, GEN C)
    1955             : {
    1956        1218 :   long i, t, l = lg(C);
    1957        1218 :   GEN NK, v = cgetg(l, t_VEC);
    1958        1218 :   if (l == 1) return v;
    1959        1218 :   t = ok_bhn_linear(F)? t_MF_LINEAR_BHN: t_MF_LINEAR;
    1960        1218 :   NK = vecmfNK(F);
    1961        4599 :   for (i = 1; i < l; i++) gel(v,i) = taglinear_i(t, NK, F, gel(C,i));
    1962        1218 :   return v;
    1963             : }
    1964             : /* vecmflinear(F,C), then divide everything by E, which has valuation 0 */
    1965             : static GEN
    1966         434 : vecmflineardiv0(GEN F, GEN C, GEN E)
    1967             : {
    1968         434 :   GEN v = vecmflinear(F, C);
    1969         434 :   long i, l = lg(v);
    1970         434 :   if (l == 1) return v;
    1971         434 :   gel(v,1) = mfdiv_val(gel(v,1), E, 0);
    1972        1729 :   for (i = 2; i < l; i++)
    1973             :   { /* v[i] /= E */
    1974        1295 :     GEN f = shallowcopy(gel(v,1));
    1975        1295 :     gel(f,2) = gel(v,i);
    1976        1295 :     gel(v,i) = f;
    1977             :   }
    1978         434 :   return v;
    1979             : }
    1980             : 
    1981             : /* Non empty linear combination of linear combinations of same
    1982             :  * F_j=\sum_i \mu_{i,j}G_i so R = \sum_i (\sum_j(\la_j\mu_{i,j})) G_i */
    1983             : static GEN
    1984        2121 : mflinear_linear(GEN F, GEN L, int strip)
    1985             : {
    1986        2121 :   long l = lg(F), j;
    1987        2121 :   GEN vF, M = cgetg(l, t_MAT);
    1988        2121 :   L = shallowcopy(L);
    1989       20097 :   for (j = 1; j < l; j++)
    1990             :   {
    1991       17976 :     GEN f = gel(F,j), c = gel(f,3), d = gel(f,4);
    1992       17976 :     if (typ(c) == t_VEC) c = shallowtrans(c);
    1993       17976 :     if (!isint1(d)) gel(L,j) = gdiv(gel(L,j),d);
    1994       17976 :     gel(M,j) = c;
    1995             :   }
    1996        2121 :   vF = gmael(F,1,2); L = RgM_RgC_mul(M,L);
    1997        2121 :   if (strip && !mflinear_strip(&vF,&L)) return mftrivial();
    1998        2121 :   return taglinear(vecmfNK(vF), vF, L);
    1999             : }
    2000             : /* F nonempty vector of forms of the form mfdiv(mflinear(B,v), E) where E
    2001             :  * does not vanish at oo, or mflinear(B,v). Apply mflinear(F, L) */
    2002             : static GEN
    2003        2121 : mflineardiv_linear(GEN F, GEN L, int strip)
    2004             : {
    2005        2121 :   long l = lg(F), j;
    2006             :   GEN v, E, f;
    2007        2121 :   if (lg(L) != l) pari_err_DIM("mflineardiv_linear");
    2008        2121 :   f = gel(F,1); /* l > 1 */
    2009        2121 :   if (mf_get_type(f) != t_MF_DIV) return mflinear_linear(F,L,strip);
    2010        1813 :   E = gel(f,3);
    2011        1813 :   v = cgetg(l, t_VEC);
    2012       18634 :   for (j = 1; j < l; j++) { GEN f = gel(F,j); gel(v,j) = gel(f,2); }
    2013        1813 :   return mfdiv_val(mflinear_linear(v,L,strip), E, 0);
    2014             : }
    2015             : static GEN
    2016         483 : vecmflineardiv_linear(GEN F, GEN M)
    2017             : {
    2018         483 :   long i, l = lg(M);
    2019         483 :   GEN v = cgetg(l, t_VEC);
    2020        2023 :   for (i = 1; i < l; i++) gel(v,i) = mflineardiv_linear(F, gel(M,i), 0);
    2021         483 :   return v;
    2022             : }
    2023             : 
    2024             : static GEN
    2025        1057 : tobasis(GEN mf, GEN F, GEN L)
    2026             : {
    2027        1057 :   if (checkmf_i(L) && mf) return mftobasis(mf, L, 0);
    2028        1050 :   if (typ(F) != t_VEC) pari_err_TYPE("mflinear",F);
    2029        1050 :   if (!is_vec_t(typ(L))) pari_err_TYPE("mflinear",L);
    2030        1050 :   if (lg(L) != lg(F)) pari_err_DIM("mflinear");
    2031        1050 :   return L;
    2032             : }
    2033             : GEN
    2034        1106 : mflinear(GEN F, GEN L)
    2035             : {
    2036        1106 :   pari_sp av = avma;
    2037        1106 :   GEN G, NK, P, mf = checkMF_i(F), N = NULL, K = NULL, CHI = NULL;
    2038             :   long i, l;
    2039        1106 :   if (mf)
    2040             :   {
    2041         728 :     GEN gk = MF_get_gk(mf);
    2042         728 :     F = MF_get_basis(F);
    2043         728 :     if (typ(gk) != t_INT)
    2044          49 :       return gc_GEN(av, mflineardiv_linear(F, L, 1));
    2045         679 :     if (itou(gk) > 1 && space_is_cusp(MF_get_space(mf)))
    2046             :     {
    2047         455 :       L = tobasis(mf, F, L);
    2048         455 :       return gc_GEN(av, mflinear_bhn(mf, L));
    2049             :     }
    2050             :   }
    2051         602 :   L = tobasis(mf, F, L);
    2052         602 :   if (!mflinear_strip(&F,&L)) return mftrivial();
    2053             : 
    2054         588 :   l = lg(F);
    2055         588 :   if (l == 2 && gequal1(gel(L,1))) return gc_GEN(av, gel(F,1));
    2056         329 :   P = pol_x(1);
    2057        1057 :   for (i = 1; i < l; i++)
    2058             :   {
    2059         735 :     GEN f = gel(F,i), c = gel(L,i), Ni, Ki;
    2060         735 :     if (!checkmf_i(f)) pari_err_TYPE("mflinear", f);
    2061         735 :     Ni = mf_get_gN(f); N = N? lcmii(N, Ni): Ni;
    2062         735 :     Ki = mf_get_gk(f);
    2063         735 :     if (!K) K = Ki;
    2064         406 :     else if (!gequal(K, Ki))
    2065           7 :       pari_err_TYPE("mflinear [different weights]", mkvec2(K,Ki));
    2066         728 :     P = mfsamefield(NULL, P, mf_get_field(f));
    2067         728 :     if (typ(c) == t_POLMOD && varn(gel(c,1)) == 1)
    2068         126 :       P = mfsamefield(NULL, P, gel(c,1));
    2069             :   }
    2070         322 :   G = znstar0(N,1);
    2071        1036 :   for (i = 1; i < l; i++)
    2072             :   {
    2073         721 :     GEN CHI2 = mf_get_CHI(gel(F,i));
    2074         721 :     CHI2 = induce(G, CHI2);
    2075         721 :     if (!CHI) CHI = CHI2;
    2076         399 :     else if (!gequal(CHI, CHI2))
    2077           7 :       pari_err_TYPE("mflinear [different characters]", mkvec2(CHI,CHI2));
    2078             :   }
    2079         315 :   NK = mkgNK(N, K, CHI, P);
    2080         315 :   return gc_GEN(av, taglinear(NK,F,L));
    2081             : }
    2082             : 
    2083             : GEN
    2084          42 : mfshift(GEN F, long sh)
    2085             : {
    2086          42 :   pari_sp av = avma;
    2087          42 :   if (!checkmf_i(F)) pari_err_TYPE("mfshift",F);
    2088          42 :   return gc_GEN(av, tag2(t_MF_SHIFT, mf_get_NK(F), F, stoi(sh)));
    2089             : }
    2090             : static long
    2091          49 : mfval(GEN F)
    2092             : {
    2093          49 :   pari_sp av = avma;
    2094          49 :   long i = 0, n, sb;
    2095             :   GEN gk, gN;
    2096          49 :   if (!checkmf_i(F)) pari_err_TYPE("mfval", F);
    2097          49 :   gN = mf_get_gN(F);
    2098          49 :   gk = mf_get_gk(F);
    2099          49 :   sb = mfsturmNgk(itou(gN), gk);
    2100          70 :   for (n = 1; n <= sb;)
    2101             :   {
    2102             :     GEN v;
    2103          63 :     if (n > 0.5*sb) n = sb+1;
    2104          63 :     v = mfcoefs_i(F, n, 1);
    2105         119 :     for (; i <= n; i++)
    2106          98 :       if (!gequal0(gel(v, i+1))) return gc_long(av,i);
    2107          21 :     n <<= 1;
    2108             :   }
    2109           7 :   return gc_long(av,-1);
    2110             : }
    2111             : 
    2112             : GEN
    2113        2275 : mfdiv_val(GEN f, GEN g, long vg)
    2114             : {
    2115             :   GEN T, N, K, NK, CHI, CHIf, CHIg;
    2116        2275 :   if (vg) { f = mfshift(f,vg); g = mfshift(g,vg); }
    2117        2275 :   N = lcmii(mf_get_gN(f), mf_get_gN(g));
    2118        2275 :   K = gsub(mf_get_gk(f), mf_get_gk(g));
    2119        2275 :   CHIf = mf_get_CHI(f);
    2120        2275 :   CHIg = mf_get_CHI(g);
    2121        2275 :   CHI = mfchiadjust(mfchardiv(CHIf, CHIg), K, itos(N));
    2122        2275 :   T = chicompat(CHI, CHIf, CHIg);
    2123        2268 :   NK = mkgNK(N, K, CHI, mfsamefield(T, mf_get_field(f), mf_get_field(g)));
    2124        2268 :   return T? tag3(t_MF_DIV, NK, f, g, T): tag2(t_MF_DIV, NK, f, g);
    2125             : }
    2126             : GEN
    2127          49 : mfdiv(GEN F, GEN G)
    2128             : {
    2129          49 :   pari_sp av = avma;
    2130          49 :   long v = mfval(G);
    2131          49 :   if (!checkmf_i(F)) pari_err_TYPE("mfdiv", F);
    2132          42 :   if (v < 0 || (v && !gequal0(mfcoefs(F, v-1, 1))))
    2133          14 :     pari_err_DOMAIN("mfdiv", "ord(G)", ">", strtoGENstr("ord(F)"),
    2134             :                     mkvec2(F, G));
    2135          28 :   return gc_GEN(av, mfdiv_val(F, G, v));
    2136             : }
    2137             : GEN
    2138         182 : mfderiv(GEN F, long m)
    2139             : {
    2140         182 :   pari_sp av = avma;
    2141             :   GEN NK, gk;
    2142         182 :   if (!checkmf_i(F)) pari_err_TYPE("mfderiv",F);
    2143         182 :   gk = gaddgs(mf_get_gk(F), 2*m);
    2144         182 :   NK = mkgNK(mf_get_gN(F), gk, mf_get_CHI(F), mf_get_field(F));
    2145         182 :   return gc_GEN(av, tag2(t_MF_DERIV, NK, F, stoi(m)));
    2146             : }
    2147             : GEN
    2148          21 : mfderivE2(GEN F, long m)
    2149             : {
    2150          21 :   pari_sp av = avma;
    2151             :   GEN NK, gk;
    2152          21 :   if (!checkmf_i(F)) pari_err_TYPE("mfderivE2",F);
    2153          21 :   if (m < 0) pari_err_DOMAIN("mfderivE2","m","<",gen_0,stoi(m));
    2154          21 :   gk = gaddgs(mf_get_gk(F), 2*m);
    2155          21 :   NK = mkgNK(mf_get_gN(F), gk, mf_get_CHI(F), mf_get_field(F));
    2156          21 :   return gc_GEN(av, tag2(t_MF_DERIVE2, NK, F, stoi(m)));
    2157             : }
    2158             : 
    2159             : GEN
    2160          28 : mftwist(GEN F, GEN D)
    2161             : {
    2162          28 :   pari_sp av = avma;
    2163             :   GEN NK, CHI, NT, Da;
    2164             :   long q;
    2165          28 :   if (!checkmf_i(F)) pari_err_TYPE("mftwist", F);
    2166          28 :   if (typ(D) != t_INT) pari_err_TYPE("mftwist", D);
    2167          28 :   Da = mpabs_shallow(D);
    2168          28 :   CHI = mf_get_CHI(F); q = mfcharconductor(CHI);
    2169          28 :   NT = glcm(glcm(mf_get_gN(F), mulsi(q, Da)), sqri(Da));
    2170          28 :   NK = mkgNK(NT, mf_get_gk(F), CHI, mf_get_field(F));
    2171          28 :   return gc_GEN(av, tag2(t_MF_TWIST, NK, F, D));
    2172             : }
    2173             : 
    2174             : /***************************************************************/
    2175             : /*                 Generic cache handling                      */
    2176             : /***************************************************************/
    2177             : enum { cache_FACT, cache_DIV, cache_H, cache_D, cache_DIH };
    2178             : typedef struct {
    2179             :   const char *name;
    2180             :   GEN cache;
    2181             :   ulong minself, maxself;
    2182             :   void (*init)(long);
    2183             :   ulong miss, maxmiss;
    2184             :   long compressed;
    2185             : } cache;
    2186             : 
    2187             : static void constfact(long lim);
    2188             : static void constdiv(long lim);
    2189             : static void consttabh(long lim);
    2190             : static void consttabdihedral(long lim);
    2191             : static void constcoredisc(long lim);
    2192             : static THREAD cache caches[] = {
    2193             : { "Factors",  NULL,  50000,    50000, &constfact, 0, 0, 0 },
    2194             : { "Divisors", NULL,  50000,    50000, &constdiv, 0, 0, 0 },
    2195             : { "H",        NULL, 100000, 10000000, &consttabh, 0, 0, 1 },
    2196             : { "CorediscF",NULL, 100000, 10000000, &constcoredisc, 0, 0, 0 },
    2197             : { "Dihedral", NULL,   1000,     3000, &consttabdihedral, 0, 0, 0 },
    2198             : };
    2199             : 
    2200             : static void
    2201         511 : cache_reset(long id) { caches[id].miss = caches[id].maxmiss = 0; }
    2202             : static void
    2203        9450 : cache_delete(long id) { guncloneNULL(caches[id].cache); }
    2204             : static void
    2205         525 : cache_set(long id, GEN S)
    2206             : {
    2207         525 :   GEN old = caches[id].cache;
    2208         525 :   caches[id].cache = gclone(S);
    2209         525 :   guncloneNULL(old);
    2210         525 : }
    2211             : 
    2212             : /* handle a cache miss: store stats, possibly reset table; return value
    2213             :  * if (now) cached; return NULL on failure. HACK: some caches contain an
    2214             :  * ulong where the 0 value is impossible, and return it (typecast to GEN) */
    2215             : static GEN
    2216   456723149 : cache_get(long id, ulong D)
    2217             : {
    2218   456723149 :   cache *S = &caches[id];
    2219   456723149 :   const ulong d = S->compressed? D>>1: D;
    2220             :   ulong max, l;
    2221             : 
    2222   456723149 :   if (!S->cache)
    2223             :   {
    2224         387 :     max = maxuu(minuu(D, S->maxself), S->minself);
    2225         387 :     S->init(max);
    2226         387 :     l = lg(S->cache);
    2227             :   }
    2228             :   else
    2229             :   {
    2230   456722762 :     l = lg(S->cache);
    2231   456722762 :     if (l <= d)
    2232             :     {
    2233         347 :       if (D > S->maxmiss) S->maxmiss = D;
    2234         347 :       if (DEBUGLEVEL >= 3)
    2235           0 :         err_printf("miss in cache %s: %lu, max = %lu\n",
    2236             :                    S->name, D, S->maxmiss);
    2237         347 :       if (S->miss++ >= 5 && D < S->maxself)
    2238             :       {
    2239          14 :         max = minuu(S->maxself, (long)(S->maxmiss * 1.2));
    2240          14 :         if (max <= S->maxself)
    2241             :         {
    2242          14 :           if (DEBUGLEVEL >= 3)
    2243           0 :             err_printf("resetting cache %s to %lu\n", S->name, max);
    2244          14 :           S->init(max); l = lg(S->cache);
    2245             :         }
    2246             :       }
    2247             :     }
    2248             :   }
    2249   456723149 :   return (l <= d)? NULL: gel(S->cache, d);
    2250             : }
    2251             : static GEN
    2252          70 : cache_report(long id)
    2253             : {
    2254          70 :   cache *S = &caches[id];
    2255          70 :   GEN v = zerocol(5);
    2256          70 :   gel(v,1) = strtoGENstr(S->name);
    2257          70 :   if (S->cache)
    2258             :   {
    2259          35 :     gel(v,2) = utoi(lg(S->cache)-1);
    2260          35 :     gel(v,3) = utoi(S->miss);
    2261          35 :     gel(v,4) = utoi(S->maxmiss);
    2262          35 :     gel(v,5) = utoi(gsizebyte(S->cache));
    2263             :   }
    2264          70 :   return v;
    2265             : }
    2266             : GEN
    2267          14 : getcache(void)
    2268             : {
    2269          14 :   pari_sp av = avma;
    2270          14 :   GEN M = cgetg(6, t_MAT);
    2271          14 :   gel(M,1) = cache_report(cache_FACT);
    2272          14 :   gel(M,2) = cache_report(cache_DIV);
    2273          14 :   gel(M,3) = cache_report(cache_H);
    2274          14 :   gel(M,4) = cache_report(cache_D);
    2275          14 :   gel(M,5) = cache_report(cache_DIH);
    2276          14 :   return gc_GEN(av, shallowtrans(M));
    2277             : }
    2278             : 
    2279             : void
    2280        1890 : pari_close_mf(void)
    2281             : {
    2282        1890 :   cache_delete(cache_FACT);
    2283        1890 :   cache_delete(cache_DIV);
    2284        1890 :   cache_delete(cache_H);
    2285        1890 :   cache_delete(cache_D);
    2286        1890 :   cache_delete(cache_DIH);
    2287        1890 : }
    2288             : 
    2289             : /*************************************************************************/
    2290             : /* a odd, update local cache (recycle memory) */
    2291             : static GEN
    2292        2131 : update_factor_cache(long a, long lim, long *pb)
    2293             : {
    2294        2131 :   const long step = 16000; /* even; don't increase this: RAM cache thrashing */
    2295        2131 :   if (a + 2*step > lim)
    2296         200 :     *pb = lim; /* fuse last 2 chunks */
    2297             :   else
    2298        1931 :     *pb = a + step;
    2299        2131 :   return vecfactoroddu_i(a, *pb);
    2300             : }
    2301             : /* assume lim < MAX_LONG/8 */
    2302             : static void
    2303          48 : constcoredisc(long lim)
    2304             : {
    2305          48 :   pari_sp av2, av = avma;
    2306          48 :   GEN D = caches[cache_D].cache, CACHE = NULL;
    2307          48 :   long cachea, cacheb, N, LIM = !D ? 4 : lg(D)-1;
    2308          48 :   if (lim <= 0) lim = 5;
    2309          48 :   if (lim <= LIM) return;
    2310          48 :   cache_reset(cache_D);
    2311          48 :   D = zero_zv(lim);
    2312          40 :   av2 = avma;
    2313          40 :   cachea = cacheb = 0;
    2314     4761995 :   for (N = 1; N <= lim; N+=2)
    2315             :   { /* N odd */
    2316             :     long i, d, d2;
    2317             :     GEN F;
    2318     4761947 :     if (N > cacheb)
    2319             :     {
    2320         580 :       set_avma(av2); cachea = N;
    2321         580 :       CACHE = update_factor_cache(N, lim, &cacheb);
    2322             :     }
    2323     4761947 :     F = gel(CACHE, ((N-cachea)>>1)+1); /* factoru(N) */
    2324     4761947 :     D[N] = d = corediscs_fact(F); /* = 3 mod 4 or 4 mod 16 */
    2325     4761899 :     d2 = odd(d)? d<<3: d<<1;
    2326     4761955 :     for (i = 1;;)
    2327             :     {
    2328     6349390 :       if ((N << i) > lim) break;
    2329     3174954 :       D[N<<i] = d2; i++;
    2330     3174954 :       if ((N << i) > lim) break;
    2331     1587435 :       D[N<<i] = d; i++;
    2332             :     }
    2333             :   }
    2334          48 :   cache_set(cache_D, D);
    2335          48 :   set_avma(av);
    2336             : }
    2337             : 
    2338             : static void
    2339         187 : constfact(long lim)
    2340             : {
    2341             :   pari_sp av;
    2342         187 :   GEN VFACT = caches[cache_FACT].cache;
    2343         187 :   long LIM = VFACT? lg(VFACT)-1: 4;
    2344         187 :   if (lim <= 0) lim = 5;
    2345         187 :   if (lim <= LIM) return;
    2346         159 :   cache_reset(cache_FACT); av = avma;
    2347         159 :   cache_set(cache_FACT, vecfactoru_i(1,lim)); set_avma(av);
    2348             : }
    2349             : static void
    2350         152 : constdiv(long lim)
    2351             : {
    2352             :   pari_sp av;
    2353         152 :   GEN VFACT, VDIV = caches[cache_DIV].cache;
    2354         152 :   long N, LIM = VDIV? lg(VDIV)-1: 4;
    2355         152 :   if (lim <= 0) lim = 5;
    2356         152 :   if (lim <= LIM) return;
    2357         152 :   constfact(lim);
    2358         152 :   VFACT = caches[cache_FACT].cache;
    2359         152 :   cache_reset(cache_DIV); av = avma;
    2360         152 :   VDIV  = cgetg(lim+1, t_VEC);
    2361     7410093 :   for (N = 1; N <= lim; N++) gel(VDIV,N) = divisorsu_fact(gel(VFACT,N));
    2362         152 :   cache_set(cache_DIV, VDIV); set_avma(av);
    2363             : }
    2364             : 
    2365             : /* n > 1, D = divisors(n); sets L = 2*lambda(n), S = sigma(n) */
    2366             : static void
    2367    14641320 : lamsig(GEN D, long *pL, long *pS)
    2368             : {
    2369    14641320 :   pari_sp av = avma;
    2370    14641320 :   long i, l = lg(D), L = 1, S = D[l-1]+1;
    2371    52758359 :   for (i = 2; i < l; i++) /* skip d = 1 */
    2372             :   {
    2373    53354398 :     long d = D[i], nd = D[l-i]; /* nd = n/d */
    2374    53354398 :     if (d < nd) { L += d; S += d + nd; }
    2375             :     else
    2376             :     {
    2377    15237359 :       L <<= 1; if (d == nd) { L += d; S += d; }
    2378    15237359 :       break;
    2379             :     }
    2380             :   }
    2381    14641320 :   set_avma(av); *pL = L; *pS = S;
    2382    15326608 : }
    2383             : /* table of 6 * Hurwitz class numbers D <= lim */
    2384             : static void
    2385         152 : consttabh(long lim)
    2386             : {
    2387         152 :   pari_sp av = avma, av2;
    2388         152 :   GEN VHDH0, VDIV, CACHE = NULL;
    2389         152 :   GEN VHDH = caches[cache_H].cache;
    2390         152 :   long r, N, cachea, cacheb, lim0 = VHDH? lg(VHDH)-1: 2, LIM = lim0 << 1;
    2391             : 
    2392         152 :   if (lim <= 0) lim = 5;
    2393         152 :   if (lim <= LIM) return;
    2394         152 :   cache_reset(cache_H);
    2395         152 :   r = lim&3L; if (r) lim += 4-r;
    2396         152 :   cache_get(cache_DIV, lim);
    2397         152 :   VDIV = caches[cache_DIV].cache;
    2398         152 :   VHDH0 = cgetg(lim/2 + 1, t_VECSMALL);
    2399         152 :   VHDH0[1] = 2;
    2400         152 :   VHDH0[2] = 3;
    2401      520388 :   for (N = 3; N <= lim0; N++) VHDH0[N] = VHDH[N];
    2402         152 :   av2 = avma;
    2403         152 :   cachea = cacheb = 0;
    2404     7873111 :   for (N = LIM + 3; N <= lim; N += 4)
    2405             :   {
    2406     7902254 :     long s = 0, limt = usqrt(N>>2), flsq = 0, ind, t, L, S;
    2407             :     GEN DN, DN2;
    2408     7856696 :     if (N + 2 >= lg(VDIV))
    2409             :     { /* use local cache */
    2410             :       GEN F;
    2411     6057785 :       if (N + 2 > cacheb)
    2412             :       {
    2413        1551 :         set_avma(av2); cachea = N;
    2414        1551 :         CACHE = update_factor_cache(N, lim+2, &cacheb);
    2415             :       }
    2416     6057785 :       F = gel(CACHE, ((N-cachea)>>1)+1); /* factoru(N) */
    2417     6057785 :       DN = divisorsu_fact(F);
    2418     5930343 :       F = gel(CACHE, ((N-cachea)>>1)+2); /* factoru(N+2) */
    2419     5930343 :       DN2 = divisorsu_fact(F);
    2420             :     }
    2421             :     else
    2422             :     { /* use global cache */
    2423     1798911 :       DN = gel(VDIV,N);
    2424     1798911 :       DN2 = gel(VDIV,N+2);
    2425             :     }
    2426     7685549 :     ind = N >> 1;
    2427   937594672 :     for (t = 1; t <= limt; t++)
    2428             :     {
    2429   929909123 :       ind -= (t<<2)-2; /* N/2 - 2t^2 */
    2430   929909123 :       if (ind) s += VHDH0[ind]; else flsq = 1;
    2431             :     }
    2432     7685549 :     lamsig(DN, &L,&S);
    2433     7615367 :     VHDH0[N >> 1] = 2*S - 3*L - 2*s + flsq;
    2434     7615367 :     s = 0; flsq = 0; limt = (usqrt(N+2) - 1) >> 1;
    2435     7663142 :     ind = (N+1) >> 1;
    2436   934131351 :     for (t = 1; t <= limt; t++)
    2437             :     {
    2438   926468209 :       ind -= t<<2; /* (N+1)/2 - 2t(t+1) */
    2439   926468209 :       if (ind) s += VHDH0[ind]; else flsq = 1;
    2440             :     }
    2441     7663142 :     lamsig(DN2, &L,&S);
    2442     7872959 :     VHDH0[(N+1) >> 1] = S - 3*(L >> 1) - s - flsq;
    2443             :   }
    2444          89 :   cache_set(cache_H, VHDH0); set_avma(av);
    2445             : }
    2446             : 
    2447             : /*************************************************************************/
    2448             : /* Core functions using factorizations, divisors of class numbers caches */
    2449             : /* TODO: myfactoru and factorization cache should be exported */
    2450             : static GEN
    2451    34098912 : myfactoru(long N)
    2452             : {
    2453    34098912 :   GEN z = cache_get(cache_FACT, N);
    2454    34098912 :   return z? gcopy(z): factoru(N);
    2455             : }
    2456             : static GEN
    2457    71683095 : mydivisorsu(long N)
    2458             : {
    2459    71683095 :   GEN z = cache_get(cache_DIV, N);
    2460    71683095 :   return z? leafcopy(z): divisorsu(N);
    2461             : }
    2462             : /* write -n = Df^2, D < 0 fundamental discriminant. Return D, set f. */
    2463             : static long
    2464   179330287 : mycoredisc2neg(ulong n, long *pf)
    2465             : {
    2466   179330287 :   ulong m, D = (ulong)cache_get(cache_D, n);
    2467   179330287 :   if (D) { *pf = usqrt(n/D); return -(long)D; }
    2468          58 :   m = mycore(n, pf);
    2469          58 :   if ((m&3) != 3) { m <<= 2; *pf >>= 1; }
    2470          58 :   return (long)-m;
    2471             : }
    2472             : /* write n = Df^2, D > 0 fundamental discriminant. Return D, set f. */
    2473             : static long
    2474          14 : mycoredisc2pos(ulong n, long *pf)
    2475             : {
    2476          14 :   ulong m = mycore(n, pf);
    2477          14 :   if ((m&3) != 1) { m <<= 2; *pf >>= 1; }
    2478          14 :   return (long)m;
    2479             : }
    2480             : 
    2481             : /* D < 0 fundamental. Return 6*hclassno(-D); faster than quadclassunit up
    2482             :  * to 5*10^5 or so */
    2483             : static ulong
    2484          54 : hclassno6_count(long D)
    2485             : {
    2486          54 :   ulong a, b, b2, h = 0, d = -D;
    2487          54 :   int f = 0;
    2488             : 
    2489          54 :   if (d > 500000) return 6 * quadclassnos(D);
    2490             :   /* this part would work with -d non fundamental */
    2491          47 :   b = d&1; b2 = (1+d)>>2;
    2492          47 :   if (!b)
    2493             :   {
    2494        1036 :     for (a=1; a*a<b2; a++)
    2495        1032 :       if (b2%a == 0) h++;
    2496           4 :     f = (a*a==b2); b=2; b2=(4+d)>>2;
    2497             :   }
    2498        9058 :   while (b2*3 < d)
    2499             :   {
    2500        9011 :     if (b2%b == 0) h++;
    2501     1445922 :     for (a=b+1; a*a < b2; a++)
    2502     1436911 :       if (b2%a == 0) h += 2;
    2503        9011 :     if (a*a == b2) h++;
    2504        9011 :     b += 2; b2 = (b*b+d)>>2;
    2505             :   }
    2506          47 :   if (b2*3 == d) return 6*h+2;
    2507          47 :   if (f) return 6*h+3;
    2508          47 :   return 6*h;
    2509             : }
    2510             : /* D0 < 0; 6 * hclassno(-D), using D = D0*F^2 */
    2511             : static long
    2512          77 : hclassno6u_2(long D0, long F)
    2513             : {
    2514             :   long h;
    2515          77 :   if (F == 1) h = hclassno6_count(D0);
    2516             :   else
    2517             :   { /* second chance */
    2518          23 :     h = (ulong)cache_get(cache_H, -D0);
    2519          23 :     if (!h) h = hclassno6_count(D0);
    2520          23 :     h *= uhclassnoF_fact(myfactoru(F), D0);
    2521             :   }
    2522          77 :   return h;
    2523             : }
    2524             : /* D > 0; 6 * hclassno(D) (6*Hurwitz). Beware, cached value for D (=0,3 mod 4)
    2525             :  * is stored at D>>1 */
    2526             : ulong
    2527     2503778 : hclassno6u(ulong D)
    2528             : {
    2529     2503778 :   ulong z = (ulong)cache_get(cache_H, D);
    2530             :   long D0, F;
    2531     2503778 :   if (z) return z;
    2532          77 :   D0 = mycoredisc2neg(D, &F);
    2533          77 :   return hclassno6u_2(D0,F);
    2534             : }
    2535             : /* same as hclassno6u without creating caches */
    2536             : ulong
    2537       86919 : hclassno6u_no_cache(ulong D)
    2538             : {
    2539       86919 :   cache *S = &caches[cache_H];
    2540             :   long D0, F;
    2541       86919 :   if (S->cache)
    2542             :   {
    2543       80138 :     const ulong d = D>>1; /* compressed */
    2544       80138 :     if ((ulong)lg(S->cache) > d) return S->cache[d];
    2545             :   }
    2546       86648 :   S = &caches[cache_D];
    2547       86648 :   if (!S->cache || (ulong)lg(S->cache) <= D) return 0;
    2548           0 :   D0 = mycoredisc2neg(D, &F);
    2549           0 :   return hclassno6u_2(D0,F);
    2550             : }
    2551             : /* same, where the decomposition D = D0*F^2 is already known */
    2552             : static ulong
    2553   158346131 : hclassno6u_i(ulong D, long D0, long F)
    2554             : {
    2555   158346131 :   ulong z = (ulong)cache_get(cache_H, D);
    2556   158346131 :   if (z) return z;
    2557           0 :   return hclassno6u_2(D0,F);
    2558             : }
    2559             : 
    2560             : /* D < -4 fundamental, 6 * h(D), ordinary class number */
    2561             : static long
    2562    10748416 : hclassno6u_fund(long D)
    2563             : {
    2564    10748416 :   ulong z = (ulong)cache_get(cache_H, -D);
    2565    10748416 :   return z? z: 6 * quadclassnos(D);
    2566             : }
    2567             : 
    2568             : /*************************************************************************/
    2569             : /*                          TRACE FORMULAS                               */
    2570             : /* CHIP primitive, initialize for t_POLMOD output */
    2571             : static GEN
    2572       33551 : mfcharinit(GEN CHIP)
    2573             : {
    2574       33551 :   long n, o, l, vt, N = mfcharmodulus(CHIP);
    2575             :   GEN c, v, V, G, Pn;
    2576       33551 :   if (N == 1) return mkvec2(mkvec(gen_1), pol_x(0));
    2577        5831 :   G = gel(CHIP,1);
    2578        5831 :   v = ncharvecexpo(G, znconrey_normalized(G, gel(CHIP,2)));
    2579        5831 :   l = lg(v); V = cgetg(l, t_VEC);
    2580        5831 :   o = mfcharorder(CHIP);
    2581        5831 :   Pn = mfcharpol(CHIP); vt = varn(Pn);
    2582        5831 :   if (o <= 2)
    2583             :   {
    2584       60620 :     for (n = 1; n < l; n++)
    2585             :     {
    2586       55867 :       if (v[n] < 0) c = gen_0; else c = v[n]? gen_m1: gen_1;
    2587       55867 :       gel(V,n) = c;
    2588             :     }
    2589             :   }
    2590             :   else
    2591             :   {
    2592       17591 :     for (n = 1; n < l; n++)
    2593             :     {
    2594       16513 :       if (v[n] < 0) c = gen_0;
    2595             :       else
    2596             :       {
    2597        9394 :         c = Qab_zeta(v[n], o, vt);
    2598        9394 :         if (typ(c) == t_POL && lg(c) >= lg(Pn)) c = RgX_rem(c, Pn);
    2599             :       }
    2600       16513 :       gel(V,n) = c;
    2601             :     }
    2602             :   }
    2603        5831 :   return mkvec2(V, Pn);
    2604             : }
    2605             : static GEN
    2606      428379 : vchip_lift(GEN VCHI, long x, GEN C)
    2607             : {
    2608      428379 :   GEN V = gel(VCHI,1);
    2609      428379 :   long F = lg(V)-1;
    2610      428379 :   if (F == 1) return C;
    2611       33005 :   x %= F;
    2612       33005 :   if (!x) return C;
    2613       33005 :   if (x <= 0) x += F;
    2614       33005 :   return gmul(C, gel(V, x));
    2615             : }
    2616             : static long
    2617   282559972 : vchip_FC(GEN VCHI) { return lg(gel(VCHI,1))-1; }
    2618             : static GEN
    2619     6586245 : vchip_mod(GEN VCHI, GEN S)
    2620     6586245 : { return (typ(S) == t_POL)? RgX_rem(S, gel(VCHI,2)): S; }
    2621             : static GEN
    2622     1992428 : vchip_polmod(GEN VCHI, GEN S)
    2623     1992428 : { return (typ(S) == t_POL)? mkpolmod(S, gel(VCHI,2)): S; }
    2624             : 
    2625             : /* contribution of scalar matrices in dimension formula */
    2626             : static GEN
    2627      366373 : A1(long N, long k) { return uutoQ(mypsiu(N)*(k-1), 12); }
    2628             : static long
    2629        7756 : ceilA1(long N, long k) { return ceildivuu(mypsiu(N) * (k-1), 12); }
    2630             : 
    2631             : /* sturm bound, slightly larger than dimension */
    2632             : long
    2633       22036 : mfsturmNk(long N, long k) { return (mypsiu(N) * k) / 12; }
    2634             : long
    2635        3346 : mfsturmNgk(long N, GEN k)
    2636             : {
    2637        3346 :   long n,d; Qtoss(k,&n,&d);
    2638        3346 :   return 1 + (mypsiu(N)*n)/(d == 1? 12: 24);
    2639             : }
    2640             : static long
    2641         427 : mfsturmmf(GEN F) { return mfsturmNgk(mf_get_N(F), mf_get_gk(F)); }
    2642             : 
    2643             : /* List of all solutions of x^2 + x + 1 = 0 modulo N, x modulo N */
    2644             : static GEN
    2645         581 : sqrtm3modN(long N)
    2646             : {
    2647             :   pari_sp av;
    2648             :   GEN fa, P, E, B, mB, A, Q, T, R, v, gen_m3;
    2649         581 :   long l, i, n, ct, fl3 = 0, Ninit;
    2650         581 :   if (!odd(N) || (N%9) == 0) return cgetg(1,t_VECSMALL);
    2651         553 :   Ninit = N;
    2652         553 :   if ((N%3) == 0) { N /= 3; fl3 = 1; }
    2653         553 :   fa = myfactoru(N); P = gel(fa, 1); E = gel(fa, 2);
    2654         553 :   l = lg(P);
    2655         749 :   for (i = 1; i < l; i++)
    2656         560 :     if ((P[i]%3) == 2) return cgetg(1,t_VECSMALL);
    2657         189 :   A = cgetg(l, t_VECSMALL);
    2658         189 :   B = cgetg(l, t_VECSMALL);
    2659         189 :   mB= cgetg(l, t_VECSMALL);
    2660         189 :   Q = cgetg(l, t_VECSMALL); gen_m3 = utoineg(3);
    2661         385 :   for (i = 1; i < l; i++)
    2662             :   {
    2663         196 :     long p = P[i], e = E[i];
    2664         196 :     Q[i] = upowuu(p,e);
    2665         196 :     B[i] = itou( Zp_sqrt(gen_m3, utoipos(p), e) );
    2666         196 :     mB[i]= Q[i] - B[i];
    2667             :   }
    2668         189 :   ct = 1 << (l-1);
    2669         189 :   T = ZV_producttree(Q);
    2670         189 :   R = ZV_chinesetree(Q,T);
    2671         189 :   v = cgetg(ct+1, t_VECSMALL);
    2672         189 :   av = avma;
    2673         581 :   for (n = 1; n <= ct; n++)
    2674             :   {
    2675         392 :     long m = n-1, r;
    2676         812 :     for (i = 1; i < l; i++)
    2677             :     {
    2678         420 :       A[i] = (m&1L)? mB[i]: B[i];
    2679         420 :       m >>= 1;
    2680             :     }
    2681         392 :     r = itou( ZV_chinese_tree(A, Q, T, R) );
    2682         462 :     if (fl3) while (r%3) r += N;
    2683         392 :     set_avma(av); v[n] = odd(r) ? (r-1) >> 1 : (r+Ninit-1) >> 1;
    2684             :   }
    2685         189 :   return v;
    2686             : }
    2687             : 
    2688             : /* number of elliptic points of order 3 in X0(N) */
    2689             : static long
    2690       10220 : nu3(long N)
    2691             : {
    2692             :   long i, l;
    2693             :   GEN P;
    2694       10220 :   if (!odd(N) || (N%9) == 0) return 0;
    2695        8995 :   if ((N%3) == 0) N /= 3;
    2696        8995 :   P = gel(myfactoru(N), 1); l = lg(P);
    2697       13195 :   for (i = 1; i < l; i++) if ((P[i]%3) == 2) return 0;
    2698        4018 :   return 1L<<(l-1);
    2699             : }
    2700             : /* number of elliptic points of order 2 in X0(N) */
    2701             : static long
    2702       17647 : nu2(long N)
    2703             : {
    2704             :   long i, l;
    2705             :   GEN P;
    2706       17647 :   if ((N&3L) == 0) return 0;
    2707       17647 :   if (!odd(N)) N >>= 1;
    2708       17647 :   P = gel(myfactoru(N), 1); l = lg(P);
    2709       22078 :   for (i = 1; i < l; i++) if ((P[i]&3L) == 3) return 0;
    2710        3969 :   return 1L<<(l-1);
    2711             : }
    2712             : 
    2713             : /* contribution of elliptic matrices of order 3 in dimension formula
    2714             :  * Only depends on CHIP the primitive char attached to CHI */
    2715             : static GEN
    2716       44135 : A21(long N, long k, GEN CHI)
    2717             : {
    2718             :   GEN res, G, chi, o;
    2719             :   long a21, i, limx, S;
    2720       44135 :   if ((N&1L) == 0) return gen_0;
    2721       21371 :   a21 = k%3 - 1;
    2722       21371 :   if (!a21) return gen_0;
    2723       20531 :   if (N <= 3) return sstoQ(a21, 3);
    2724       10801 :   if (!CHI) return sstoQ(nu3(N) * a21, 3);
    2725         581 :   res = sqrtm3modN(N); limx = (N - 1) >> 1;
    2726         581 :   G = gel(CHI,1); chi = gel(CHI,2);
    2727         581 :   o = gmfcharorder(CHI);
    2728         973 :   for (S = 0, i = 1; i < lg(res); i++)
    2729             :   { /* (x,N) = 1; S += chi(x) + chi(x^2) */
    2730         392 :     long x = res[i];
    2731         392 :     if (x <= limx)
    2732             :     { /* CHI(x)=e(c/o), 3rd-root of 1 */
    2733         196 :       GEN c = znchareval(G, chi, utoi(x), o);
    2734         196 :       if (!signe(c)) S += 2; else S--;
    2735             :     }
    2736             :   }
    2737         581 :   return sstoQ(a21 * S, 3);
    2738             : }
    2739             : 
    2740             : /* List of all square roots of -1 modulo N */
    2741             : static GEN
    2742         595 : sqrtm1modN(long N)
    2743             : {
    2744             :   pari_sp av;
    2745             :   GEN fa, P, E, B, mB, A, Q, T, R, v;
    2746         595 :   long l, i, n, ct, fleven = 0;
    2747         595 :   if ((N&3L) == 0) return cgetg(1,t_VECSMALL);
    2748         595 :   if ((N&1L) == 0) { N >>= 1; fleven = 1; }
    2749         595 :   fa = myfactoru(N); P = gel(fa,1); E = gel(fa,2);
    2750         595 :   l = lg(P);
    2751         945 :   for (i = 1; i < l; i++)
    2752         665 :     if ((P[i]&3L) == 3) return cgetg(1,t_VECSMALL);
    2753         280 :   A = cgetg(l, t_VECSMALL);
    2754         280 :   B = cgetg(l, t_VECSMALL);
    2755         280 :   mB= cgetg(l, t_VECSMALL);
    2756         280 :   Q = cgetg(l, t_VECSMALL);
    2757         574 :   for (i = 1; i < l; i++)
    2758             :   {
    2759         294 :     long p = P[i], e = E[i];
    2760         294 :     Q[i] = upowuu(p,e);
    2761         294 :     B[i] = itou( Zp_sqrt(gen_m1, utoipos(p), e) );
    2762         294 :     mB[i]= Q[i] - B[i];
    2763             :   }
    2764         280 :   ct = 1 << (l-1);
    2765         280 :   T = ZV_producttree(Q);
    2766         280 :   R = ZV_chinesetree(Q,T);
    2767         280 :   v = cgetg(ct+1, t_VECSMALL);
    2768         280 :   av = avma;
    2769         868 :   for (n = 1; n <= ct; n++)
    2770             :   {
    2771         588 :     long m = n-1, r;
    2772        1232 :     for (i = 1; i < l; i++)
    2773             :     {
    2774         644 :       A[i] = (m&1L)? mB[i]: B[i];
    2775         644 :       m >>= 1;
    2776             :     }
    2777         588 :     r = itou( ZV_chinese_tree(A, Q, T, R) );
    2778         588 :     if (fleven && !odd(r)) r += N;
    2779         588 :     set_avma(av); v[n] = r;
    2780             :   }
    2781         280 :   return v;
    2782             : }
    2783             : 
    2784             : /* contribution of elliptic matrices of order 4 in dimension formula.
    2785             :  * Only depends on CHIP the primitive char attached to CHI */
    2786             : static GEN
    2787       44135 : A22(long N, long k, GEN CHI)
    2788             : {
    2789             :   GEN G, chi, o, res;
    2790             :   long S, a22, i, limx, o2;
    2791       44135 :   if ((N&3L) == 0) return gen_0;
    2792       30380 :   a22 = (k & 3L) - 1; /* (k % 4) - 1 */
    2793       30380 :   if (!a22) return gen_0;
    2794       30310 :   if (N <= 2) return sstoQ(a22, 4);
    2795       18452 :   if (!CHI) return sstoQ(nu2(N)*a22, 4);
    2796         805 :   if (mfcharparity(CHI) == -1) return gen_0;
    2797         595 :   res = sqrtm1modN(N); limx = (N - 1) >> 1;
    2798         595 :   G = gel(CHI,1); chi = gel(CHI,2);
    2799         595 :   o = gmfcharorder(CHI);
    2800         595 :   o2 = itou(o)>>1;
    2801        1183 :   for (S = 0, i = 1; i < lg(res); i++)
    2802             :   { /* (x,N) = 1, S += real(chi(x)) */
    2803         588 :     long x = res[i];
    2804         588 :     if (x <= limx)
    2805             :     { /* CHI(x)=e(c/o), 4th-root of 1 */
    2806         294 :       long c = itou( znchareval(G, chi, utoi(x), o) );
    2807         294 :       if (!c) S++; else if (c == o2) S--;
    2808             :     }
    2809             :   }
    2810         595 :   return sstoQ(a22 * S, 2);
    2811             : }
    2812             : 
    2813             : /* sumdiv(N,d,eulerphi(gcd(d,N/d))) */
    2814             : static long
    2815       39116 : nuinf(long N)
    2816             : {
    2817       39116 :   GEN fa = myfactoru(N), P = gel(fa,1), E = gel(fa,2);
    2818       39116 :   long i, t = 1, l = lg(P);
    2819       82999 :   for (i=1; i<l; i++)
    2820             :   {
    2821       43883 :     long p = P[i], e = E[i];
    2822       43883 :     if (odd(e))
    2823       35091 :       t *= upowuu(p,e>>1) << 1;
    2824             :     else
    2825        8792 :       t *= upowuu(p,(e>>1)-1) * (p+1);
    2826             :   }
    2827       39116 :   return t;
    2828             : }
    2829             : 
    2830             : /* contribution of hyperbolic matrices in dimension formula */
    2831             : static GEN
    2832       44583 : A3(long N, long FC)
    2833             : {
    2834             :   long i, S, NF, l;
    2835             :   GEN D;
    2836       44583 :   if (FC == 1) return uutoQ(nuinf(N),2);
    2837        5467 :   D = mydivisorsu(N); l = lg(D);
    2838        5467 :   S = 0; NF = N/FC;
    2839       42840 :   for (i = 1; i < l; i++)
    2840             :   {
    2841       37373 :     long g = ugcd(D[i], D[l-i]);
    2842       37373 :     if (NF%g == 0) S += myeulerphiu(g);
    2843             :   }
    2844        5467 :   return uutoQ(S, 2);
    2845             : }
    2846             : 
    2847             : /* special contribution in weight 2 in dimension formula */
    2848             : static long
    2849       43666 : A4(long k, long FC)
    2850       43666 : { return (k==2 && FC==1)? 1: 0; }
    2851             : /* gcd(x,N) */
    2852             : static long
    2853   288402905 : myugcd(GEN GCD, ulong x)
    2854             : {
    2855   288402905 :   ulong N = lg(GCD)-1;
    2856   288402905 :   if (x >= N) x %= N;
    2857   288402905 :   return GCD[x+1];
    2858             : }
    2859             : /* 1_{gcd(x,N) = 1} * chi(x), return NULL if 0 */
    2860             : static GEN
    2861   411157547 : mychicgcd(GEN GCD, GEN VCHI, long x)
    2862             : {
    2863   411157547 :   long N = lg(GCD)-1;
    2864   411157547 :   if (N == 1) return gen_1;
    2865   336008349 :   x = umodsu(x, N);
    2866   336008349 :   if (GCD[x+1] != 1) return NULL;
    2867   274839143 :   x %= vchip_FC(VCHI); if (!x) return gen_1;
    2868     6668046 :   return gel(gel(VCHI,1), x);
    2869             : }
    2870             : 
    2871             : /* contribution of scalar matrices to trace formula */
    2872             : static GEN
    2873     6564045 : TA1(long N, long k, GEN VCHI, GEN GCD, long n)
    2874             : {
    2875             :   GEN S;
    2876             :   ulong m;
    2877     6564045 :   if (!uissquareall(n, &m)) return gen_0;
    2878      398979 :   if (m == 1) return A1(N,k); /* common */
    2879      358113 :   S = mychicgcd(GCD, VCHI, m);
    2880      358113 :   return S? gmul(gmul(powuu(m, k-2), A1(N,k)), S): gen_0;
    2881             : }
    2882             : 
    2883             : /* All square roots modulo 4N, x modulo 2N, precomputed to accelerate TA2 */
    2884             : static GEN
    2885      129171 : mksqr(long N)
    2886             : {
    2887      129171 :   pari_sp av = avma;
    2888      129171 :   long x, N2 = N << 1, N4 = N << 2;
    2889      129171 :   GEN v = const_vec(N2, cgetg(1, t_VECSMALL));
    2890      129171 :   gel(v, N2) = mkvecsmall(0); /* x = 0 */
    2891     3523569 :   for (x = 1; x <= N; x++)
    2892             :   {
    2893     3394398 :     long r = (((x*x - 1)%N4) >> 1) + 1;
    2894     3394398 :     gel(v,r) = vecsmall_append(gel(v,r), x);
    2895             :   }
    2896      129171 :   return gc_GEN(av, v);
    2897             : }
    2898             : 
    2899             : static GEN
    2900      129171 : mkgcd(long N)
    2901             : {
    2902             :   GEN GCD, d;
    2903             :   long i, N2;
    2904      129171 :   if (N == 1) return mkvecsmall(N);
    2905      106246 :   GCD = cgetg(N + 1, t_VECSMALL);
    2906      106246 :   d = GCD+1; /* GCD[i+1] = d[i] = gcd(i,N) = gcd(N-i,N), i = 0..N-1 */
    2907      106246 :   d[0] = N; d[1] = d[N-1] = 1; N2 = N>>1;
    2908     1664670 :   for (i = 2; i <= N2; i++) d[i] = d[N-i] = ugcd(N, i);
    2909      106246 :   return GCD;
    2910             : }
    2911             : 
    2912             : /* Table of \sum_{x^2-tx+n=0 mod Ng}chi(x) for all g dividing gcd(N,F),
    2913             :  * F^2 largest such that (t^2-4n)/F^2=0 or 1 mod 4; t >= 0 */
    2914             : static GEN
    2915    16069092 : mutglistall(long t, long N, long NF, GEN VCHI, long n, GEN MUP, GEN L, GEN GCD)
    2916             : {
    2917    16069092 :   long i, lx = lg(L);
    2918    16069092 :   GEN DNF = mydivisorsu(NF), v = zerovec(NF);
    2919    16069092 :   long j, g, lDNF = lg(DNF);
    2920    44948705 :   for (i = 1; i < lx; i++)
    2921             :   {
    2922    28879613 :     long x = (L[i] + t) >> 1, y, lD;
    2923    28879613 :     GEN D, c = mychicgcd(GCD, VCHI, x);
    2924    28879613 :     if (L[i] && L[i] != N)
    2925             :     {
    2926    19329393 :       GEN c2 = mychicgcd(GCD, VCHI, t - x);
    2927    19329393 :       if (c2) c = c? gadd(c, c2): c2;
    2928             :     }
    2929    28879613 :     if (!c) continue;
    2930    22310113 :     y = (x*(x - t) + n) / N; /* exact division */
    2931    22310113 :     D = mydivisorsu(ugcd(labs(y), NF)); lD = lg(D);
    2932    60130499 :     for (j=1; j < lD; j++) { g = D[j]; gel(v,g) = gadd(gel(v,g), c); }
    2933             :   }
    2934             :   /* j = 1 corresponds to g = 1, and MUP[1] = 1 */
    2935    37214644 :   for (j=2; j < lDNF; j++) { g = DNF[j]; gel(v,g) = gmulsg(MUP[g], gel(v,g)); }
    2936    16069092 :   return v;
    2937             : }
    2938             : 
    2939             : /* special case (N,F) = 1: easier */
    2940             : static GEN
    2941   163261104 : mutg1(long t, long N, GEN VCHI, GEN L, GEN GCD)
    2942             : {
    2943   163261104 :   GEN S = NULL;
    2944   163261104 :   long i, lx = lg(L);
    2945   342969170 :   for (i = 1; i < lx; i++)
    2946             :   {
    2947   179708066 :     long x = (L[i] + t) >> 1;
    2948   179708066 :     GEN c = mychicgcd(GCD, VCHI, x);
    2949   179708066 :     if (c) S = S? gadd(S, c): c;
    2950   179708066 :     if (L[i] && L[i] != N)
    2951             :     {
    2952    99891078 :       c = mychicgcd(GCD, VCHI, t - x);
    2953    99891078 :       if (c) S = S? gadd(S, c): c;
    2954             :     }
    2955   179708066 :     if (S && !signe(S)) S = NULL; /* strive hard to add gen_0 */
    2956             :   }
    2957   163261104 :   return S; /* single value */
    2958             : }
    2959             : 
    2960             : /* n > 2, return P_n = \sum_{0<=j<=n/2} (-1)^j binomial(n-j,j) X^j
    2961             :  * (2x)^n P_n (1 / (4x^2)) = polchebyshev(n, 2) */
    2962             : GEN
    2963      402722 : mfrhopol(long n)
    2964             : {
    2965             : #ifdef LONG_IS_64BIT
    2966      345234 :   const long M = 2642249;
    2967             : #else
    2968       57488 :   const long M = 1629;
    2969             : #endif
    2970      402722 :   long j, d = n >> 1; /* >= 1 */
    2971      402722 :   GEN P = cgetg(d + 3, t_POL);
    2972             : 
    2973      402722 :   if (n > M) pari_err_IMPL("mfrhopol for large weight"); /* avoid overflow */
    2974      402722 :   P[1] = evalvarn(0)|evalsigne(1);
    2975      402722 :   gel(P,2) = gen_1;
    2976      402722 :   gel(P,3) = utoineg(n-1); /* j = 1 */
    2977      402722 :   if (d > 1) gel(P,4) = utoipos(((n-3)*(n-2)) >> 1); /* j = 2 */
    2978      402722 :   if (d > 2) gel(P,5) = utoineg(((n-5)*(n-4)*(n-3)) / 6); /* j = 3 */
    2979     1608783 :   for (j = 4; j <= d; j++)
    2980     1206061 :     gel(P,j+2) = diviuexact(mulis(gel(P,j+1), -(n-2*j+1)*(n-2*j+2)), (n-j+1)*j);
    2981      402722 :   return P;
    2982             : }
    2983             : 
    2984             : /* polrecip(Q)(x), assume Q(0) = 1 */
    2985             : GEN
    2986     4054376 : mfrhopol_u_eval(GEN Q, ulong x)
    2987             : {
    2988     4054376 :   GEN T = addiu(gel(Q,3), x);
    2989     4054376 :   long l = lg(Q), j;
    2990    40991293 :   for (j = 4; j < l; j++) T = addii(gel(Q,j), mului(x, T));
    2991     4054373 :   return T;
    2992             : }
    2993             : GEN
    2994       56622 : mfrhopol_eval(GEN Q, GEN x)
    2995             : {
    2996             :   long l, j;
    2997             :   GEN T;
    2998       56622 :   if (lgefint(x) == 3) return mfrhopol_u_eval(Q, x[2]);
    2999           0 :   l = lg(Q); T = addii(gel(Q,3), x);
    3000           0 :   for (j = 4; j < l; j++) T = addii(gel(Q,j), mulii(x, T));
    3001           0 :   return T;
    3002             : }
    3003             : /* t >= 0. If nu odd, let [N, T] = [(nu - 1)/2, t]; else let [N, T] = [nu/2, 1].
    3004             :  * We have t2 = t^2 and Q(X) = sum_{0<=j<=N} (-1)^j binomial(nu-j,j) n^j X^j
    3005             :  * U_nu(z) = polchebyshev(nu, 2, z)
    3006             :  *   = sum_{0<=j<=N} (-1)^j binomial(nu-j,j) (2z)^(nu-2*j))
    3007             :  * Return C n^(nu/2) U_nu(t / (2*sqrt(n)))
    3008             :  *   = C sum_{0<=j<=N} (-1)^j binomial(nu-j,j) n^j t^(nu - 2j)
    3009             :  *   = C T sum_{0<=j<=N} (-1)^j binomial(nu-j,j) n^j (t^2)^(N - j)
    3010             :  *   = C T polrecip(Q)(t^2); note that Q(0) = 1 */
    3011             : static GEN
    3012   170066746 : mfrhopow(GEN C, GEN Q, long nu, long t, long t2, long n)
    3013             : {
    3014             :   GEN T;
    3015   170066746 :   switch (nu)
    3016             :   {
    3017   162107834 :     case 0: return C;
    3018     2168698 :     case 1: return gmulsg(t, C);
    3019     1664866 :     case 2: return gmulsg(t2 - n, C);
    3020       51275 :     case 3: return gmul(mulss(t, t2 - 2*n), C);
    3021     4074073 :     default:
    3022     4074073 :       if (!t) return gmul(gel(Q, lg(Q) - 1), C);
    3023     3997754 :       T = mfrhopol_u_eval(Q, t2); if (odd(nu)) T = mului(t, T);
    3024     3997754 :       return gmul(T, C);
    3025             :   }
    3026             : }
    3027             : 
    3028             : static GEN
    3029   324085950 : TA2_t(long t, long N, long N4, long n, long n4, long nu, GEN Q,
    3030             :       GEN VCHI, GEN SQRTS, GEN MUP, GEN GCD)
    3031             : {
    3032   324085950 :   long F, NF, D0, t2 = t*t, D = n4 - t2; /* > 0 */
    3033   324085950 :   GEN sh, L = gel(SQRTS, (umodsu(-D - 1, N4) >> 1) + 1);
    3034             : 
    3035   324085950 :   if (lg(L) == 1) return NULL;
    3036   179330196 :   D0 = mycoredisc2neg(D, &F);
    3037   179330196 :   NF = myugcd(GCD, F);
    3038   179330196 :   if (NF == 1)
    3039             :   { /* (N,F) = 1 => single value in mutglistall */
    3040   163261104 :     GEN mut = mutg1(t, N, VCHI, L, GCD);
    3041   163261104 :     if (!mut) return NULL;
    3042   158346131 :     sh = gmulgu(mut, hclassno6u_i(D,D0,F));
    3043             :   }
    3044             :   else
    3045             :   {
    3046    16069092 :     GEN v = mutglistall(t, N, NF, VCHI, n, MUP, L, GCD);
    3047    16069092 :     GEN DF = mydivisorsu(F);
    3048    16069092 :     long i, lDF = lg(DF);
    3049    16069092 :     sh = gen_0;
    3050    64474719 :     for (i = 1; i < lDF; i++)
    3051             :     {
    3052    48405627 :       long Ff, f = DF[i], g = myugcd(GCD, f);
    3053    48405627 :       GEN mut = gel(v, g);
    3054    48405627 :       if (gequal0(mut)) continue;
    3055    31401426 :       Ff = DF[lDF-i]; /* F/f */
    3056    31401426 :       if (Ff > 1)
    3057             :       {
    3058    22452173 :         GEN P = gel(myfactoru(Ff), 1);
    3059    22452173 :         long j, lP = lg(P);
    3060    49502654 :         for (j = 1; j < lP; j++) { long p = P[j]; Ff -= kross(D0, p)*Ff/p; }
    3061    22452173 :         mut = gmulsg(Ff, mut);
    3062             :       }
    3063    31401426 :       sh = gadd(sh, mut);
    3064             :     }
    3065    16069092 :     if (gequal0(sh)) return NULL;
    3066    11720615 :     if (D0 == -3) sh = gmul2n(sh, 1);
    3067    11225060 :     else if (D0 == -4) sh = gmulgu(sh, 3);
    3068    10748416 :     else sh = gmulgu(sh, hclassno6u_fund(D0));
    3069             :   }
    3070   170066746 :   return mfrhopow(sh, Q, nu, t, t2, n);
    3071             : }
    3072             : 
    3073             : /* contribution of elliptic matrices to trace formula */
    3074             : static GEN
    3075     6564045 : TA2(long N, long k, GEN VCHI, long n, GEN SQRTS, GEN MUP, GEN GCD)
    3076             : {
    3077     6564045 :   long N4 = N << 2, n4 = n << 2, nu = k - 2;
    3078     6564045 :   long st = (!odd(N) && odd(n)) ? 2 : 1;
    3079     6564045 :   long t, limt = usqrt(n4 - 1);
    3080     6564045 :   GEN s, S = gen_0, Q = nu > 3 ? ZX_z_unscale(mfrhopol(nu), n) : NULL;
    3081             : 
    3082             :   /* actually compute 6*S to ensure integrality */
    3083   324390485 :   for (t = st; t <= limt; t += st) /* t^2 < 4n */
    3084             :   {
    3085   317826440 :     pari_sp av = avma;
    3086   317826440 :     s = TA2_t(t, N, N4, n, n4, nu, Q, VCHI, SQRTS, MUP, GCD);
    3087   317826440 :     if (s) S = gc_upto(av, gadd(S, s)); else set_avma(av);
    3088             :   }
    3089     6564045 :   if (!odd(k))
    3090             :   {
    3091     6259510 :     s = TA2_t(0, N, N4, n, n4, nu, Q, VCHI, SQRTS, MUP, GCD);
    3092             :     /* s/2 is the only term involving a denominator (= 2) */
    3093     6259510 :     if (s) S = gadd(S, gmul2n(s, -1));
    3094             :   }
    3095     6564045 :   return gdivgu(S, 6);
    3096             : }
    3097             : 
    3098             : /* compute global auxiliary data for TA3 */
    3099             : static GEN
    3100      129171 : mkbez(long N, long FC)
    3101             : {
    3102      129171 :   long ct, i, NF = N/FC;
    3103      129171 :   GEN w, D = mydivisorsu(N);
    3104      129171 :   long l = lg(D);
    3105             : 
    3106      129171 :   w = cgetg(l, t_VEC);
    3107      374556 :   for (i = ct = 1; i < l; i++)
    3108             :   {
    3109      351631 :     long u, v, h, c = D[i], Nc = D[l-i];
    3110      351631 :     if (c > Nc) break;
    3111      245385 :     h = cbezout(c, Nc, &u, &v);
    3112      245385 :     if (h == 1) /* shortcut */
    3113      177051 :       gel(w, ct++) = mkvecsmall4(1,u*c,1,i);
    3114       68334 :     else if (!(NF%h))
    3115       57582 :       gel(w, ct++) = mkvecsmall4(h,u*(c/h),myeulerphiu(h),i);
    3116             :   }
    3117      129171 :   setlg(w,ct); stackdummy((pari_sp)(w+ct),(pari_sp)(w+l));
    3118      129171 :   return w;
    3119             : }
    3120             : 
    3121             : /* contribution of hyperbolic matrices to trace formula, d * nd = n,
    3122             :  * DN = divisorsu(N) */
    3123             : static GEN
    3124    34044534 : auxsum(GEN VCHI, GEN GCD, long d, long nd, GEN DN, GEN BEZ)
    3125             : {
    3126    34044534 :   GEN S = gen_0;
    3127    34044534 :   long ct, g = nd - d, lDN = lg(DN), lBEZ = lg(BEZ);
    3128    87676498 :   for (ct = 1; ct < lBEZ; ct++)
    3129             :   {
    3130    53631964 :     GEN y, B = gel(BEZ, ct);
    3131    53631964 :     long ic, c, Nc, uch, h = B[1];
    3132    53631964 :     if (g%h) continue;
    3133    52409078 :     uch = B[2];
    3134    52409078 :     ic  = B[4];
    3135    52409078 :     c = DN[ic];
    3136    52409078 :     Nc= DN[lDN - ic]; /* Nc = N/c */
    3137    52409078 :     if (ugcd(Nc, nd) == 1)
    3138    43984571 :       y = mychicgcd(GCD, VCHI, d + uch*g); /* 0 if (c,d) > 1 */
    3139             :     else
    3140     8424507 :       y = NULL;
    3141    52409078 :     if (c != Nc && ugcd(Nc, d) == 1)
    3142             :     {
    3143    39006713 :       GEN y2 = mychicgcd(GCD, VCHI, nd - uch*g); /* 0 if (c,nd) > 1 */
    3144    39006713 :       if (y2) y = y? gadd(y, y2): y2;
    3145             :     }
    3146    52409078 :     if (y) S = gadd(S, gmulsg(B[3], y));
    3147             :   }
    3148    34044534 :   return S;
    3149             : }
    3150             : 
    3151             : static GEN
    3152     6564045 : TA3(long N, long k, GEN VCHI, GEN GCD, GEN Dn, GEN BEZ)
    3153             : {
    3154     6564045 :   GEN S = gen_0, DN = mydivisorsu(N);
    3155     6564045 :   long i, l = lg(Dn);
    3156    40608579 :   for (i = 1; i < l; i++)
    3157             :   {
    3158    40567713 :     long d = Dn[i], nd = Dn[l-i]; /* = n/d */
    3159             :     GEN t, u;
    3160    40567713 :     if (d > nd) break;
    3161    34044534 :     t = auxsum(VCHI, GCD, d, nd, DN, BEZ);
    3162    34044534 :     if (isintzero(t)) continue;
    3163    32683853 :     u = powuu(d,k-1); if (d == nd) u = gmul2n(u,-1);
    3164    32683853 :     S = gadd(S, gmul(u,t));
    3165             :   }
    3166     6564045 :   return S;
    3167             : }
    3168             : 
    3169             : /* special contribution in weight 2 in trace formula */
    3170             : static long
    3171     6564045 : TA4(long k, GEN VCHIP, GEN Dn, GEN GCD)
    3172             : {
    3173             :   long i, l, S;
    3174     6564045 :   if (k != 2 || vchip_FC(VCHIP) != 1) return 0;
    3175     5687416 :   l = lg(Dn); S = 0;
    3176    66354498 :   for (i = 1; i < l; i++)
    3177             :   {
    3178    60667082 :     long d = Dn[i]; /* gcd(N,n/d) == 1? */
    3179    60667082 :     if (myugcd(GCD, Dn[l-i]) == 1) S += d;
    3180             :   }
    3181     5687416 :   return S;
    3182             : }
    3183             : 
    3184             : /* precomputation of products occurring im mutg, again to accelerate TA2 */
    3185             : static GEN
    3186      129171 : mkmup(long N)
    3187             : {
    3188      129171 :   GEN fa = myfactoru(N), P = gel(fa,1), D = divisorsu_fact(fa);
    3189      129171 :   long i, lP = lg(P), lD = lg(D);
    3190      129171 :   GEN MUP = zero_zv(N);
    3191      129171 :   MUP[1] = 1;
    3192      452403 :   for (i = 2; i < lD; i++)
    3193             :   {
    3194      323232 :     long j, g = D[i], Ng = D[lD-i]; /*  N/g */
    3195      886011 :     for (j = 1; j < lP; j++) { long p = P[j]; if (Ng%p) g += g/p; }
    3196      323232 :     MUP[D[i]] = g;
    3197             :   }
    3198      129171 :   return MUP;
    3199             : }
    3200             : 
    3201             : /* quadratic nonresidues mod p; p odd prime, p^2 fits in a long */
    3202             : static GEN
    3203        2814 : non_residues(long p)
    3204             : {
    3205        2814 :   long i, j, p2 = p >> 1;
    3206        2814 :   GEN v = cgetg(p2+1, t_VECSMALL), w = const_vecsmall(p-1, 1);
    3207        4571 :   for (i = 2; i <= p2; i++) w[(i*i) % p] = 0; /* no need to check 1 */
    3208        9142 :   for (i = 2, j = 1; i < p; i++) if (w[i]) v[j++] = i;
    3209        2814 :   return v;
    3210             : }
    3211             : 
    3212             : /* CHIP primitive. Return t_VECSMALL v of length q such that
    3213             :  * Tr^new_{N,CHIP}(n) = 0 whenever v[(n%q) + 1] is nonzero */
    3214             : static GEN
    3215       33649 : mfnewzerodata(long N, GEN CHIP)
    3216             : {
    3217       33649 :   GEN V, M, L, faN = myfactoru(N), PN = gel(faN,1), EN = gel(faN,2);
    3218       33649 :   GEN G = gel(CHIP,1), chi = gel(CHIP,2);
    3219       33649 :   GEN fa = znstar_get_faN(G), P = ZV_to_zv(gel(fa,1)), E = gel(fa,2);
    3220       33649 :   long i, mod, j = 1, l = lg(PN);
    3221             : 
    3222       33649 :   M = cgetg(l, t_VECSMALL); M[1] = 0;
    3223       33649 :   V = cgetg(l, t_VEC);
    3224             :   /* Tr^new(n) = 0 if (n mod M[i]) in V[i]  */
    3225       33649 :   if ((N & 3) == 0)
    3226             :   {
    3227       13153 :     long e = EN[1];
    3228       13153 :     long c = (lg(P) > 1 && P[1] == 2)? E[1]: 0; /* c = v_2(FC) */
    3229             :     /* e >= 2 */
    3230       13153 :     if (c == e-1) return NULL; /* Tr^new = 0 */
    3231       13048 :     if (c == e)
    3232             :     {
    3233        3941 :       if (e == 2)
    3234             :       { /* sc: -4 */
    3235        1946 :         gel(V,1) = mkvecsmall(3);
    3236        1946 :         M[1] = 4;
    3237             :       }
    3238        1995 :       else if (e == 3)
    3239             :       { /* sc: -8 (CHI_2(-1)=-1<=>chi[1]=1) and 8 (CHI_2(-1)=1 <=> chi[1]=0) */
    3240        1995 :         long t = signe(gel(chi,1))? 7: 3;
    3241        1995 :         gel(V,1) = mkvecsmall2(5, t);
    3242        1995 :         M[1] = 8;
    3243             :       }
    3244             :     }
    3245        9107 :     else if (e == 5 && c == 3)
    3246         154 :     { /* sc: -8 (CHI_2(-1)=-1<=>chi[1]=1) and 8 (CHI_2(-1)=1 <=> chi[1]=0) */
    3247         154 :       long t = signe(gel(chi,1))? 7: 3;
    3248         154 :       gel(V,1) = mkvecsmalln(6, 2L,4L,5L,6L,8L,t);
    3249         154 :       M[1] = 8;
    3250             :     }
    3251        8953 :     else if ((e == 4 && c == 2) || (e == 5 && c <= 2) || (e == 6 && c <= 2)
    3252        7378 :          || (e >= 7 && c == e - 3))
    3253             :     { /* sc: 4 */
    3254        1575 :       gel(V,1) = mkvecsmall3(0,2,3);
    3255        1575 :       M[1] = 4;
    3256             :     }
    3257        7378 :     else if ((e <= 4 && c == 0) || (e >= 5 && c == e - 2))
    3258             :     { /* sc: 2 */
    3259        7021 :       gel(V,1) = mkvecsmall(0);
    3260        7021 :       M[1] = 2;
    3261             :     }
    3262         357 :     else if ((e == 6 && c == 3) || (e >= 7 && c <= e - 4))
    3263             :     { /* sc: -2 */
    3264         357 :       gel(V,1) = mkvecsmalln(7, 0L,2L,3L,4L,5L,6L,7L);
    3265         357 :       M[1] = 8;
    3266             :     }
    3267             :   }
    3268       33544 :   j = M[1]? 2: 1;
    3269       71526 :   for (i = odd(N)? 1: 2; i < l; i++) /* skip p=2, done above */
    3270             :   {
    3271       37982 :     long p = PN[i], e = EN[i];
    3272       37982 :     long z = zv_search(P, p), c = z? E[z]: 0; /* c = v_p(FC) */
    3273       37982 :     if ((e <= 2 && c == 1 && itos(gel(chi,z)) == (p>>1)) /* ord(CHI_p)=2 */
    3274       35791 :         || (e >= 3 && c <= e - 2))
    3275        2814 :     { /* sc: -p */
    3276        2814 :       GEN v = non_residues(p);
    3277        2814 :       if (e != 1) v = vecsmall_prepend(v, 0);
    3278        2814 :       gel(V,j) = v;
    3279        2814 :       M[j] = p; j++;
    3280             :     }
    3281       35168 :     else if (e >= 2 && c < e)
    3282             :     { /* sc: p */
    3283        2660 :       gel(V,j) = mkvecsmall(0);
    3284        2660 :       M[j] = p; j++;
    3285             :     }
    3286             :   }
    3287       33544 :   if (j == 1) return cgetg(1, t_VECSMALL);
    3288       15603 :   setlg(V,j); setlg(M,j); mod = zv_prod(M);
    3289       15603 :   L = zero_zv(mod);
    3290       34125 :   for (i = 1; i < j; i++)
    3291             :   {
    3292       18522 :     GEN v = gel(V,i);
    3293       18522 :     long s, m = M[i], lv = lg(v);
    3294       48132 :     for (s = 1; s < lv; s++)
    3295             :     {
    3296       29610 :       long a = v[s] + 1;
    3297       56679 :       do { L[a] = 1; a += m; } while (a <= mod);
    3298             :     }
    3299             :   }
    3300       15603 :   return L;
    3301             : }
    3302             : /* v=mfnewzerodata(N,CHI); returns TRUE if newtrace(n) must be zero,
    3303             :  * (but newtrace(n) may still be zero if we return FALSE) */
    3304             : static long
    3305     2673683 : mfnewchkzero(GEN v, long n) { long q = lg(v)-1; return q && v[(n%q) + 1]; }
    3306             : 
    3307             : /* if (!VCHIP): from mftraceform_cusp;
    3308             :  * else from initnewtrace and CHI is known to be primitive */
    3309             : static GEN
    3310      129171 : inittrace(long N, GEN CHI, GEN VCHIP)
    3311             : {
    3312             :   long FC;
    3313      129171 :   if (VCHIP)
    3314      129164 :     FC = mfcharmodulus(CHI);
    3315             :   else
    3316           7 :     VCHIP = mfcharinit(mfchartoprimitive(CHI, &FC));
    3317      129171 :   return mkvecn(5, mksqr(N), mkmup(N), mkgcd(N), VCHIP, mkbez(N, FC));
    3318             : }
    3319             : 
    3320             : /* p > 2 prime; return a sorted t_VECSMALL of primes s.t Tr^new(p) = 0 for all
    3321             :  * weights > 2 */
    3322             : static GEN
    3323       33544 : inittrconj(long N, long FC)
    3324             : {
    3325             :   GEN fa, P, E, v;
    3326             :   long i, k, l;
    3327             : 
    3328       33544 :   if (FC != 1) return cgetg(1,t_VECSMALL);
    3329             : 
    3330       27713 :   fa = myfactoru(N >> vals(N));
    3331       27713 :   P = gel(fa,1); l = lg(P);
    3332       27713 :   E = gel(fa,2);
    3333       27713 :   v = cgetg(l, t_VECSMALL);
    3334       60298 :   for (i = k = 1; i < l; i++)
    3335             :   {
    3336       32585 :     long j, p = P[i]; /* > 2 */
    3337       78610 :     for (j = 1; j < l; j++)
    3338       46025 :       if (j != i && E[j] == 1 && kross(-p, P[j]) == 1) v[k++] = p;
    3339             :   }
    3340       27713 :   setlg(v,k); return v;
    3341             : }
    3342             : 
    3343             : /* assume CHIP primitive, f(CHIP) | N; NZ = mfnewzerodata(N,CHIP) */
    3344             : static GEN
    3345       33544 : initnewtrace_i(long N, GEN CHIP, GEN NZ)
    3346             : {
    3347       33544 :   GEN T = const_vec(N, cgetg(1,t_VEC)), D, VCHIP;
    3348       33544 :   long FC = mfcharmodulus(CHIP), N1, N2, i, l;
    3349             : 
    3350       33544 :   if (!NZ) NZ = mkvecsmall(1); /*Tr^new = 0; initialize data nevertheless*/
    3351       33544 :   VCHIP = mfcharinit(CHIP);
    3352       33544 :   N1 = N/FC; newd_params(N1, &N2);
    3353       33544 :   D = mydivisorsu(N1/N2); l = lg(D);
    3354       33544 :   N2 *= FC;
    3355      162708 :   for (i = 1; i < l; i++)
    3356             :   {
    3357      129164 :     long M = D[i]*N2;
    3358      129164 :     gel(T,M) = inittrace(M, CHIP, VCHIP);
    3359             :   }
    3360       33544 :   gel(T,N) = shallowconcat(gel(T,N), mkvec2(NZ, inittrconj(N,FC)));
    3361       33544 :   return T;
    3362             : }
    3363             : /* don't initialize if Tr^new = 0, return NULL */
    3364             : static GEN
    3365       33649 : initnewtrace(long N, GEN CHI)
    3366             : {
    3367       33649 :   GEN CHIP = mfchartoprimitive(CHI, NULL), NZ = mfnewzerodata(N,CHIP);
    3368       33649 :   return NZ? initnewtrace_i(N, CHIP, NZ): NULL;
    3369             : }
    3370             : 
    3371             : /* (-1)^k */
    3372             : static long
    3373        8295 : m1pk(long k) { return odd(k)? -1 : 1; }
    3374             : static long
    3375        7931 : badchar(long N, long k, GEN CHI)
    3376        7931 : { return mfcharparity(CHI) != m1pk(k) || (CHI && N % mfcharconductor(CHI)); }
    3377             : 
    3378             : 
    3379             : static long
    3380       43743 : mfcuspdim_i(long N, long k, GEN CHI, GEN vSP)
    3381             : {
    3382       43743 :   pari_sp av = avma;
    3383             :   long FC;
    3384             :   GEN s;
    3385       43743 :   if (k <= 0) return 0;
    3386       43743 :   if (k == 1) return CHI? mf1cuspdim(N, CHI, vSP): 0;
    3387       43484 :   FC = CHI? mfcharconductor(CHI): 1;
    3388       43484 :   if (FC == 1) CHI = NULL;
    3389       43484 :   s = gsub(A1(N, k), gadd(A21(N, k, CHI), A22(N, k, CHI)));
    3390       43484 :   s = gadd(s, gsubsg(A4(k, FC), A3(N, FC)));
    3391       43484 :   return gc_long(av, itos(s));
    3392             : }
    3393             : /* dimension of space of cusp forms S_k(\G_0(N),CHI)
    3394             :  * Only depends on CHIP the primitive char attached to CHI */
    3395             : long
    3396        3458 : mfcuspdim(long N, long k, GEN CHI) { return mfcuspdim_i(N, k, CHI, NULL); }
    3397             : 
    3398             : /* dimension of whole space M_k(\G_0(N),CHI)
    3399             :  * Only depends on CHIP the primitive char attached to CHI; assumes !badchar */
    3400             : long
    3401         868 : mffulldim(long N, long k, GEN CHI)
    3402             : {
    3403         868 :   pari_sp av = avma;
    3404         868 :   long FC = CHI? mfcharconductor(CHI): 1;
    3405             :   GEN s;
    3406         868 :   if (k <= 0) return (k == 0 && FC == 1)? 1: 0;
    3407         868 :   if (k == 1) return gc_long(av, itos(A3(N, FC)) + mf1cuspdim(N, CHI, NULL));
    3408         651 :   if (FC == 1) CHI = NULL;
    3409         651 :   s = gsub(A1(N, k), gadd(A21(N, k, CHI), A22(N, k, CHI)));
    3410         651 :   s = gadd(s, A3(N, FC));
    3411         651 :   return gc_long(av, itos(s));
    3412             : }
    3413             : 
    3414             : /* Dimension of the space of Eisenstein series */
    3415             : long
    3416         231 : mfeisensteindim(long N, long k, GEN CHI)
    3417             : {
    3418         231 :   pari_sp av = avma;
    3419         231 :   long s, FC = CHI? mfcharconductor(CHI): 1;
    3420         231 :   if (k <= 0) return (k == 0 && FC == 1)? 1: 0;
    3421         231 :   s = itos(gmul2n(A3(N, FC), 1));
    3422         231 :   if (k > 1) s -= A4(k, FC); else s >>= 1;
    3423         231 :   return gc_long(av,s);
    3424             : }
    3425             : 
    3426             : enum { _SQRTS = 1, _MUP, _GCD, _VCHIP, _BEZ, _NEWLZ, _TRCONJ };
    3427             : /* Trace of T(n) on space of cuspforms; only depends on CHIP the primitive char
    3428             :  * attached to CHI */
    3429             : static GEN
    3430     6564045 : mfcusptrace_i(long N, long k, long n, GEN Dn, GEN S)
    3431             : {
    3432     6564045 :   pari_sp av = avma;
    3433             :   GEN a, b, VCHIP, GCD;
    3434             :   long t;
    3435     6564045 :   if (!n) return gen_0;
    3436     6564045 :   VCHIP = gel(S,_VCHIP);
    3437     6564045 :   GCD = gel(S,_GCD);
    3438     6564045 :   t = TA4(k, VCHIP, Dn, GCD);
    3439     6564045 :   a = TA1(N, k, VCHIP, GCD, n); if (t) a = gaddgs(a,t);
    3440     6564045 :   b = TA2(N, k, VCHIP, n, gel(S,_SQRTS), gel(S,_MUP), GCD);
    3441     6564045 :   b = gadd(b, TA3(N, k, VCHIP, GCD, Dn, gel(S,_BEZ)));
    3442     6564045 :   b = gsub(a,b);
    3443     6564045 :   if (typ(b) != t_POL) return gc_upto(av, b);
    3444       50834 :   return gc_GEN(av, vchip_polmod(VCHIP, b));
    3445             : }
    3446             : 
    3447             : static GEN
    3448     7850904 : mfcusptracecache(long N, long k, long n, GEN Dn, GEN S, cachenew_t *cache)
    3449             : {
    3450     7850904 :   GEN C = NULL, T = gel(cache->vfull,N);
    3451     7850904 :   long lcache = lg(T);
    3452     7850904 :   if (n < lcache) C = gel(T, n);
    3453     7850904 :   if (C) cache->cuspHIT++; else C = mfcusptrace_i(N, k, n, Dn, S);
    3454     7850904 :   cache->cuspTOTAL++;
    3455     7850904 :   if (n < lcache) gel(T,n) = C;
    3456     7850904 :   return C;
    3457             : }
    3458             : 
    3459             : /* return the divisors of n, known to be among the elements of D */
    3460             : static GEN
    3461      336280 : div_restrict(GEN D, ulong n)
    3462             : {
    3463             :   long i, j, l;
    3464      336280 :   GEN v, VDIV = caches[cache_DIV].cache;
    3465      336280 :   if (lg(VDIV) > n) return gel(VDIV,n);
    3466           0 :   l = lg(D);
    3467           0 :   v = cgetg(l, t_VECSMALL);
    3468           0 :   for (i = j = 1; i < l; i++)
    3469             :   {
    3470           0 :     ulong d = D[i];
    3471           0 :     if (n % d == 0) v[j++] = d;
    3472             :   }
    3473           0 :   setlg(v,j); return v;
    3474             : }
    3475             : 
    3476             : /* for some prime divisors of N, Tr^new(p) = 0 */
    3477             : static int
    3478      270911 : trconj(GEN T, long N, long n)
    3479      270911 : { return (lg(T) > 1 && N % n == 0 && zv_search(T, n)); }
    3480             : 
    3481             : /* n > 0; trace formula on new space */
    3482             : static GEN
    3483     2673683 : mfnewtrace_i(long N, long k, long n, cachenew_t *cache)
    3484             : {
    3485     2673683 :   GEN VCHIP, s, Dn, DN1, SN, S = cache->DATA;
    3486             :   long FC, N1, N2, N1N2, g, i, j, lDN1;
    3487             : 
    3488     2673683 :   if (!S) return gen_0;
    3489     2673683 :   SN = gel(S,N);
    3490     2673683 :   if (mfnewchkzero(gel(SN,_NEWLZ), n)) return gen_0;
    3491     1941643 :   if (k > 2 && trconj(gel(SN,_TRCONJ), N, n)) return gen_0;
    3492     1941594 :   VCHIP = gel(SN, _VCHIP); FC = vchip_FC(VCHIP);
    3493     1941594 :   N1 = N/FC; newt_params(N1, n, FC, &g, &N2);
    3494     1941594 :   N1N2 = N1/N2;
    3495     1941594 :   DN1 = mydivisorsu(N1N2); lDN1 = lg(DN1);
    3496     1941594 :   N2 *= FC;
    3497     1941594 :   Dn = mydivisorsu(n); /* this one is probably out of cache */
    3498     1941594 :   s = gmulsg(mubeta2(N1N2,n), mfcusptracecache(N2, k, n, Dn, gel(S,N2), cache));
    3499     7514624 :   for (i = 2; i < lDN1; i++)
    3500             :   { /* skip M1 = 1, done above */
    3501     5573030 :     long M1 = DN1[i], N1M1 = DN1[lDN1-i];
    3502     5573030 :     GEN Dg = mydivisorsu(ugcd(M1, g));
    3503     5573030 :     M1 *= N2;
    3504     5573030 :     s = gadd(s, gmulsg(mubeta2(N1M1,n),
    3505     5573030 :                        mfcusptracecache(M1, k, n, Dn, gel(S,M1), cache)));
    3506     5909310 :     for (j = 2; j < lg(Dg); j++) /* skip d = 1, done above */
    3507             :     {
    3508      336280 :       long d = Dg[j], ndd = n/(d*d), M = M1/d;
    3509      336280 :       GEN z = mulsi(mubeta2(N1M1,ndd), powuu(d,k-1)), C = vchip_lift(VCHIP,d,z);
    3510      336280 :       GEN Dndd = div_restrict(Dn, ndd);
    3511      336280 :       s = gadd(s, gmul(C, mfcusptracecache(M, k, ndd, Dndd, gel(S,M), cache)));
    3512             :     }
    3513     5573030 :     s = vchip_mod(VCHIP, s);
    3514             :   }
    3515     1941594 :   return vchip_polmod(VCHIP, s);
    3516             : }
    3517             : 
    3518             : static GEN
    3519       12355 : get_DIH(long N)
    3520             : {
    3521       12355 :   GEN x = cache_get(cache_DIH, N);
    3522       12355 :   return x? gcopy(x): mfdihedral(N);
    3523             : }
    3524             : static GEN
    3525        2373 : get_vDIH(long N, GEN D)
    3526             : {
    3527        2373 :   GEN x = const_vec(N, NULL);
    3528             :   long i, l;
    3529        2373 :   if (!D) D = mydivisorsu(N);
    3530        2373 :   l = lg(D);
    3531       14504 :   for (i = 1; i < l; i++) { long d = D[i]; gel(x, d) = get_DIH(d); }
    3532        2373 :   return x;
    3533             : }
    3534             : 
    3535             : /* divisors of N which are multiple of F */
    3536             : static GEN
    3537         322 : divisorsNF(long N, long F)
    3538             : {
    3539         322 :   GEN D = mydivisorsu(N / F);
    3540         322 :   long l = lg(D), i;
    3541         833 :   for (i = 1; i < l; i++) D[i] = N / D[i];
    3542         322 :   return D;
    3543             : }
    3544             : /* mfcuspdim(N,k,CHI) - mfnewdim(N,k,CHI); CHIP primitive (for efficiency) */
    3545             : static long
    3546        8512 : mfolddim_i(long N, long k, GEN CHIP, GEN vSP)
    3547             : {
    3548        8512 :   long S, i, l, F = mfcharmodulus(CHIP), N1 = N / F, N2;
    3549             :   GEN D;
    3550        8512 :   newd_params(N1, &N2); /* will ensure mubeta != 0 */
    3551        8512 :   D = mydivisorsu(N1/N2); l = lg(D); S = 0;
    3552        8512 :   if (k == 1 && !vSP) vSP = get_vDIH(N, divisorsNF(N, F));
    3553       32795 :   for (i = 2; i < l; i++)
    3554             :   {
    3555       24283 :     long d = mfcuspdim_i(N / D[i], k, CHIP, vSP);
    3556       24283 :     if (d) S -= mubeta(D[i]) * d;
    3557             :   }
    3558        8512 :   return S;
    3559             : }
    3560             : long
    3561         224 : mfolddim(long N, long k, GEN CHI)
    3562             : {
    3563         224 :   pari_sp av = avma;
    3564         224 :   GEN CHIP = mfchartoprimitive(CHI, NULL);
    3565         224 :   return gc_long(av, mfolddim_i(N, k, CHIP, NULL));
    3566             : }
    3567             : /* Only depends on CHIP the primitive char attached to CHI; assumes !badchar */
    3568             : long
    3569       16002 : mfnewdim(long N, long k, GEN CHI)
    3570             : {
    3571             :   pari_sp av;
    3572             :   long S, F;
    3573       16002 :   GEN vSP, CHIP = mfchartoprimitive(CHI, &F);
    3574       16002 :   vSP = (k == 1)? get_vDIH(N, divisorsNF(N, F)): NULL;
    3575       16002 :   S = mfcuspdim_i(N, k, CHIP, vSP); if (!S) return 0;
    3576        8015 :   av = avma; return gc_long(av, S - mfolddim_i(N, k, CHIP, vSP));
    3577             : }
    3578             : 
    3579             : /* trace form, given as closure */
    3580             : static GEN
    3581         980 : mftraceform_new(long N, long k, GEN CHI)
    3582             : {
    3583             :   GEN T;
    3584         980 :   if (k == 1) return initwt1newtrace(mfinit_Nkchi(N, 1, CHI, mf_CUSP, 0));
    3585         959 :   T = initnewtrace(N,CHI); if (!T) return mftrivial();
    3586         959 :   return tag(t_MF_NEWTRACE, mkNK(N,k,CHI), T);
    3587             : }
    3588             : static GEN
    3589          14 : mftraceform_cusp(long N, long k, GEN CHI)
    3590             : {
    3591          14 :   if (k == 1) return initwt1trace(mfinit_Nkchi(N, 1, CHI, mf_CUSP, 0));
    3592           7 :   return tag(t_MF_TRACE, mkNK(N,k,CHI), inittrace(N,CHI,NULL));
    3593             : }
    3594             : static GEN
    3595          98 : mftraceform_i(GEN NK, long space)
    3596             : {
    3597             :   GEN CHI;
    3598             :   long N, k;
    3599          98 :   checkNK(NK, &N, &k, &CHI, 0);
    3600          98 :   if (!mfdim_Nkchi(N, k, CHI, space)) return mftrivial();
    3601          77 :   switch(space)
    3602             :   {
    3603          56 :     case mf_NEW: return mftraceform_new(N, k, CHI);
    3604          14 :     case mf_CUSP:return mftraceform_cusp(N, k, CHI);
    3605             :   }
    3606           7 :   pari_err_DOMAIN("mftraceform", "space", "=", utoi(space), NK);
    3607             :   return NULL;/*LCOV_EXCL_LINE*/
    3608             : }
    3609             : GEN
    3610          98 : mftraceform(GEN NK, long space)
    3611          98 : { pari_sp av = avma; return gc_GEN(av, mftraceform_i(NK,space)); }
    3612             : 
    3613             : static GEN
    3614       18466 : hecke_data(long N, long n)
    3615       18466 : { return mkvecsmall3(n, u_ppo(n, N), N); }
    3616             : /* 1/2-integral weight */
    3617             : static GEN
    3618          84 : heckef2_data(long N, long n)
    3619             : {
    3620             :   ulong f, fN, fN2;
    3621          84 :   if (!uissquareall(n, &f)) return NULL;
    3622          77 :   fN = u_ppo(f, N); fN2 = fN*fN;
    3623          77 :   return mkvec2(myfactoru(fN), mkvecsmall4(n, N, fN2, n/fN2));
    3624             : }
    3625             : /* N = mf_get_N(F) or a multiple */
    3626             : static GEN
    3627       25704 : mfhecke_i(long n, long N, GEN F)
    3628             : {
    3629       25704 :   if (n == 1) return F;
    3630       18081 :   return tag2(t_MF_HECKE, mf_get_NK(F), hecke_data(N,n), F);
    3631             : }
    3632             : 
    3633             : GEN
    3634         119 : mfhecke(GEN mf, GEN F, long n)
    3635             : {
    3636         119 :   pari_sp av = avma;
    3637             :   GEN NK, CHI, gk, DATA;
    3638             :   long N, nk, dk;
    3639         119 :   mf = checkMF(mf);
    3640         119 :   if (!checkmf_i(F)) pari_err_TYPE("mfhecke",F);
    3641         119 :   if (n <= 0) pari_err_TYPE("mfhecke [n <= 0]", stoi(n));
    3642         119 :   if (n == 1) return gcopy(F);
    3643         119 :   gk = mf_get_gk(F);
    3644         119 :   Qtoss(gk,&nk,&dk);
    3645         119 :   CHI = mf_get_CHI(F);
    3646         119 :   N = MF_get_N(mf);
    3647         119 :   if (dk == 2)
    3648             :   {
    3649          77 :     DATA = heckef2_data(N,n);
    3650          77 :     if (!DATA) return mftrivial();
    3651             :   }
    3652             :   else
    3653          42 :     DATA = hecke_data(N,n);
    3654         112 :   NK = mkgNK(lcmii(stoi(N), mf_get_gN(F)), gk, CHI, mf_get_field(F));
    3655         112 :   return gc_GEN(av, tag2(t_MF_HECKE, NK, DATA, F));
    3656             : }
    3657             : 
    3658             : /* form F given by closure, compute B(d)(F) as closure (q -> q^d) */
    3659             : static GEN
    3660       36981 : mfbd_i(GEN F, long d)
    3661             : {
    3662             :   GEN D, NK, gk, CHI;
    3663       36981 :   if (d == 1) return F;
    3664       13762 :   if (d <= 0) pari_err_TYPE("mfbd [d <= 0]", stoi(d));
    3665       13762 :   if (mf_get_type(F) != t_MF_BD) D = utoi(d);
    3666           7 :   else { D = mului(d, gel(F,3)); F = gel(F,2); }
    3667       13762 :   gk = mf_get_gk(F); CHI = mf_get_CHI(F);
    3668       13762 :   if (typ(gk) != t_INT) CHI = mfcharmul(CHI, get_mfchar(utoi(d << 2)));
    3669       13762 :   NK = mkgNK(muliu(mf_get_gN(F), d), gk, CHI, mf_get_field(F));
    3670       13762 :   return tag2(t_MF_BD, NK, F, D);
    3671             : }
    3672             : GEN
    3673         266 : mfbd(GEN F, long d)
    3674             : {
    3675         266 :   pari_sp av = avma;
    3676         266 :   if (!checkmf_i(F)) pari_err_TYPE("mfbd",F);
    3677         266 :   return gc_GEN(av, mfbd_i(F, d));
    3678             : }
    3679             : 
    3680             : /* A[i+1] = a(t*i^2) */
    3681             : static GEN
    3682         112 : RgV_shimura(GEN A, long n, long t, long N, long r, GEN CHI)
    3683             : {
    3684         112 :   GEN R, a0, Pn = mfcharpol(CHI);
    3685         112 :   long m, st, ord = mfcharorder(CHI), vt = varn(Pn), Nt = t == 1? N: ulcm(N,t);
    3686             : 
    3687         112 :   R = cgetg(n + 2, t_VEC);
    3688         112 :   st = odd(r)? -t: t;
    3689         112 :   a0 = gel(A, 1);
    3690         112 :   if (!gequal0(a0))
    3691             :   {
    3692          14 :     long o = mfcharorder(CHI);
    3693          14 :     if (st != 1 && odd(o)) o <<= 1;
    3694          14 :     a0 = gmul(a0, charLFwtk(Nt, r, CHI, o, st));
    3695             :   }
    3696         112 :   gel(R, 1) = a0;
    3697         672 :   for (m = 1; m <= n; m++)
    3698             :   {
    3699         560 :     GEN Dm = mydivisorsu(u_ppo(m, Nt)), S = gel(A, m*m + 1);
    3700         560 :     long i, l = lg(Dm);
    3701         833 :     for (i = 2; i < l; i++)
    3702             :     { /* (e,Nt) = 1; skip i = 1: e = 1, done above */
    3703         273 :       long e = Dm[i], me = m / e, a = mfcharevalord(CHI, e, ord);
    3704         273 :       GEN c, C = powuu(e, r - 1);
    3705         273 :       if (kross(st, e) == -1) C = negi(C);
    3706         273 :       c = Qab_Czeta(a, ord, C, vt);
    3707         273 :       S = gadd(S, gmul(c, gel(A, me*me + 1)));
    3708             :     }
    3709         560 :     gel(R, m+1) = S;
    3710             :   }
    3711         112 :   return degpol(Pn) > 1? gmodulo(R, Pn): R;
    3712             : }
    3713             : 
    3714             : static long
    3715          35 : mfisinkohnen(GEN mf, GEN F)
    3716             : {
    3717          35 :   GEN v, gk = MF_get_gk(mf), CHI = MF_get_CHI(mf);
    3718          35 :   long i, eps, N4 = MF_get_N(mf) >> 2, sb = mfsturmNgk(N4 << 4, gk) + 1;
    3719          35 :   eps = N4 % mfcharconductor(CHI)? -1 : 1;
    3720          35 :   if (odd(MF_get_r(mf))) eps = -eps;
    3721          35 :   v = mfcoefs(F, sb, 1);
    3722         910 :   for (i = 2;     i <= sb; i+=4) if (!gequal0(gel(v,i+1))) return 0;
    3723         462 :   for (i = 2+eps; i <= sb; i+=4) if (!gequal0(gel(v,i+1))) return 0;
    3724          21 :   return 1;
    3725             : }
    3726             : 
    3727             : static long
    3728          49 : mfshimura_space_cusp(GEN mf)
    3729             : {
    3730             :   long N4;
    3731          49 :   if (MF_get_r(mf) == 1 && (N4 = MF_get_N(mf) >> 2) >= 4)
    3732             :   {
    3733          21 :     GEN E = gel(myfactoru(N4), 2);
    3734          21 :     long ma = vecsmall_max(E);
    3735          21 :     if (ma > 2 || (ma == 2 && !mfcharistrivial(MF_get_CHI(mf)))) return 0;
    3736             :   }
    3737          35 :   return 1;
    3738             : }
    3739             : 
    3740             : static GEN
    3741          56 : mfshifin(GEN mf2, GEN G)
    3742             : {
    3743          56 :   GEN res = mftobasis_i(mf2, G);
    3744             :   /* not mflinear(mf2,): we want lowest possible level */
    3745          56 :   G = mflinear(MF_get_basis(mf2), res);
    3746          56 :   return mkvec3(mf2, G, res);
    3747             : }
    3748             : 
    3749             : /* t is either a positive squarefree integer or a fundamental
    3750             :    discriminant of sign (-1)^r. */
    3751             : GEN
    3752          63 : mfshimura(GEN mf, GEN F, long t)
    3753             : {
    3754          63 :   pari_sp av = avma;
    3755             :   GEN G, mf2, CHI;
    3756          63 :   long sb, M, r, N, space = mf_FULL;
    3757             : 
    3758          63 :   if (!checkmf_i(F)) pari_err_TYPE("mfshimura",F);
    3759          63 :   mf = checkMF(mf);
    3760          63 :   r = MF_get_r(mf);
    3761          63 :   if (r <= 0) pari_err_DOMAIN("mfshimura", "weight", "<=", ghalf, mf_get_gk(F));
    3762          63 :   if (t <= 0 || !uissquarefree(t))
    3763             :   {
    3764          14 :     GEN gD = stoi(t);
    3765          14 :     if (!t || !isfundamental(gD) || (t < 0 && !odd(r)) || (t > 0 && odd(r)))
    3766           7 :       pari_err_TYPE("mfshimura [t]", stoi(t));
    3767           7 :     if (odd(t)) t = -t;
    3768             :     else
    3769             :     {
    3770             :       GEN SH;
    3771           7 :       if (t < 0) t = -t;
    3772           7 :       SH = mfshimura(mf, F, t >> 2); mf2 = gel(SH, 1);
    3773           7 :       G = mfhecke(mf2, gel(SH, 2), 2);
    3774           7 :       return gc_GEN(av, mfshifin(mf2, G));
    3775             :     }
    3776             :   }
    3777          49 :   N = MF_get_N(mf); M = N >> 1;
    3778          49 :   if (mfiscuspidal(mf,F))
    3779             :   {
    3780          35 :     if (mfshimura_space_cusp(mf)) space = mf_CUSP;
    3781          35 :     if (mfisinkohnen(mf,F)) M = N >> 2;
    3782             :   }
    3783          49 :   CHI = MF_get_CHI(mf);
    3784          49 :   mf2 = mfinit_Nkchi(M, r << 1, mfcharpow(CHI, gen_2), space, 0);
    3785          49 :   sb = mfsturm(mf2);
    3786          49 :   G = RgV_shimura(mfcoefs_i(F, sb*sb, t), sb, t, N, r, CHI);
    3787          49 :   return gc_GEN(av, mfshifin(mf2, G));
    3788             : }
    3789             : 
    3790             : /* W ZabM (ZM if n = 1), a t_INT or NULL, b t_INT, ZXQ mod P or NULL.
    3791             :  * Write a/b = A/d with d t_INT and A Zab return [W,d,A,P] */
    3792             : static GEN
    3793        7819 : mkMinv(GEN W, GEN a, GEN b, GEN P)
    3794             : {
    3795        7819 :   GEN A = (b && typ(b) == t_POL)? Q_remove_denom(QXQ_inv(b,P), &b): NULL;
    3796        7819 :   if (a && b)
    3797             :   {
    3798        1358 :     a = Qdivii(a,b);
    3799        1358 :     if (typ(a) == t_INT) b = gen_1; else { b = gel(a,2); a = gel(a,1); }
    3800        1358 :     if (is_pm1(a)) a = NULL;
    3801             :   }
    3802        7819 :   if (a) A = A? ZX_Z_mul(A,a): a; else if (!A) A = gen_1;
    3803        7819 :   if (!b) b = gen_1;
    3804        7819 :   if (!P) P = gen_0;
    3805        7819 :   return mkvec4(W,b,A,P);
    3806             : }
    3807             : /* M square invertible QabM, return [M',d], M*M' = d*Id */
    3808             : static GEN
    3809         609 : QabM_Minv(GEN M, GEN P, long n)
    3810             : {
    3811             :   GEN dW, W, dM;
    3812         609 :   M = Q_remove_denom(M, &dM);
    3813         609 :   W = P? ZabM_inv(liftpol_shallow(M), P, n, &dW): ZM_inv(M, &dW);
    3814         609 :   return mkMinv(W, dM, dW, P);
    3815             : }
    3816             : /* Simplified form of mfclean, after a QabM_indexrank: M a ZabM with full
    3817             :  * column rank and z = indexrank(M) is known */
    3818             : static GEN
    3819         861 : mfclean2(GEN M, GEN z, GEN P, long n)
    3820             : {
    3821         861 :   GEN d, Minv, y = gel(z,1), W = rowpermute(M, y);
    3822         861 :   W = P? ZabM_inv(liftpol_shallow(W), P, n, &d): ZM_inv(W, &d);
    3823         861 :   M = rowslice(M, 1, y[lg(y)-1]);
    3824         861 :   Minv = mkMinv(W, NULL, d, P);
    3825         861 :   return mkvec3(y, Minv, M);
    3826             : }
    3827             : /* M QabM, lg(M)>1 and [y,z] its rank profile. Let Minv be the inverse of the
    3828             :  * invertible square matrix in mkMinv format. Return [y,Minv, M[..y[#y],]]
    3829             :  * P cyclotomic polynomial of order n > 2 or NULL */
    3830             : static GEN
    3831        5047 : mfclean(GEN M, GEN P, long n, int ratlift)
    3832             : {
    3833        5047 :   GEN W, v, y, z, d, Minv, dM, MdM = Q_remove_denom(M, &dM);
    3834        5047 :   if (n <= 2)
    3835        3941 :     W = ZM_pseudoinv(MdM, &v, &d);
    3836             :   else
    3837        1106 :     W = ZabM_pseudoinv_i(liftpol_shallow(MdM), P, n, &v, &d, ratlift);
    3838        5047 :   y = gel(v,1);
    3839        5047 :   z = gel(v,2);
    3840        5047 :   if (lg(z) != lg(MdM)) M = vecpermute(M,z);
    3841        5047 :   M = rowslice(M, 1, y[lg(y)-1]);
    3842        5047 :   Minv = mkMinv(W, dM, d, P);
    3843        5047 :   return mkvec3(y, Minv, M);
    3844             : }
    3845             : /* call mfclean using only CHI */
    3846             : static GEN
    3847        4095 : mfcleanCHI(GEN M, GEN CHI, int ratlift)
    3848             : {
    3849        4095 :   long n = mfcharorder(CHI);
    3850        4095 :   GEN P = (n <= 2)? NULL: mfcharpol(CHI);
    3851        4095 :   return mfclean(M, P, n, ratlift);
    3852             : }
    3853             : 
    3854             : /* DATA component of a t_MF_NEWTRACE. Was it stripped to save memory ? */
    3855             : static int
    3856       34601 : newtrace_stripped(GEN DATA)
    3857       34601 : { return DATA && (lg(DATA) == 5 && typ(gel(DATA,3)) == t_INT); }
    3858             : /* f a t_MF_NEWTRACE */
    3859             : static GEN
    3860       34601 : newtrace_DATA(long N, GEN f)
    3861             : {
    3862       34601 :   GEN DATA = gel(f,2);
    3863       34601 :   return newtrace_stripped(DATA)? initnewtrace(N, DATA): DATA;
    3864             : }
    3865             : /* reset cachenew for new level incorporating new DATA, tf a t_MF_NEWTRACE
    3866             :  * (+ possibly initialize 'full' for new allowed levels) */
    3867             : static void
    3868       34601 : reset_cachenew(cachenew_t *cache, long N, GEN tf)
    3869             : {
    3870             :   long i, n, l;
    3871       34601 :   GEN v, DATA = newtrace_DATA(N,tf);
    3872       34601 :   cache->DATA = DATA;
    3873       34601 :   if (!DATA) return;
    3874       34496 :   n = cache->n;
    3875       34496 :   v = cache->vfull; l = N+1; /* = lg(DATA) */
    3876     2223592 :   for (i = 1; i < l; i++)
    3877     2189096 :     if (typ(gel(v,i)) == t_INT && lg(gel(DATA,i)) != 1)
    3878       54551 :       gel(v,i) = const_vec(n, NULL);
    3879       34496 :   cache->VCHIP = gel(gel(DATA,N),_VCHIP);
    3880             : }
    3881             : /* initialize a cache of newtrace / cusptrace up to index n and level | N;
    3882             :  * DATA may be NULL (<=> Tr^new = 0). tf a t_MF_NEWTRACE */
    3883             : static void
    3884       13650 : init_cachenew(cachenew_t *cache, long n, long N, GEN tf)
    3885             : {
    3886       13650 :   long i, l = N+1; /* = lg(tf.DATA) when DATA != NULL */
    3887             :   GEN v;
    3888       13650 :   cache->n = n;
    3889       13650 :   cache->vnew = v = cgetg(l, t_VEC);
    3890      957593 :   for (i = 1; i < l; i++) gel(v,i) = (N % i)? gen_0: const_vec(n, NULL);
    3891       13650 :   cache->newHIT = cache->newTOTAL = cache->cuspHIT = cache->cuspTOTAL = 0;
    3892       13650 :   cache->vfull = v = zerovec(N);
    3893       13650 :   reset_cachenew(cache, N, tf);
    3894       13650 : }
    3895             : static void
    3896       17780 : dbg_cachenew(cachenew_t *C)
    3897             : {
    3898       17780 :   if (DEBUGLEVEL >= 2 && C)
    3899           0 :     err_printf("newtrace cache hits: new = %ld/%ld, cusp = %ld/%ld\n",
    3900             :                     C->newHIT, C->newTOTAL, C->cuspHIT, C->cuspTOTAL);
    3901       17780 : }
    3902             : 
    3903             : /* newtrace_{N,k}(d*i), i = n0, ..., n */
    3904             : static GEN
    3905      185598 : colnewtrace(long n0, long n, long d, long N, long k, cachenew_t *cache)
    3906             : {
    3907      185598 :   GEN v = cgetg(n-n0+2, t_COL);
    3908             :   long i;
    3909     4831841 :   for (i = n0; i <= n; i++) gel(v, i-n0+1) = mfnewtracecache(N, k, i*d, cache);
    3910      185598 :   return v;
    3911             : }
    3912             : /* T_n(l*m0, l*(m0+1), ..., l*m) F, F = t_MF_NEWTRACE [N,k],DATA, cache
    3913             :  * contains DATA != NULL as well as cached values of F */
    3914             : static GEN
    3915       91581 : heckenewtrace(long m0, long m, long l, long N, long NBIG, long k, long n, cachenew_t *cache)
    3916             : {
    3917       91581 :   long lD, a, k1, nl = n*l;
    3918       91581 :   GEN D, V, v = colnewtrace(m0, m, nl, N, k, cache); /* d=1 */
    3919             :   GEN VCHIP;
    3920       91581 :   if (n == 1) return v;
    3921       63259 :   VCHIP = cache->VCHIP;
    3922       63259 :   D = mydivisorsu(u_ppo(n, NBIG)); lD = lg(D);
    3923       63259 :   k1 = k - 1;
    3924      155358 :   for (a = 2; a < lD; a++)
    3925             :   { /* d > 1, (d,NBIG) = 1 */
    3926       92099 :     long i, j, d = D[a], c = ugcd(l, d), dl = d/c, m0d = ceildivuu(m0, dl);
    3927       92099 :     GEN C = vchip_lift(VCHIP, d, powuu(d, k1));
    3928             :     /* m0=0: i = 1 => skip F(0) = 0 */
    3929       92099 :     if (!m0) { i = 1; j = dl; } else { i = 0; j = m0d*dl; }
    3930       92099 :     V = colnewtrace(m0d, m/dl, nl/(d*c), N, k, cache);
    3931             :     /* C = chi(d) d^(k-1) */
    3932     1105314 :     for (; j <= m; i++, j += dl)
    3933     1013215 :       gel(v,j-m0+1) = gadd(gel(v,j-m0+1), vchip_mod(VCHIP, gmul(C,gel(V,i+1))));
    3934             :   }
    3935       63259 :   return v;
    3936             : }
    3937             : 
    3938             : /* Given v = an[i], return an[d*i], i=0..n */
    3939             : static GEN
    3940        2646 : anextract(GEN v, long n, long d)
    3941             : {
    3942        2646 :   long i, id, l = n + 2;
    3943        2646 :   GEN w = cgetg(l, t_VEC);
    3944        2646 :   if (d == 1)
    3945        7322 :     for (i = 1; i < l; i++) gel(w, i) = gel(v, i);
    3946             :   else
    3947       22169 :     for (i = id = 1; i < l; i++, id += d) gel(w, i) = gel(v, id);
    3948        2646 :   return w;
    3949             : }
    3950             : /* T_n(F)(0, l, ..., l*m) */
    3951             : static GEN
    3952        2723 : hecke_i(long m, long l, GEN V, GEN F, GEN DATA)
    3953             : {
    3954             :   long k, n, nNBIG, NBIG, lD, M, a, t, nl;
    3955             :   GEN D, v, CHI;
    3956        2723 :   if (typ(DATA) == t_VEC)
    3957             :   { /* 1/2-integral k */
    3958          98 :     if (!V) { GEN S = gel(DATA,2); V = mfcoefs_i(F, m*l*S[3], S[4]); }
    3959          98 :     return RgV_heckef2(m, l, V, F, DATA);
    3960             :   }
    3961        2625 :   k = mf_get_k(F);
    3962        2625 :   n = DATA[1]; nl = n*l;
    3963        2625 :   nNBIG = DATA[2];
    3964        2625 :   NBIG = DATA[3];
    3965        2625 :   if (nNBIG == 1) return V? V: mfcoefs_i(F,m,nl);
    3966        1869 :   if (!V && mf_get_type(F) == t_MF_NEWTRACE)
    3967             :   { /* inline F to allow cache, T_n at level NBIG acting on Tr^new(N,k,CHI) */
    3968             :     cachenew_t cache;
    3969         546 :     long N = mf_get_N(F);
    3970         546 :     init_cachenew(&cache, m*nl, N, F);
    3971         546 :     v = heckenewtrace(0, m, l, N, NBIG, k, n, &cache);
    3972         546 :     dbg_cachenew(&cache);
    3973         546 :     settyp(v, t_VEC); return v;
    3974             :   }
    3975        1323 :   CHI = mf_get_CHI(F);
    3976        1323 :   D = mydivisorsu(nNBIG); lD = lg(D);
    3977        1323 :   M = m + 1;
    3978        1323 :   t = nNBIG * ugcd(nNBIG, l);
    3979        1323 :   if (!V) V = mfcoefs_i(F, m * t, nl / t); /* usually nl = t */
    3980        1323 :   v = anextract(V, m, t); /* mfcoefs(F, m, nl); d = 1 */
    3981        2646 :   for (a = 2; a < lD; a++)
    3982             :   { /* d > 1, (d, NBIG) = 1 */
    3983        1323 :     long d = D[a], c = ugcd(l, d), dl = d/c, i, idl;
    3984        1323 :     GEN chi = mfchareval(CHI, d);
    3985        1323 :     GEN C = k ? gmul(chi, powuu(d, k-1)): chi;
    3986        1323 :     GEN w = anextract(V, m/dl, t/(d*c)); /* mfcoefs(F, m/dl, nl/(d*c)) */
    3987        7322 :     for (i = idl = 1; idl <= M; i++, idl += dl)
    3988        5999 :       gel(v,idl) = gadd(gel(v,idl), gmul(C, gel(w,i)));
    3989             :   }
    3990        1323 :   return v;
    3991             : }
    3992             : 
    3993             : static GEN
    3994       12544 : mkmf(GEN x1, GEN x2, GEN x3, GEN x4, GEN x5)
    3995             : {
    3996       12544 :   GEN MF = obj_init(5, MF_SPLITN);
    3997       12544 :   gel(MF,1) = x1;
    3998       12544 :   gel(MF,2) = x2;
    3999       12544 :   gel(MF,3) = x3;
    4000       12544 :   gel(MF,4) = x4;
    4001       12544 :   gel(MF,5) = x5; return MF;
    4002             : }
    4003             : 
    4004             : /* return an integer b such that p | b => T_p^k Tr^new = 0, for all k > 0 */
    4005             : static long
    4006        7756 : get_badj(long N, long FC)
    4007             : {
    4008        7756 :   GEN fa = myfactoru(N), P = gel(fa,1), E = gel(fa,2);
    4009        7756 :   long i, b = 1, l = lg(P);
    4010       20587 :   for (i = 1; i < l; i++)
    4011       12831 :     if (E[i] > 1 && u_lval(FC, P[i]) < E[i]) b *= P[i];
    4012        7756 :   return b;
    4013             : }
    4014             : /* in place, assume perm strictly increasing */
    4015             : static void
    4016        1372 : vecpermute_inplace(GEN v, GEN perm)
    4017             : {
    4018        1372 :   long i, l = lg(perm);
    4019       11914 :   for (i = 1; i < l; i++) gel(v,i) = gel(v,perm[i]);
    4020        1372 : }
    4021             : 
    4022             : /* Find basis of newspace using closures; assume k >= 2 and !badchar.
    4023             :  * Return NULL if space is empty, else
    4024             :  * [mf1, list of closures T(j)traceform, list of corresponding j, matrix] */
    4025             : static GEN
    4026       15757 : mfnewinit(long N, long k, GEN CHI, cachenew_t *cache, long init)
    4027             : {
    4028             :   GEN S, vj, M, CHIP, mf1, listj, P, tf;
    4029             :   long j, ct, ctlj, dim, jin, SB, sb, two, ord, FC, badj;
    4030             : 
    4031       15757 :   dim = mfnewdim(N, k, CHI);
    4032       15757 :   if (!dim && !init) return NULL;
    4033        7756 :   sb = mfsturmNk(N, k);
    4034        7756 :   CHIP = mfchartoprimitive(CHI, &FC);
    4035             :   /* remove newtrace data from S to save space in output: negligible slowdown */
    4036        7756 :   tf = tag(t_MF_NEWTRACE, mkNK(N,k,CHIP), CHIP);
    4037        7756 :   badj = get_badj(N, FC);
    4038             :   /* try sbsmall first: Sturm bound not sharp for new space */
    4039        7756 :   SB = ceilA1(N, k);
    4040        7756 :   listj = cgetg(2*sb + 3, t_VECSMALL);
    4041      377461 :   for (j = ctlj = 1; ctlj < 2*sb + 3; j++)
    4042      369705 :     if (ugcd(j, badj) == 1) listj[ctlj++] = j;
    4043        7756 :   if (init)
    4044             :   {
    4045        4200 :     init_cachenew(cache, (SB+1)*listj[dim+1], N, tf);
    4046        4200 :     if (init == -1 || !dim) return NULL; /* old space or dim = 0 */
    4047             :   }
    4048             :   else
    4049        3556 :     reset_cachenew(cache, N, tf);
    4050             :   /* cache.DATA is not NULL */
    4051        7287 :   ord = mfcharorder(CHIP);
    4052        7287 :   P = ord <= 2? NULL: mfcharpol(CHIP);
    4053        7287 :   vj = cgetg(dim+1, t_VECSMALL);
    4054        7287 :   M = cgetg(dim+1, t_MAT);
    4055        7294 :   for (two = 1, ct = 0, jin = 1; two <= 2; two++)
    4056             :   {
    4057        7294 :     long a, jlim = jin + sb;
    4058       22876 :     for (a = jin; a <= jlim; a++)
    4059             :     {
    4060             :       GEN z, vecz;
    4061       22869 :       ct++; vj[ct] = listj[a];
    4062       22869 :       gel(M, ct) = heckenewtrace(0, SB, 1, N, N, k, vj[ct], cache);
    4063       22869 :       if (ct < dim) continue;
    4064             : 
    4065        7973 :       z = QabM_indexrank(M, P, ord);
    4066        7973 :       vecz = gel(z, 2); ct = lg(vecz) - 1;
    4067        7973 :       if (ct == dim) { M = mkvec3(z, gen_0, M); break; } /*maximal rank, done*/
    4068         686 :       vecpermute_inplace(M, vecz);
    4069         686 :       vecpermute_inplace(vj, vecz);
    4070             :     }
    4071        7294 :     if (a <= jlim) break;
    4072             :     /* sbsmall was not sufficient, use Sturm bound: must extend M */
    4073          70 :     for (j = 1; j <= ct; j++)
    4074             :     {
    4075          63 :       GEN t = heckenewtrace(SB + 1, sb, 1, N, N, k, vj[j], cache);
    4076          63 :       gel(M,j) = shallowconcat(gel(M, j), t);
    4077             :     }
    4078           7 :     jin = jlim + 1; SB = sb;
    4079             :   }
    4080        7287 :   S = cgetg(dim + 1, t_VEC);
    4081       29428 :   for (j = 1; j <= dim; j++) gel(S, j) = mfhecke_i(vj[j], N, tf);
    4082        7287 :   dbg_cachenew(cache);
    4083        7287 :   mf1 = mkvec4(utoipos(N), utoipos(k), CHI, utoi(mf_NEW));
    4084        7287 :   return mkmf(mf1, cgetg(1,t_VEC), S, vj, M);
    4085             : }
    4086             : /* k > 1 integral, mf space is mf_CUSP or mf_FULL */
    4087             : static GEN
    4088          49 : mfinittonew(GEN mf)
    4089             : {
    4090          49 :   GEN CHI = MF_get_CHI(mf), S = MF_get_S(mf), vMjd = MFcusp_get_vMjd(mf);
    4091          49 :   GEN M = MF_get_M(mf), vj, mf1;
    4092          49 :   long i, j, l, l0 = lg(S), N0 = MF_get_N(mf);
    4093         252 :   for (i = l0-1; i > 0; i--)
    4094             :   {
    4095         238 :     long N = gel(vMjd,i)[1];
    4096         238 :     if (N != N0) break;
    4097             :   }
    4098          49 :   if (i == l0-1) return NULL;
    4099          42 :   S = vecslice(S, i+1, l0-1); /* forms of conductor N0 */
    4100          42 :   l = lg(S); vj = cgetg(l, t_VECSMALL);
    4101         245 :   for (j = 1; j < l; j++) vj[j] = gel(vMjd,j+i)[2];
    4102          42 :   M = vecslice(M, lg(M)-lg(S)+1, lg(M)-1); /* their coefficients */
    4103          42 :   M = mfcleanCHI(M, CHI, 0);
    4104          42 :   mf1 = mkvec4(utoipos(N0), MF_get_gk(mf), CHI, utoi(mf_NEW));
    4105          42 :   return mkmf(mf1, cgetg(1,t_VEC), S, vj, M);
    4106             : }
    4107             : 
    4108             : /* Bd(f)[m0..m], v = f[ceil(m0/d)..floor(m/d)], m0d = ceil(m0/d) */
    4109             : static GEN
    4110       84665 : RgC_Bd_expand(long m0, long m, GEN v, long d, long m0d)
    4111             : {
    4112             :   long i, j;
    4113             :   GEN w;
    4114       84665 :   if (d == 1) return v;
    4115       24080 :   w = zerocol(m-m0+1);
    4116       24080 :   if (!m0) { i = 1; j = d; } else { i = 0; j = m0d*d; }
    4117      477421 :   for (; j <= m; i++, j += d) gel(w,j-m0+1) = gel(v,i+1);
    4118       24080 :   return w;
    4119             : }
    4120             : /* S a nonempty vector of t_MF_BD(t_MF_HECKE(t_MF_NEWTRACE)); M the matrix
    4121             :  * of their coefficients r*0, r*1, ..., r*m0 (~ mfvectomat) or NULL (empty),
    4122             :  * extend it to coeffs up to m > m0. The forms B_d(T_j(tf_N))in S should be
    4123             :  * sorted by level N, then j, then increasing d. No reordering here. */
    4124             : static GEN
    4125        9282 : bhnmat_extend(GEN M, long m, long r, GEN S, cachenew_t *cache)
    4126             : {
    4127        9282 :   long i, mr, m0, m0r, Nold = 0, jold = 0, l = lg(S);
    4128        9282 :   GEN MAT = cgetg(l, t_MAT), v = NULL;
    4129        9282 :   if (M) { m0 = nbrows(M); m0r = m0 * r; } else m0 = m0r = 0;
    4130        9282 :   mr = m*r;
    4131       93947 :   for (i = 1; i < l; i++)
    4132             :   {
    4133             :     long d, j, md, N;
    4134       84665 :     GEN c, f = bhn_parse(gel(S,i), &d,&j); /* t_MF_NEWTRACE */
    4135       84665 :     N = mf_get_N(f);
    4136       84665 :     md = ceildivuu(m0r,d);
    4137       84665 :     if (N != Nold) { reset_cachenew(cache, N, f); Nold = N; jold = 0; }
    4138       84665 :     if (!cache->DATA) { gel(MAT,i) = zerocol(m+1); continue; }
    4139       84665 :     if (j != jold || md)
    4140       68103 :     { v = heckenewtrace(md, mr/d, 1, N, N, mf_get_k(f), j,cache); jold=j; }
    4141       84665 :     c = RgC_Bd_expand(m0r, mr, v, d, md);
    4142       84665 :     if (r > 1) c = c_deflate(m-m0, r, c);
    4143       84665 :     if (M) c = shallowconcat(gel(M,i), c);
    4144       84665 :     gel(MAT,i) = c;
    4145             :   }
    4146        9282 :   return MAT;
    4147             : }
    4148             : 
    4149             : /* k > 1 */
    4150             : static GEN
    4151        3269 : mfinitcusp(long N, long k, GEN CHI, cachenew_t *cache, long space)
    4152             : {
    4153             :   long L, l, lDN1, FC, N1, d1, i, init;
    4154        3269 :   GEN vS, vMjd, DN1, vmf, CHIP = mfchartoprimitive(CHI, &FC);
    4155             : 
    4156        3269 :   d1 = (space == mf_OLD)? mfolddim_i(N, k, CHIP, NULL): mfcuspdim(N, k, CHIP);
    4157        3269 :   if (!d1) return NULL;
    4158        2961 :   N1 = N/FC; DN1 = mydivisorsu(N1); lDN1 = lg(DN1);
    4159        2961 :   init = (space == mf_OLD)? -1: 1;
    4160        2961 :   vmf = cgetg(lDN1, t_VEC);
    4161       17479 :   for (i = lDN1 - 1, l = 1; i; i--)
    4162             :   { /* by decreasing level to allow cache */
    4163       14518 :     GEN mf = mfnewinit(FC*DN1[i], k, CHIP, cache, init);
    4164       14518 :     if (mf) gel(vmf, l++) = mf;
    4165       14518 :     init = 0;
    4166             :   }
    4167        2961 :   setlg(vmf,l); vmf = vecreverse(vmf); /* reorder by increasing level */
    4168             : 
    4169        2961 :   L = mfsturmNk(N, k)+1;
    4170        2961 :   vS = vectrunc_init(L);
    4171        2961 :   vMjd = vectrunc_init(L);
    4172        9380 :   for (i = 1; i < l; i++)
    4173             :   {
    4174        6419 :     GEN DNM, mf = gel(vmf,i), S = MF_get_S(mf), vj = MFnew_get_vj(mf);
    4175        6419 :     long a, lDNM, lS = lg(S), M = MF_get_N(mf);
    4176        6419 :     DNM = mydivisorsu(N / M); lDNM = lg(DNM);
    4177       26404 :     for (a = 1; a < lS; a++)
    4178             :     {
    4179       19985 :       GEN tf = gel(S,a);
    4180       19985 :       long b, j = vj[a];
    4181       49553 :       for (b = 1; b < lDNM; b++)
    4182             :       {
    4183       29568 :         long d = DNM[b];
    4184       29568 :         vectrunc_append(vS, mfbd_i(tf, d));
    4185       29568 :         vectrunc_append(vMjd, mkvecsmall3(M, j, d));
    4186             :       }
    4187             :     }
    4188             :   }
    4189        2961 :   return mkmf(NULL, cgetg(1, t_VEC), vS, vMjd, NULL);
    4190             : }
    4191             : 
    4192             : long
    4193        4592 : mfsturm_mf(GEN mf)
    4194             : {
    4195        4592 :   GEN Mindex = MF_get_Mindex(mf);
    4196        4592 :   long n = lg(Mindex)-1;
    4197        4592 :   return n? Mindex[n]-1: 0;
    4198             : }
    4199             : 
    4200             : long
    4201         833 : mfsturm(GEN T)
    4202             : {
    4203             :   long N, nk, dk;
    4204         833 :   GEN CHI, mf = checkMF_i(T);
    4205         833 :   if (mf) return mfsturm_mf(mf);
    4206           7 :   checkNK2(T, &N, &nk, &dk, &CHI, 0);
    4207           7 :   return dk == 1 ? mfsturmNk(N, nk) : mfsturmNk(N, (nk + 1) >> 1);
    4208             : }
    4209             : long
    4210         196 : mfisequal(GEN F, GEN G, long lim)
    4211             : {
    4212         196 :   pari_sp av = avma;
    4213             :   long b;
    4214         196 :   if (!checkmf_i(F)) pari_err_TYPE("mfisequal",F);
    4215         196 :   if (!checkmf_i(G)) pari_err_TYPE("mfisequal",G);
    4216         196 :   b = lim? lim: maxss(mfsturmmf(F), mfsturmmf(G));
    4217         196 :   return gc_long(av, gequal(mfcoefs_i(F, b, 1), mfcoefs_i(G, b, 1)));
    4218             : }
    4219             : 
    4220             : GEN
    4221          35 : mffields(GEN mf)
    4222             : {
    4223          35 :   if (checkmf_i(mf)) return gcopy(mf_get_field(mf));
    4224          35 :   mf = checkMF(mf); return gcopy(MF_get_fields(mf));
    4225             : }
    4226             : 
    4227             : GEN
    4228         364 : mfeigenbasis(GEN mf)
    4229             : {
    4230         364 :   pari_sp ltop = avma;
    4231             :   GEN F, S, v, vP;
    4232             :   long i, l, k, dS;
    4233             : 
    4234         364 :   mf = checkMF(mf);
    4235         364 :   k = MF_get_k(mf);
    4236         364 :   S = MF_get_S(mf); dS = lg(S)-1;
    4237         364 :   if (!dS) return cgetg(1, t_VEC);
    4238         357 :   F = MF_get_newforms(mf);
    4239         357 :   vP = MF_get_fields(mf);
    4240         357 :   if (k == 1)
    4241             :   {
    4242         210 :     if (MF_get_space(mf) == mf_FULL)
    4243             :     {
    4244          14 :       long dE = lg(MF_get_E(mf)) - 1;
    4245          14 :       if (dE) F = rowslice(F, dE+1, dE+dS);
    4246             :     }
    4247         210 :     v = vecmflineardiv_linear(S, F);
    4248         210 :     l = lg(v);
    4249             :   }
    4250             :   else
    4251             :   {
    4252         147 :     GEN (*L)(GEN, GEN) = (MF_get_space(mf) == mf_FULL)? mflinear: mflinear_bhn;
    4253         147 :     l = lg(F); v = cgetg(l, t_VEC);
    4254         511 :     for (i = 1; i < l; i++) gel(v,i) = L(mf, gel(F,i));
    4255             :   }
    4256         945 :   for (i = 1; i < l; i++) mf_setfield(gel(v,i), gel(vP,i));
    4257         357 :   return gc_GEN(ltop, v);
    4258             : }
    4259             : 
    4260             : /* Minv = [M, d, A], v a t_COL; A a Zab, d a t_INT; return (A/d) * M*v */
    4261             : static GEN
    4262        7938 : Minv_RgC_mul(GEN Minv, GEN v)
    4263             : {
    4264        7938 :   GEN M = gel(Minv,1), d = gel(Minv,2), A = gel(Minv,3);
    4265        7938 :   v = RgM_RgC_mul(M, v);
    4266        7938 :   if (!equali1(A))
    4267             :   {
    4268        2072 :     if (typ(A) == t_POL && degpol(A) > 0) A = mkpolmod(A, gel(Minv,4));
    4269        2072 :     v = RgC_Rg_mul(v, A);
    4270             :   }
    4271        7938 :   if (!equali1(d)) v = RgC_Rg_div(v, d);
    4272        7938 :   return v;
    4273             : }
    4274             : static GEN
    4275        1309 : Minv_RgM_mul(GEN Minv, GEN B)
    4276             : {
    4277        1309 :   long j, l = lg(B);
    4278        1309 :   GEN M = cgetg(l, t_MAT);
    4279        6090 :   for (j = 1; j < l; j++) gel(M,j) = Minv_RgC_mul(Minv, gel(B,j));
    4280        1309 :   return M;
    4281             : }
    4282             : /* B * Minv; allow B = NULL for Id */
    4283             : static GEN
    4284        2450 : RgM_Minv_mul(GEN B, GEN Minv)
    4285             : {
    4286        2450 :   GEN M = gel(Minv,1), d = gel(Minv,2), A = gel(Minv,3);
    4287        2450 :   if (B) M = RgM_mul(B, M);
    4288        2450 :   if (!equali1(A))
    4289             :   {
    4290         980 :     if (typ(A) == t_POL) A = mkpolmod(A, gel(Minv,4));
    4291         980 :     M = RgM_Rg_mul(M, A);
    4292             :   }
    4293        2450 :   if (!equali1(d)) M = RgM_Rg_div(M,d);
    4294        2450 :   return M;
    4295             : }
    4296             : 
    4297             : /* perm vector of strictly increasing indices, v a vector or arbitrary length;
    4298             :  * the last r entries of perm fall beyond v.
    4299             :  * Return v o perm[1..(-r)], discarding the last r entries of v */
    4300             : static GEN
    4301        1610 : vecpermute_partial(GEN v, GEN perm, long *r)
    4302             : {
    4303        1610 :   long i, n = lg(v)-1, l = lg(perm);
    4304             :   GEN w;
    4305        1610 :   if (perm[l-1] <= n) { *r = 0; return vecpermute(v,perm); }
    4306          63 :   for (i = 1; i < l; i++)
    4307          63 :     if (perm[i] > n) break;
    4308          21 :   *r = l - i; l = i;
    4309          21 :   w = cgetg(l, typ(v));
    4310          63 :   for (i = 1; i < l; i++) gel(w,i) = gel(v,perm[i]);
    4311          21 :   return w;
    4312             : }
    4313             : 
    4314             : /* given form F, find coeffs of F on mfbasis(mf). If power series, not
    4315             :  * guaranteed correct if precision less than Sturm bound */
    4316             : static GEN
    4317        1449 : mftobasis_i(GEN mf, GEN F)
    4318             : {
    4319             :   GEN v, Mindex, Minv;
    4320        1449 :   if (!MF_get_dim(mf)) return cgetg(1, t_COL);
    4321        1449 :   Mindex = MF_get_Mindex(mf);
    4322        1449 :   Minv = MF_get_Minv(mf);
    4323        1449 :   if (checkmf_i(F))
    4324             :   {
    4325         294 :     long n = Mindex[lg(Mindex)-1];
    4326         294 :     v = vecpermute(mfcoefs_i(F, n, 1), Mindex);
    4327         294 :     return Minv_RgC_mul(Minv, v);
    4328             :   }
    4329             :   else
    4330             :   {
    4331        1155 :     GEN A = gel(Minv,1), d = gel(Minv,2);
    4332             :     long r;
    4333        1155 :     v = F;
    4334        1155 :     switch(typ(F))
    4335             :     {
    4336           0 :       case t_SER: v = sertocol(v);
    4337        1155 :       case t_VEC: case t_COL: break;
    4338           0 :       default: pari_err_TYPE("mftobasis", F);
    4339             :     }
    4340        1155 :     if (lg(v) == 1) pari_err_TYPE("mftobasis",v);
    4341        1155 :     v = vecpermute_partial(v, Mindex, &r);
    4342        1155 :     if (!r) return Minv_RgC_mul(Minv, v); /* single solution */
    4343             :     /* affine space of dimension r */
    4344          21 :     v = RgM_RgC_mul(vecslice(A, 1, lg(v)-1), v);
    4345          21 :     if (!equali1(d)) v = RgC_Rg_div(v,d);
    4346          21 :     return mkvec2(v, vecslice(A, lg(A)-r, lg(A)-1));
    4347             :   }
    4348             : }
    4349             : 
    4350             : static GEN
    4351         910 : const_mat(long n, GEN x)
    4352             : {
    4353         910 :   long j, l = n+1;
    4354         910 :   GEN A = cgetg(l,t_MAT);
    4355        6902 :   for (j = 1; j < l; j++) gel(A,j) = const_col(n, x);
    4356         910 :   return A;
    4357             : }
    4358             : 
    4359             : /* L is the mftobasis of a form on CUSP space. We allow mf_FULL or mf_CUSP */
    4360             : static GEN
    4361         455 : mftonew_i(GEN mf, GEN L, long *plevel)
    4362             : {
    4363             :   GEN S, listMjd, CHI, res, Aclos, Acoef, D, perm;
    4364         455 :   long N1, LC, lD, i, l, t, level, N = MF_get_N(mf);
    4365             : 
    4366         455 :   if (MF_get_k(mf) == 1) pari_err_IMPL("mftonew in weight 1");
    4367         455 :   listMjd = MFcusp_get_vMjd(mf);
    4368         455 :   CHI = MF_get_CHI(mf); LC = mfcharconductor(CHI);
    4369         455 :   S = MF_get_S(mf);
    4370             : 
    4371         455 :   N1 = N/LC;
    4372         455 :   D = mydivisorsu(N1); lD = lg(D);
    4373         455 :   perm = cgetg(N1+1, t_VECSMALL);
    4374        3451 :   for (i = 1; i < lD; i++) perm[D[i]] = i;
    4375         455 :   Aclos = const_mat(lD-1, cgetg(1,t_VEC));
    4376         455 :   Acoef = const_mat(lD-1, cgetg(1,t_VEC));
    4377         455 :   l = lg(listMjd);
    4378        4669 :   for (i = 1; i < l; i++)
    4379             :   {
    4380             :     long M, d;
    4381             :     GEN v;
    4382        4214 :     if (gequal0(gel(L,i))) continue;
    4383         469 :     v = gel(listMjd, i);
    4384         469 :     M = perm[ v[1]/LC ];
    4385         469 :     d = perm[ v[3] ];
    4386         469 :     gcoeff(Aclos,M,d) = vec_append(gcoeff(Aclos,M,d), gel(S,i));
    4387         469 :     gcoeff(Acoef,M,d) = shallowconcat(gcoeff(Acoef,M,d), gel(L,i));
    4388             :   }
    4389         455 :   res = cgetg(l, t_VEC); level = 1;
    4390        3451 :   for (i = t = 1; i < lD; i++)
    4391             :   {
    4392        2996 :     long j, M = D[i]*LC;
    4393        2996 :     GEN gM = utoipos(M);
    4394       26530 :     for (j = 1; j < lD; j++)
    4395             :     {
    4396       23534 :       GEN vf = gcoeff(Aclos,i,j), C, NK;
    4397             :       long d;
    4398       23534 :       if (lg(vf) == 1) continue;
    4399         427 :       d = D[j];
    4400         427 :       C = gcoeff(Acoef,i,j);
    4401         427 :       NK = mf_get_NK(gel(vf, 1));
    4402         427 :       if (d > 1)
    4403             :       { /* remove mfbd(, d) wrappers */
    4404         175 :         long h, lf = lg(vf);
    4405         357 :         for (h = 1; h < lf; h++)
    4406             :         {
    4407         182 :           GEN fd = gel(vf, h);
    4408         182 :           if (mf_get_type(fd) != t_MF_BD || !equaliu(gel(fd,3), d))
    4409           0 :             pari_err_BUG("mftonew [inconsistent multiplier]");
    4410         182 :           gel(vf, h) = gel(fd, 2);
    4411             :         }
    4412             :       }
    4413         427 :       level = ulcm(level, M*d);
    4414         427 :       gel(res,t++) = mkvec3(gM, utoipos(d), mflinear_i(NK,vf,C));
    4415             :     }
    4416             :   }
    4417         455 :   if (plevel) *plevel = level;
    4418         455 :   setlg(res, t); return res;
    4419             : }
    4420             : GEN
    4421         217 : mftonew(GEN mf, GEN F)
    4422             : {
    4423         217 :   pari_sp av = avma;
    4424             :   GEN ES;
    4425             :   long s;
    4426         217 :   mf = checkMF(mf);
    4427         217 :   s = MF_get_space(mf);
    4428         217 :   if (s != mf_FULL && s != mf_CUSP)
    4429           7 :     pari_err_TYPE("mftonew [not a full or cuspidal space]", mf);
    4430         210 :   ES = mftobasisES(mf,F);
    4431         203 :   if (!gequal0(gel(ES,1)))
    4432           0 :     pari_err_TYPE("mftonew [not a cuspidal form]", F);
    4433         203 :   F = gel(ES,2);
    4434         203 :   return gc_GEN(av, mftonew_i(mf,F, NULL));
    4435             : }
    4436             : 
    4437             : static GEN mfeisenstein_i(long k, GEN CHI1, GEN CHI2);
    4438             : 
    4439             : /* mfinit(F * Theta) */
    4440             : static GEN
    4441          98 : mf2init(GEN mf)
    4442             : {
    4443          98 :   GEN CHI = MF_get_CHI(mf), gk = gadd(MF_get_gk(mf), ghalf);
    4444          98 :   long N = MF_get_N(mf);
    4445          98 :   return mfinit_Nkchi(N, itou(gk), mfchiadjust(CHI, gk, N), mf_FULL, 0);
    4446             : }
    4447             : 
    4448             : static long
    4449         637 : mfvec_first_cusp(GEN v)
    4450             : {
    4451         637 :   long i, l = lg(v);
    4452        1533 :   for (i = 1; i < l; i++)
    4453             :   {
    4454        1428 :     GEN F = gel(v,i);
    4455        1428 :     long t = mf_get_type(F);
    4456        1428 :     if (t == t_MF_BD) { F = gel(F,2); t = mf_get_type(F); }
    4457        1428 :     if (t == t_MF_HECKE) { F = gel(F,3); t = mf_get_type(F); }
    4458        1428 :     if (t == t_MF_NEWTRACE) break;
    4459             :   }
    4460         637 :   return i;
    4461             : }
    4462             : /* vF a vector of mf F of type DIV(LINEAR(BAS,L), f) in (lcm) level N,
    4463             :  * F[2]=LINEAR(BAS,L), F[2][2]=BAS=fixed basis (Eisenstein or bhn type),
    4464             :  * F[2][3]=L, F[3]=f; mfvectomat(vF, n) */
    4465             : static GEN
    4466         644 : mflineardivtomat(long N, GEN vF, long n)
    4467             : {
    4468         644 :   GEN F, M, f, fc, ME, dB, B, a0, V = NULL;
    4469         644 :   long lM, lF = lg(vF), j;
    4470             : 
    4471         644 :   if (lF == 1) return cgetg(1,t_MAT);
    4472         637 :   F = gel(vF,1);
    4473         637 :   if (lg(F) == 5)
    4474             :   { /* chicompat */
    4475         273 :     V = gmael(F,4,4);
    4476         273 :     if (typ(V) == t_INT) V = NULL;
    4477             :   }
    4478         637 :   M = gmael(F,2,2); /* BAS */
    4479         637 :   lM = lg(M);
    4480         637 :   j = mfvec_first_cusp(M);
    4481         637 :   if (j == 1) ME = NULL;
    4482             :   else
    4483             :   { /* BAS starts by Eisenstein */
    4484         161 :     ME = mfvectomat(vecslice(M,1,j-1), n, 1);
    4485         161 :     M = vecslice(M, j,lM-1);
    4486             :   }
    4487         637 :   M = bhnmat_extend_nocache(NULL, N, n, 1, M);
    4488         637 :   if (ME) M = shallowconcat(ME,M);
    4489             :   /* M = mfcoefs of BAS */
    4490         637 :   B = cgetg(lF, t_MAT);
    4491         637 :   dB= cgetg(lF, t_VEC);
    4492        3157 :   for (j = 1; j < lF; j++)
    4493             :   {
    4494        2520 :     GEN g = gel(vF, j); /* t_MF_DIV */
    4495        2520 :     gel(B,j) = RgM_RgC_mul(M, gmael(g,2,3));
    4496        2520 :     gel(dB,j)= gmael(g,2,4);
    4497             :   }
    4498         637 :   f = mfcoefsser(gel(F,3),n);
    4499         637 :   a0 = polcoef_i(f, 0, -1);
    4500         637 :   if (gequal0(a0) || gequal1(a0))
    4501         336 :     a0 = NULL;
    4502             :   else
    4503         301 :     f = gdiv(ser_unscale(f, a0), a0);
    4504         637 :   fc = ginv(f);
    4505        3157 :   for (j = 1; j < lF; j++)
    4506             :   {
    4507        2520 :     pari_sp av = avma;
    4508        2520 :     GEN LISer = RgV_to_ser_full(gel(B,j)), f;
    4509        2520 :     if (a0) LISer = gdiv(ser_unscale(LISer, a0), a0);
    4510        2520 :     f = gmul(LISer, fc);
    4511        2520 :     if (a0) f = ser_unscale(f, ginv(a0));
    4512        2520 :     f = sertocol(f); setlg(f, n+2);
    4513        2520 :     if (!gequal1(gel(dB,j))) f = RgC_Rg_div(f, gel(dB,j));
    4514        2520 :     gel(B,j) = gc_upto(av,f);
    4515             :   }
    4516         637 :   if (V) B = gmodulo(QabM_tracerel(V, 0, B), gel(V,1));
    4517         637 :   return B;
    4518             : }
    4519             : 
    4520             : static GEN
    4521         350 : mfheckemat_mfcoefs(GEN mf, GEN B, GEN DATA)
    4522             : {
    4523         350 :   GEN Mindex = MF_get_Mindex(mf), Minv = MF_get_Minv(mf);
    4524         350 :   long j, l = lg(B), sb = mfsturm_mf(mf);
    4525         350 :   GEN b = MF_get_basis(mf), Q = cgetg(l, t_VEC);
    4526        1827 :   for (j = 1; j < l; j++)
    4527             :   {
    4528        1477 :     GEN v = hecke_i(sb, 1, gel(B,j), gel(b,j), DATA); /* Tn b[j] */
    4529        1477 :     settyp(v,t_COL); gel(Q,j) = vecpermute(v, Mindex);
    4530             :   }
    4531         350 :   return Minv_RgM_mul(Minv,Q);
    4532             : }
    4533             : /* T_p^2, p prime, 1/2-integral weight; B = mfcoefs(mf,sb*p^2,1) or (mf,sb,p^2)
    4534             :  * if p|N */
    4535             : static GEN
    4536           7 : mfheckemat_mfcoefs_p2(GEN mf, long p, GEN B)
    4537             : {
    4538           7 :   pari_sp av = avma;
    4539           7 :   GEN DATA = heckef2_data(MF_get_N(mf), p*p);
    4540           7 :   return gc_upto(av, mfheckemat_mfcoefs(mf, B, DATA));
    4541             : }
    4542             : /* convert Mindex from row-index to mfcoef indexation: a(n) is stored in
    4543             :  * mfcoefs()[n+1], so subtract 1 from all indices */
    4544             : static GEN
    4545          49 : Mindex_as_coef(GEN mf)
    4546             : {
    4547          49 :   GEN v, Mindex = MF_get_Mindex(mf);
    4548          49 :   long i, l = lg(Mindex);
    4549          49 :   v = cgetg(l, t_VECSMALL);
    4550         210 :   for (i = 1; i < l; i++) v[i] = Mindex[i]-1;
    4551          49 :   return v;
    4552             : }
    4553             : /* T_p, p prime; B = mfcoefs(mf,sb*p,1) or (mf,sb,p) if p|N; integral weight */
    4554             : static GEN
    4555          35 : mfheckemat_mfcoefs_p(GEN mf, long p, GEN B)
    4556             : {
    4557          35 :   pari_sp av = avma;
    4558          35 :   GEN vm, Q, C, Minv = MF_get_Minv(mf);
    4559          35 :   long lm, k, i, j, l = lg(B), N = MF_get_N(mf);
    4560             : 
    4561          35 :   if (N % p == 0) return Minv_RgM_mul(Minv, rowpermute(B, MF_get_Mindex(mf)));
    4562          21 :   k = MF_get_k(mf);
    4563          21 :   C = gmul(mfchareval(MF_get_CHI(mf), p), powuu(p, k-1));
    4564          21 :   vm = Mindex_as_coef(mf); lm = lg(vm);
    4565          21 :   Q = cgetg(l, t_MAT);
    4566         147 :   for (j = 1; j < l; j++) gel(Q,j) = cgetg(lm, t_COL);
    4567         147 :   for (i = 1; i < lm; i++)
    4568             :   {
    4569         126 :     long m = vm[i], mp = m*p;
    4570         126 :     GEN Cm = (m % p) == 0? C : NULL;
    4571        1260 :     for (j = 1; j < l; j++)
    4572             :     {
    4573        1134 :       GEN S = gel(B,j), s = gel(S, mp + 1);
    4574        1134 :       if (Cm) s = gadd(s, gmul(C, gel(S, m/p + 1)));
    4575        1134 :       gcoeff(Q, i, j) = s;
    4576             :     }
    4577             :   }
    4578          21 :   return gc_upto(av, Minv_RgM_mul(Minv,Q));
    4579             : }
    4580             : /* Matrix of T(p), p prime, dim(mf) > 0 and integral weight */
    4581             : static GEN
    4582         343 : mfheckemat_p(GEN mf, long p)
    4583             : {
    4584         343 :   pari_sp av = avma;
    4585         343 :   long N = MF_get_N(mf), sb = mfsturm_mf(mf);
    4586         343 :   GEN B = (N % p)? mfcoefs_mf(mf, sb * p, 1): mfcoefs_mf(mf, sb, p);
    4587         343 :   return gc_upto(av, mfheckemat_mfcoefs(mf, B, hecke_data(N,p)));
    4588             : }
    4589             : 
    4590             : /* mf_NEW != (0), weight > 1, p prime. Use
    4591             :  * T(p) T(j) = T(j*p) + p^{k-1} \chi(p) 1_{p | j, p \nmid N} T(j/p) */
    4592             : static GEN
    4593         924 : mfnewmathecke_p(GEN mf, long p)
    4594             : {
    4595         924 :   pari_sp av = avma;
    4596         924 :   GEN tf, vj = MFnew_get_vj(mf), CHI = MF_get_CHI(mf);
    4597         924 :   GEN Mindex = MF_get_Mindex(mf), Minv = MF_get_Minv(mf);
    4598         924 :   long N = MF_get_N(mf), k = MF_get_k(mf);
    4599         924 :   long i, j, lvj = lg(vj), lim = vj[lvj-1] * p;
    4600         924 :   GEN M, perm, V, need = zero_zv(lim);
    4601         924 :   GEN C = (N % p)? gmul(mfchareval(CHI,p), powuu(p,k-1)): NULL;
    4602         924 :   tf = mftraceform_new(N, k, CHI);
    4603        4004 :   for (i = 1; i < lvj; i++)
    4604             :   {
    4605        3080 :     j = vj[i]; need[j*p] = 1;
    4606        3080 :     if (N % p && j % p == 0) need[j/p] = 1;
    4607             :   }
    4608         924 :   perm = zero_zv(lim);
    4609         924 :   V = cgetg(lim+1, t_VEC);
    4610       12754 :   for (i = j = 1; i <= lim; i++)
    4611       11830 :     if (need[i]) { gel(V,j) = mfhecke_i(i, N, tf); perm[i] = j; j++; }
    4612         924 :   setlg(V, j);
    4613         924 :   V = bhnmat_extend_nocache(NULL, N, mfsturm_mf(mf), 1, V);
    4614         924 :   V = rowpermute(V, Mindex); /* V[perm[i]] = coeffs(T_i newtrace) */
    4615         924 :   M = cgetg(lvj, t_MAT);
    4616        4004 :   for (i = 1; i < lvj; i++)
    4617             :   {
    4618             :     GEN t;
    4619        3080 :     j = vj[i]; t = gel(V, perm[j*p]);
    4620        3080 :     if (C && j % p == 0) t = RgC_add(t, RgC_Rg_mul(gel(V, perm[j/p]),C));
    4621        3080 :     gel(M,i) = t;
    4622             :   }
    4623         924 :   return gc_upto(av, Minv_RgM_mul(Minv, M));
    4624             : }
    4625             : 
    4626             : GEN
    4627          77 : mfheckemat(GEN mf, GEN vn)
    4628             : {
    4629          77 :   pari_sp av = avma;
    4630          77 :   long lv, lvP, i, N, dim, nk, dk, p, sb, flint = (typ(vn)==t_INT);
    4631             :   GEN CHI, res, vT, FA, B, vP;
    4632             : 
    4633          77 :   mf = checkMF(mf);
    4634          77 :   if (typ(vn) != t_VECSMALL) vn = gtovecsmall(vn);
    4635          77 :   N = MF_get_N(mf); CHI = MF_get_CHI(mf); Qtoss(MF_get_gk(mf), &nk, &dk);
    4636          77 :   dim = MF_get_dim(mf);
    4637          77 :   lv = lg(vn);
    4638          77 :   res = cgetg(lv, t_VEC);
    4639          77 :   FA = cgetg(lv, t_VEC);
    4640          77 :   vP = cgetg(lv, t_VEC);
    4641          77 :   vT = const_vec(vecsmall_max(vn), NULL);
    4642         182 :   for (i = 1; i < lv; i++)
    4643             :   {
    4644         105 :     ulong n = (ulong)labs(vn[i]);
    4645             :     GEN fa;
    4646         105 :     if (!n) pari_err_TYPE("mfheckemat", vn);
    4647         105 :     if (dk == 1 || uissquareall(n, &n)) fa = myfactoru(n);
    4648           0 :     else { n = 0; fa = myfactoru(1); } /* dummy: T_{vn[i]} = 0 */
    4649         105 :     vn[i] = n;
    4650         105 :     gel(FA,i) = fa;
    4651         105 :     gel(vP,i) = gel(fa,1);
    4652             :   }
    4653          77 :   vP = shallowconcat1(vP); vecsmall_sort(vP);
    4654          77 :   vP = vecsmall_uniq_sorted(vP); /* all primes occurring in vn */
    4655          77 :   lvP = lg(vP); if (lvP == 1) goto END;
    4656          56 :   p = vP[lvP-1];
    4657          56 :   sb = mfsturm_mf(mf);
    4658          56 :   if (dk == 1 && nk != 1 && MF_get_space(mf) == mf_NEW)
    4659          21 :     B = NULL; /* special purpose mfnewmathecke_p is faster */
    4660          35 :   else if (lvP == 2 && N % p == 0)
    4661          21 :     B = mfcoefs_mf(mf, sb, dk==2? p*p: p); /* single prime | N, can optimize */
    4662             :   else
    4663          14 :     B = mfcoefs_mf(mf, sb * (dk==2? p*p: p), 1); /* general initialization */
    4664         126 :   for (i = 1; i < lvP; i++)
    4665             :   {
    4666          70 :     long j, l, q, e = 1;
    4667             :     GEN C, Tp, u1, u0;
    4668          70 :     p = vP[i];
    4669         189 :     for (j = 1; j < lv; j++) e = maxss(e, z_lval(vn[j], p));
    4670          70 :     if (!B)
    4671          28 :       Tp = mfnewmathecke_p(mf, p);
    4672          42 :     else if (dk == 2)
    4673           7 :       Tp = mfheckemat_mfcoefs_p2(mf,p, (lvP==2||N%p)? B: matdeflate(sb,p*p,B));
    4674             :     else
    4675          35 :       Tp = mfheckemat_mfcoefs_p(mf, p, (lvP==2||N%p)? B: matdeflate(sb,p,B));
    4676          70 :     gel(vT, p) = Tp;
    4677          70 :     if (e == 1) continue;
    4678          14 :     u0 = gen_1;
    4679          14 :     if (dk == 2)
    4680             :     {
    4681           0 :       C = N % p? gmul(mfchareval(CHI,p*p), powuu(p, nk-2)): NULL;
    4682           0 :       if (e == 2) u0 = uutoQ(p+1,p); /* special case T_{p^4} */
    4683             :     }
    4684             :     else
    4685          14 :       C = N % p? gmul(mfchareval(CHI,p),   powuu(p, nk-1)): NULL;
    4686          28 :     for (u1=Tp, q=p, l=2; l <= e; l++)
    4687             :     { /* u0 = T_{p^{l-2}}, u1 = T_{p^{l-1}} for l > 2 */
    4688          14 :       GEN v = gmul(Tp, u1);
    4689          14 :       if (C) v = gsub(v, gmul(C, u0));
    4690             :       /* q = p^l, vT[q] = T_q for k integer else T_{q^2} */
    4691          14 :       q *= p; u0 = u1; gel(vT, q) = u1 = v;
    4692             :     }
    4693             :   }
    4694          56 : END:
    4695             :   /* vT[p^e] = T_{p^e} for all p^e occurring below */
    4696         182 :   for (i = 1; i < lv; i++)
    4697             :   {
    4698         105 :     long n = vn[i], j, lP;
    4699             :     GEN fa, P, E, M;
    4700         105 :     if (n == 0) { gel(res,i) = zeromat(dim,dim); continue; }
    4701         105 :     if (n == 1) { gel(res,i) = matid(dim); continue; }
    4702          77 :     fa = gel(FA,i);
    4703          77 :     P = gel(fa,1); lP = lg(P);
    4704          77 :     E = gel(fa,2); M = gel(vT, upowuu(P[1], E[1]));
    4705          84 :     for (j = 2; j < lP; j++) M = RgM_mul(M, gel(vT, upowuu(P[j], E[j])));
    4706          77 :     gel(res,i) = M;
    4707             :   }
    4708          77 :   if (flint) res = gel(res,1);
    4709          77 :   return gc_GEN(av, res);
    4710             : }
    4711             : 
    4712             : /* f = \sum_i v[i] T_listj[i] (Trace Form) attached to v; replace by f/a_1(f) */
    4713             : static GEN
    4714        1540 : mf_normalize(GEN mf, GEN v)
    4715             : {
    4716        1540 :   GEN c, dc = NULL, M = MF_get_M(mf), Mindex = MF_get_Mindex(mf);
    4717        1540 :   v = Q_primpart(v);
    4718        1540 :   c = RgMrow_RgC_mul(M, v, 2); /* a_1(f) */
    4719        1540 :   if (gequal1(c)) return v;
    4720         945 :   if (typ(c) == t_POL) c = gmodulo(c, mfcharpol(MF_get_CHI(mf)));
    4721         945 :   if (typ(c) == t_POLMOD && varn(gel(c,1)) == 1 && degpol(gel(c,1)) >= 40
    4722           7 :                          && Mindex[1] == 2
    4723           7 :                          && mfcharorder(MF_get_CHI(mf)) <= 2)
    4724           7 :   { /* normalize using expansion at infinity (small coefficients) */
    4725           7 :     GEN w, P = gel(c,1), a1 = gel(c,2);
    4726           7 :     long i, l = lg(Mindex);
    4727           7 :     w = cgetg(l, t_COL);
    4728           7 :     gel(w,1) = gen_1;
    4729         280 :     for (i = 2; i < l; i++)
    4730             :     {
    4731         273 :       c = liftpol_shallow(RgMrow_RgC_mul(M, v, Mindex[i]));
    4732         273 :       gel(w,i) = QXQ_div(c, a1, P);
    4733             :     }
    4734             :     /* w = expansion at oo of normalized form */
    4735           7 :     v = Minv_RgC_mul(MF_get_Minv(mf), Q_remove_denom(w, &dc));
    4736           7 :     v = gmodulo(v, P); /* back to mfbasis coefficients */
    4737             :   }
    4738             :   else
    4739             :   {
    4740         938 :     c = ginv(c);
    4741         938 :     if (typ(c) == t_POLMOD) c = Q_remove_denom(c, &dc);
    4742         938 :     v = RgC_Rg_mul(v, c);
    4743             :   }
    4744         945 :   if (dc) v = RgC_Rg_div(v, dc);
    4745         945 :   return v;
    4746             : }
    4747             : static void
    4748         455 : pol_red(GEN NF, GEN *pP, GEN *pa, long flag)
    4749             : {
    4750         455 :   GEN dP, a, P = *pP;
    4751         455 :   long d = degpol(P);
    4752             : 
    4753         455 :   *pa = a = pol_x(varn(P));
    4754         455 :   if (d * (NF ? nf_get_degree(NF): 1) > 30) return;
    4755             : 
    4756         448 :   dP = RgX_disc(P);
    4757         448 :   if (typ(dP) != t_INT)
    4758         112 :   { dP = gnorm(dP); if (typ(dP) != t_INT) pari_err_BUG("mfnewsplit"); }
    4759         448 :   if (d == 2 || expi(dP) < 62)
    4760             :   {
    4761         413 :     if (expi(dP) < 31)
    4762         406 :       P = NF? rnfpolredabs(NF, P,flag): polredabs0(P,flag);
    4763             :     else
    4764           7 :       P = NF? rnfpolredbest(NF,P,flag): polredbest(P,flag);
    4765         413 :     if (flag)
    4766             :     {
    4767         385 :       a = gel(P,2); if (typ(a) == t_POLMOD) a = gel(a,2);
    4768         385 :       P = gel(P,1);
    4769             :     }
    4770             :   }
    4771         448 :   *pP = P;
    4772         448 :   *pa = a;
    4773             : }
    4774             : 
    4775             : /* Diagonalize and normalize. See mfsplit for meaning of flag. */
    4776             : static GEN
    4777        1092 : mfspclean(GEN mf, GEN mf0, GEN NF, long ord, GEN simplesp, long flag)
    4778             : {
    4779        1092 :   const long vz = 1;
    4780        1092 :   long i, l = lg(simplesp), dim = MF_get_dim(mf);
    4781        1092 :   GEN res = cgetg(l, t_MAT), pols = cgetg(l, t_VEC);
    4782        1092 :   GEN zeros = (mf == mf0)? NULL: zerocol(dim - MF_get_dim(mf0));
    4783        2660 :   for (i = 1; i < l; i++)
    4784             :   {
    4785        1568 :     GEN ATP = gel(simplesp, i), A = gel(ATP,1), P = gel(ATP,3);
    4786        1568 :     long d = degpol(P);
    4787        1568 :     GEN a, v = (flag && d > flag)? NULL: gel(A,1);
    4788        1568 :     if (d == 1) P = pol_x(vz);
    4789             :     else
    4790             :     {
    4791         455 :       pol_red(NF, &P, &a, !!v);
    4792         455 :       if (v)
    4793             :       { /* Mod(a,P) root of charpoly(T), K*gpowers(a) = eigenvector of T */
    4794         427 :         GEN K, den, M = cgetg(d+1, t_MAT), T = gel(ATP,2);
    4795             :         long j;
    4796         427 :         T = shallowtrans(T);
    4797         427 :         gel(M,1) = vec_ei(d,1); /* basis of cyclic vectors */
    4798        1372 :         for (j = 2; j <= d; j++) gel(M,j) = RgM_RgC_mul(T, gel(M,j-1));
    4799         427 :         M = Q_primpart(M);
    4800         147 :         K = NF? ZabM_inv(liftpol_shallow(M), nf_get_pol(NF), ord, &den)
    4801         427 :               : ZM_inv(M,&den);
    4802         427 :         K = shallowtrans(K);
    4803         427 :         v = gequalX(a)? pol_x_powers(d, vz): RgXQ_powers(a, d-1, P);
    4804         427 :         v = gmodulo(RgM_RgC_mul(A, RgM_RgC_mul(K,v)), P);
    4805             :       }
    4806             :     }
    4807        1568 :     if (v)
    4808             :     {
    4809        1540 :       v = mf_normalize(mf0, v); if (zeros) v = shallowconcat(zeros,v);
    4810        1540 :       gel(res,i) = v; if (flag) setlg(res,i+1);
    4811             :     }
    4812             :     else
    4813          28 :       gel(res,i) = zerocol(dim);
    4814        1568 :     gel(pols,i) = P;
    4815             :   }
    4816        1092 :   return mkvec2(res, pols);
    4817             : }
    4818             : 
    4819             : /* return v = v_{X-r}(P), and set Z = P / (X-r)^v */
    4820             : static long
    4821          70 : RgX_valrem_root(GEN P, GEN r, GEN *Z)
    4822             : {
    4823             :   long v;
    4824         140 :   for (v = 0; degpol(P); v++)
    4825             :   {
    4826         140 :     GEN t, Q = RgX_div_by_X_x(P, r, &t);
    4827         140 :     if (!gequal0(t)) break;
    4828          70 :     P = Q;
    4829             :   }
    4830          70 :   *Z = P; return v;
    4831             : }
    4832             : static GEN
    4833        1533 : mynffactor(GEN NF, GEN P, long dimlim)
    4834             : {
    4835             :   long i, l, v;
    4836             :   GEN R, E;
    4837        1533 :   if (dimlim != 1)
    4838             :   {
    4839         966 :     R = NF? nffactor(NF, P): QX_factor(P);
    4840         966 :     if (!dimlim) return R;
    4841          21 :     E = gel(R,2);
    4842          21 :     R = gel(R,1); l = lg(R);
    4843          98 :     for (i = 1; i < l; i++)
    4844          91 :       if (degpol(gel(R,i)) > dimlim) break;
    4845          21 :     if (i == 1) return NULL;
    4846          21 :     setlg(E,i);
    4847          21 :     setlg(R,i); return mkmat2(R, E);
    4848             :   }
    4849             :   /* dimlim = 1 */
    4850         567 :   R = nfroots(NF, P); l = lg(R);
    4851         567 :   if (l == 1) return NULL;
    4852         504 :   v = varn(P);
    4853         504 :   settyp(R, t_COL);
    4854         504 :   if (degpol(P) == l-1)
    4855         448 :     E = const_col(l-1, gen_1);
    4856             :   else
    4857             :   {
    4858          56 :     E = cgetg(l, t_COL);
    4859         126 :     for (i = 1; i < l; i++) gel(E,i) = utoi(RgX_valrem_root(P, gel(R,i), &P));
    4860             :   }
    4861         504 :   R = deg1_from_roots(R, v);
    4862         504 :   return mkmat2(R, E);
    4863             : }
    4864             : 
    4865             : /* Let K be a number field attached to NF (Q if NF = NULL). A K-vector
    4866             :  * space of dimension d > 0 is given by a t_MAT A (n x d, full column rank)
    4867             :  * giving a K-basis, X a section (d x n: left pseudo-inverse of A). Return a
    4868             :  * pair (T, fa), where T is an element of the Hecke algebra (a sum of Tp taken
    4869             :  * from vector vTp) acting on A (a d x d t_MAT) and fa is the factorization of
    4870             :  * its characteristic polynomial, limited to factors of degree <= dimlim if
    4871             :  * dimlim != 0 (return NULL if there are no factors of degree <= dimlim) */
    4872             : static GEN
    4873        1358 : findbestsplit(GEN NF, GEN vTp, GEN A, GEN X, long dimlim, long vz)
    4874             : {
    4875        1358 :   GEN T = NULL, Tkeep = NULL, fakeep = NULL;
    4876        1358 :   long lmax = 0, i, lT = lg(vTp);
    4877        1785 :   for (i = 1; i < lT; i++)
    4878             :   {
    4879        1785 :     GEN D, P, E, fa, TpA = gel(vTp,i);
    4880             :     long l;
    4881        2828 :     if (typ(TpA) == t_INT) break;
    4882        1533 :     if (lg(TpA) > lg(A)) TpA = RgM_mul(X, RgM_mul(TpA, A)); /* Tp | A */
    4883        1533 :     T = T ? RgM_add(T, TpA) : TpA;
    4884        1533 :     if (!NF) { P = QM_charpoly_ZX(T); setvarn(P, vz); }
    4885             :     else
    4886             :     {
    4887         294 :       P = charpoly(Q_remove_denom(T, &D), vz);
    4888         294 :       if (D) P = gdiv(RgX_unscale(P, D), powiu(D, degpol(P)));
    4889             :     }
    4890        1533 :     fa = mynffactor(NF, P, dimlim);
    4891        1533 :     if (!fa) return NULL;
    4892        1470 :     E = gel(fa, 2);
    4893             :     /* characteristic polynomial is separable ? */
    4894        1470 :     if (isint1(vecmax(E))) { Tkeep = T; fakeep = fa; break; }
    4895         427 :     l = lg(E);
    4896             :     /* characteristic polynomial has more factors than before ? */
    4897         427 :     if (l > lmax) { lmax = l; Tkeep = T; fakeep = fa; }
    4898             :   }
    4899        1295 :   return mkvec2(Tkeep, fakeep);
    4900             : }
    4901             : 
    4902             : static GEN
    4903         294 : nfcontent(GEN nf, GEN v)
    4904             : {
    4905         294 :   long i, l = lg(v);
    4906         294 :   GEN c = gel(v,1);
    4907        1512 :   for (i = 2; i < l; i++) c = idealadd(nf, c, gel(v,i));
    4908         294 :   if (typ(c) == t_MAT && gequal1(gcoeff(c,1,1))) c = gen_1;
    4909         294 :   return c;
    4910             : }
    4911             : static GEN
    4912         455 : nf_primpart(GEN nf, GEN x)
    4913             : {
    4914         455 :   switch(typ(x))
    4915             :   {
    4916         294 :     case t_COL:
    4917             :     {
    4918         294 :       GEN A = matalgtobasis(nf, x), c = nfcontent(nf, A);
    4919         294 :       if (typ(c) == t_INT) return x;
    4920          35 :       c = idealred_elt(nf,c);
    4921          35 :       A = Q_primpart( nfC_nf_mul(nf, A, Q_primpart(nfinv(nf,c))) );
    4922          35 :       A = liftpol_shallow( matbasistoalg(nf, A) );
    4923          35 :       if (gexpo(A) > gexpo(x)) A = x;
    4924          35 :       return A;
    4925             :     }
    4926         455 :     case t_MAT: pari_APPLY_same(nf_primpart(nf, gel(x,i)));
    4927           0 :     default:
    4928           0 :       pari_err_TYPE("nf_primpart", x);
    4929             :       return NULL; /*LCOV_EXCL_LINE*/
    4930             :   }
    4931             : }
    4932             : 
    4933             : /* rotate entries of v to accomodate new entry 'x' (push out oldest entry) */
    4934             : static void
    4935        1239 : vecpush(GEN v, GEN x)
    4936             : {
    4937             :   long i;
    4938        6195 :   for (i = lg(v)-1; i > 1; i--) gel(v,i) = gel(v,i-1);
    4939        1239 :   gel(v,1) = x;
    4940        1239 : }
    4941             : 
    4942             : /* sort t_VEC of vector spaces by increasing dimension */
    4943             : static GEN
    4944        1092 : sort_by_dim(GEN v)
    4945             : {
    4946        1092 :   long i, l = lg(v);
    4947        1092 :   GEN D = cgetg(l, t_VECSMALL);
    4948        2660 :   for (i = 1; i < l; i++) D[i] = lg(gmael(v,i,2));
    4949        1092 :   return vecpermute(v , vecsmall_indexsort(D));
    4950             : }
    4951             : static GEN
    4952        1092 : split_starting_space(GEN mf)
    4953             : {
    4954        1092 :   long d = MF_get_dim(mf), d2;
    4955        1092 :   GEN id = matid(d);
    4956        1092 :   switch(MF_get_space(mf))
    4957             :   {
    4958        1085 :     case mf_NEW:
    4959        1085 :     case mf_CUSP: return mkvec2(id, id);
    4960             :   }
    4961           7 :   d2 = lg(MF_get_S(mf))-1;
    4962           7 :   return mkvec2(vecslice(id, d-d2+1,d),
    4963             :                 shallowconcat(zeromat(d2,d-d2),matid(d2)));
    4964             : }
    4965             : /* If dimlim > 0, keep only the dimension <= dimlim eigenspaces.
    4966             :  * See mfsplit for the meaning of flag. */
    4967             : static GEN
    4968        1491 : split_ii(GEN mf, long dimlim, long flag, GEN vSP, long *pnewd)
    4969             : {
    4970             :   forprime_t iter;
    4971        1491 :   GEN CHI = MF_get_CHI(mf), empty = cgetg(1, t_VEC), mf0 = mf;
    4972             :   GEN NF, POLCYC, todosp, Tpbigvec, simplesp;
    4973        1491 :   long N = MF_get_N(mf), k = MF_get_k(mf);
    4974        1491 :   long ord, FC, NEWT, dimsimple = 0, newd = -1;
    4975        1491 :   const long NBH = 5, vz = 1;
    4976             :   ulong p;
    4977             : 
    4978        1491 :   switch(MF_get_space(mf))
    4979             :   {
    4980        1197 :     case mf_NEW: break;
    4981         287 :     case mf_CUSP:
    4982             :     case mf_FULL:
    4983             :     {
    4984             :       GEN CHIP;
    4985         287 :       if (k > 1) { mf0 = mfinittonew(mf); break; }
    4986         259 :       CHIP = mfchartoprimitive(CHI, NULL);
    4987         259 :       newd = lg(MF_get_S(mf))-1 - mfolddim_i(N, k, CHIP, vSP);
    4988         259 :       break;
    4989             :     }
    4990           7 :     default: pari_err_TYPE("mfsplit [space does not contain newspace]", mf);
    4991             :       return NULL; /*LCOV_EXCL_LINE*/
    4992             :   }
    4993        1484 :   if (newd < 0) newd = mf0? MF_get_dim(mf0): 0;
    4994        1484 :   *pnewd = newd;
    4995        1484 :   if (!newd) return mkvec2(cgetg(1, t_MAT), empty);
    4996             : 
    4997        1092 :   NEWT = (k > 1 && MF_get_space(mf0) == mf_NEW);
    4998        1092 :   todosp = mkvec( split_starting_space(mf0) );
    4999        1092 :   simplesp = empty;
    5000        1092 :   FC = mfcharconductor(CHI);
    5001        1092 :   ord = mfcharorder(CHI);
    5002        1092 :   if (ord <= 2) NF = POLCYC = NULL;
    5003             :   else
    5004             :   {
    5005         210 :     POLCYC = mfcharpol(CHI);
    5006         210 :     NF = nfinit(POLCYC,DEFAULTPREC);
    5007             :   }
    5008        1092 :   Tpbigvec = zerovec(NBH);
    5009        1092 :   u_forprime_init(&iter, 2, ULONG_MAX);
    5010        1526 :   while (dimsimple < newd && (p = u_forprime_next(&iter)))
    5011             :   {
    5012             :     GEN nextsp;
    5013             :     long ind;
    5014        1526 :     if (N % (p*p) == 0 && N/p % FC == 0) continue; /* T_p = 0 in this case */
    5015        1239 :     vecpush(Tpbigvec, NEWT? mfnewmathecke_p(mf0,p): mfheckemat_p(mf0,p));
    5016        1239 :     nextsp = empty;
    5017        1638 :     for (ind = 1; ind < lg(todosp); ind++)
    5018             :     {
    5019        1358 :       GEN tmp = gel(todosp, ind), fa, P, E, D, Tp, DTp;
    5020        1358 :       GEN A = gel(tmp, 1);
    5021        1358 :       GEN X = gel(tmp, 2);
    5022             :       long lP, i;
    5023        1358 :       tmp = findbestsplit(NF, Tpbigvec, A, X, dimlim, vz);
    5024        1477 :       if (!tmp) continue; /* nothing there */
    5025        1295 :       Tp = gel(tmp, 1);
    5026        1295 :       fa = gel(tmp, 2);
    5027        1295 :       P = gel(fa, 1);
    5028        1295 :       E = gel(fa, 2); lP = lg(P);
    5029             :       /* lP > 1 */
    5030        1295 :       if (DEBUGLEVEL) err_printf("Exponents = %Ps\n", E);
    5031        1295 :       if (lP == 2)
    5032             :       {
    5033         868 :         GEN P1 = gel(P,1);
    5034         868 :         long e1 = itos(gel(E,1)), d1 = degpol(P1);
    5035         868 :         if (e1 * d1 == lg(Tp)-1)
    5036             :         {
    5037         819 :           if (e1 > 1) nextsp = vec_append(nextsp, mkvec2(A,X));
    5038             :           else
    5039             :           { /* simple module */
    5040         721 :             simplesp = vec_append(simplesp, mkvec3(A,Tp,P1));
    5041         980 :             if ((dimsimple += d1) == newd) goto END;
    5042             :           }
    5043         119 :           continue;
    5044             :         }
    5045             :       }
    5046             :       /* Found splitting */
    5047         476 :       DTp = Q_remove_denom(Tp, &D);
    5048        1295 :       for (i = 1; i < lP; i++)
    5049             :       {
    5050        1078 :         GEN Ai, Xi, dXi, AAi, v, y, Pi = gel(P,i);
    5051        1078 :         Ai = RgX_RgM_eval(D? RgX_rescale(Pi,D): Pi, DTp);
    5052        1078 :         Ai = QabM_ker(Ai, POLCYC, ord);
    5053        1078 :         if (NF) Ai = nf_primpart(NF, Ai);
    5054             : 
    5055        1078 :         AAi = RgM_mul(A, Ai);
    5056             :         /* gives section, works on nonsquare matrices */
    5057        1078 :         Xi = QabM_pseudoinv(Ai, POLCYC, ord, &v, &dXi);
    5058        1078 :         Xi = RgM_Rg_div(Xi, dXi);
    5059        1078 :         y = gel(v,1);
    5060        1078 :         if (isint1(gel(E,i)))
    5061             :         {
    5062         847 :           GEN Tpi = RgM_mul(Xi, RgM_mul(rowpermute(Tp,y), Ai));
    5063         847 :           simplesp = vec_append(simplesp, mkvec3(AAi, Tpi, Pi));
    5064         847 :           if ((dimsimple += degpol(Pi)) == newd) goto END;
    5065             :         }
    5066             :         else
    5067             :         {
    5068         231 :           Xi = RgM_mul(Xi, rowpermute(X,y));
    5069         231 :           nextsp = vec_append(nextsp, mkvec2(AAi, Xi));
    5070             :         }
    5071             :       }
    5072             :     }
    5073         280 :     todosp = nextsp; if (lg(todosp) == 1) break;
    5074             :   }
    5075           0 : END:
    5076        1092 :   if (DEBUGLEVEL) err_printf("end split, need to clean\n");
    5077        1092 :   return mfspclean(mf, mf0, NF, ord, sort_by_dim(simplesp), flag);
    5078             : }
    5079             : static GEN
    5080          42 : dim_filter(GEN v, long dim)
    5081             : {
    5082          42 :   GEN P = gel(v,2);
    5083          42 :   long j, l = lg(P);
    5084         175 :   for (j = 1; j < l; j++)
    5085         161 :     if (degpol(gel(P,j)) > dim)
    5086             :     {
    5087          28 :       v = mkvec2(vecslice(gel(v,1),1,j-1), vecslice(P,1,j-1));
    5088          28 :       break;
    5089             :     }
    5090          42 :   return v;
    5091             : }
    5092             : static long
    5093         287 : dim_sum(GEN v)
    5094             : {
    5095         287 :   GEN P = gel(v,2);
    5096         287 :   long j, l = lg(P), d = 0;
    5097         707 :   for (j = 1; j < l; j++) d += degpol(gel(P,j));
    5098         287 :   return d;
    5099             : }
    5100             : static GEN
    5101        1169 : split_i(GEN mf, long dimlim, long flag)
    5102        1169 : { long junk; return split_ii(mf, dimlim, flag, NULL, &junk); }
    5103             : /* mf is either already split or output by mfinit. Splitting is done only for
    5104             :  * newspace except in weight 1. If flag = 0 (default) split completely.
    5105             :  * If flag = d > 0, only give the Galois polynomials in degree > d
    5106             :  * Flag is ignored if dimlim = 1. */
    5107             : GEN
    5108         112 : mfsplit(GEN mf0, long dimlim, long flag)
    5109             : {
    5110         112 :   pari_sp av = avma;
    5111         112 :   GEN v, mf = checkMF_i(mf0);
    5112         112 :   if (!mf) pari_err_TYPE("mfsplit", mf0);
    5113         112 :   if ((v = obj_check(mf, MF_SPLIT)))
    5114          42 :   { if (dimlim) v = dim_filter(v, dimlim); }
    5115          70 :   else if (dimlim && (v = obj_check(mf, MF_SPLITN)))
    5116          21 :   { v = (itos(gel(v,1)) >= dimlim)? dim_filter(gel(v,2), dimlim): NULL; }
    5117         112 :   if (!v)
    5118             :   {
    5119             :     long newd;
    5120          70 :     v = split_ii(mf, dimlim, flag, NULL, &newd);
    5121          70 :     if (lg(v) == 1) obj_insert(mf, MF_SPLITN, mkvec2(utoi(dimlim), v));
    5122          70 :     else if (!flag)
    5123             :     {
    5124          49 :       if (dim_sum(v) == newd) obj_insert(mf, MF_SPLIT,v);
    5125          21 :       else obj_insert(mf, MF_SPLITN, mkvec2(utoi(dimlim), v));
    5126             :     }
    5127             :   }
    5128         112 :   return gc_GEN(av, v);
    5129             : }
    5130             : static GEN
    5131         252 : split(GEN mf) { return split_i(mf,0,0); }
    5132             : GEN
    5133         819 : MF_get_newforms(GEN mf) { return gel(obj_checkbuild(mf,MF_SPLIT,&split),1); }
    5134             : GEN
    5135         616 : MF_get_fields(GEN mf) { return gel(obj_checkbuild(mf,MF_SPLIT,&split),2); }
    5136             : 
    5137             : /*************************************************************************/
    5138             : /*                     Modular forms of Weight 1                         */
    5139             : /*************************************************************************/
    5140             : /* S_1(G_0(N)), small N. Return 1 if definitely empty; return 0 if maybe
    5141             :  * nonempty  */
    5142             : static int
    5143       16632 : wt1empty(long N)
    5144             : {
    5145       16632 :   if (N <= 100) switch (N)
    5146             :   { /* nonempty [32/100] */
    5147        5453 :     case 23: case 31: case 39: case 44: case 46:
    5148             :     case 47: case 52: case 55: case 56: case 57:
    5149             :     case 59: case 62: case 63: case 68: case 69:
    5150             :     case 71: case 72: case 76: case 77: case 78:
    5151             :     case 79: case 80: case 83: case 84: case 87:
    5152             :     case 88: case 92: case 93: case 94: case 95:
    5153        5453 :     case 99: case 100: return 0;
    5154        3549 :     default: return 1;
    5155             :   }
    5156        7630 :   if (N <= 600) switch(N)
    5157             :   { /* empty [111/500] */
    5158         336 :     case 101: case 102: case 105: case 106: case 109:
    5159             :     case 113: case 121: case 122: case 123: case 125:
    5160             :     case 130: case 134: case 137: case 146: case 149:
    5161             :     case 150: case 153: case 157: case 162: case 163:
    5162             :     case 169: case 170: case 173: case 178: case 181:
    5163             :     case 182: case 185: case 187: case 193: case 194:
    5164             :     case 197: case 202: case 205: case 210: case 218:
    5165             :     case 221: case 226: case 233: case 241: case 242:
    5166             :     case 245: case 246: case 250: case 257: case 265:
    5167             :     case 267: case 269: case 274: case 277: case 281:
    5168             :     case 289: case 293: case 298: case 305: case 306:
    5169             :     case 313: case 314: case 317: case 326: case 337:
    5170             :     case 338: case 346: case 349: case 353: case 361:
    5171             :     case 362: case 365: case 369: case 370: case 373:
    5172             :     case 374: case 377: case 386: case 389: case 394:
    5173             :     case 397: case 401: case 409: case 410: case 421:
    5174             :     case 425: case 427: case 433: case 442: case 449:
    5175             :     case 457: case 461: case 466: case 481: case 482:
    5176             :     case 485: case 490: case 493: case 509: case 514:
    5177             :     case 521: case 530: case 533: case 534: case 538:
    5178             :     case 541: case 545: case 554: case 557: case 562:
    5179             :     case 565: case 569: case 577: case 578: case 586:
    5180         336 :     case 593: return 1;
    5181        6979 :     default: return 0;
    5182             :   }
    5183         315 :   return 0;
    5184             : }
    5185             : 
    5186             : static GEN
    5187          28 : initwt1trace(GEN mf)
    5188             : {
    5189          28 :   GEN S = MF_get_S(mf), v, H;
    5190             :   long l, i;
    5191          28 :   if (lg(S) == 1) return mftrivial();
    5192          28 :   H = mfheckemat(mf, Mindex_as_coef(mf));
    5193          28 :   l = lg(H); v = cgetg(l, t_VEC);
    5194          63 :   for (i = 1; i < l; i++) gel(v,i) = gtrace(gel(H,i));
    5195          28 :   v = Minv_RgC_mul(MF_get_Minv(mf), v);
    5196          28 :   return mflineardiv_linear(S, v, 1);
    5197             : }
    5198             : static GEN
    5199          21 : initwt1newtrace(GEN mf)
    5200             : {
    5201          21 :   GEN v, D, S, Mindex, CHI = MF_get_CHI(mf);
    5202          21 :   long FC, lD, i, sb, N1, N2, lM, N = MF_get_N(mf);
    5203          21 :   CHI = mfchartoprimitive(CHI, &FC);
    5204          21 :   if (N % FC || mfcharparity(CHI) == 1) return mftrivial();
    5205          21 :   D = mydivisorsu(N/FC); lD = lg(D);
    5206          21 :   S = MF_get_S(mf);
    5207          21 :   if (lg(S) == 1) return mftrivial();
    5208          21 :   N2 = newd_params2(N);
    5209          21 :   N1 = N / N2;
    5210          21 :   Mindex = MF_get_Mindex(mf);
    5211          21 :   lM = lg(Mindex);
    5212          21 :   sb = Mindex[lM-1];
    5213          21 :   v = zerovec(sb+1);
    5214          42 :   for (i = 1; i < lD; i++)
    5215             :   {
    5216          21 :     long M = FC*D[i], j;
    5217          21 :     GEN tf = initwt1trace(M == N? mf: mfinit_Nkchi(M, 1, CHI, mf_CUSP, 0));
    5218             :     GEN listd, w;
    5219          21 :     if (mf_get_type(tf) == t_MF_CONST) continue;
    5220          21 :     w = mfcoefs_i(tf, sb, 1);
    5221          21 :     if (M == N) { v = gadd(v, w); continue; }
    5222           0 :     listd = mydivisorsu(u_ppo(ugcd(N/M, N1), FC));
    5223           0 :     for (j = 1; j < lg(listd); j++)
    5224             :     {
    5225           0 :       long d = listd[j], d2 = d*d; /* coprime to FC */
    5226           0 :       GEN dk = mfchareval(CHI, d);
    5227           0 :       long NMd = N/(M*d), m;
    5228           0 :       for (m = 1; m <= sb/d2; m++)
    5229             :       {
    5230           0 :         long be = mubeta2(NMd, m);
    5231           0 :         if (be)
    5232             :         {
    5233           0 :           GEN c = gmul(dk, gmulsg(be, gel(w, m+1)));
    5234           0 :           long n = m*d2;
    5235           0 :           gel(v, n+1) = gadd(gel(v, n+1), c);
    5236             :         }
    5237             :       }
    5238             :     }
    5239             :   }
    5240          21 :   if (gequal0(gel(v,2))) return mftrivial();
    5241          21 :   v = vecpermute(v,Mindex);
    5242          21 :   v = Minv_RgC_mul(MF_get_Minv(mf), v);
    5243          21 :   return mflineardiv_linear(S, v, 1);
    5244             : }
    5245             : 
    5246             : /* i*p + 1, i*p < lim corresponding to a_p(f_j), a_{2p}(f_j)...  */
    5247             : static GEN
    5248        1834 : pindices(long p, long lim)
    5249             : {
    5250        1834 :   GEN v = cgetg(lim, t_VECSMALL);
    5251             :   long i, ip;
    5252       22190 :   for (i = 1, ip = p + 1; ip < lim; i++, ip += p) v[i] = ip;
    5253        1834 :   setlg(v, i); return v;
    5254             : }
    5255             : 
    5256             : /* assume !wt1empty(N), in particular N>25 */
    5257             : /* Returns [[lim,p], mf (weight 2), p*lim x dim matrix] */
    5258             : static GEN
    5259        1834 : mf1_pre(long N)
    5260             : {
    5261             :   pari_timer tt;
    5262             :   GEN mf, v, L, I, M, Minv, den;
    5263             :   long B, lim, LIM, p;
    5264             : 
    5265        1834 :   if (DEBUGLEVEL) timer_start(&tt);
    5266        1834 :   mf = mfinit_Nkchi(N, 2, mfchartrivial(), mf_CUSP, 0);
    5267        1834 :   if (DEBUGLEVEL)
    5268           0 :     timer_printf(&tt, "mf1basis [pre]: S_2(%ld), dim = %ld",
    5269             :                  N, MF_get_dim(mf));
    5270        1834 :   M = MF_get_M(mf); Minv = MF_get_Minv(mf); den = gel(Minv,2);
    5271        1834 :   B = mfsturm_mf(mf);
    5272        1834 :   if (uisprime(N))
    5273             :   {
    5274         392 :     lim = 2 * MF_get_dim(mf); /* ensure mfstabiter's first kernel ~ square */
    5275         392 :     p = 2;
    5276             :   }
    5277             :   else
    5278             :   {
    5279             :     forprime_t S;
    5280        1442 :     u_forprime_init(&S, 2, N);
    5281        2576 :     while ((p = u_forprime_next(&S)))
    5282        2576 :       if (N % p) break;
    5283        1442 :     lim = B + 1;
    5284             :   }
    5285        1834 :   LIM = (N & (N - 1))? 2 * lim: 3 * lim; /* N power of 2 ? */
    5286        1834 :   L = mkvecsmall4(lim, LIM, mfsturmNk(N,1), p);
    5287        1834 :   M = bhnmat_extend_nocache(M, N, LIM-1, 1, MF_get_S(mf));
    5288        1834 :   if (DEBUGLEVEL) timer_printf(&tt, "mf1basis [pre]: bnfmat_extend");
    5289        1834 :   v = pindices(p, LIM);
    5290        1834 :   if (!LIM) return mkvec4(L, mf, M, v);
    5291        1834 :   I = RgM_Rg_div(ZM_mul(rowslice(M, B+2, LIM), gel(Minv,1)), den);
    5292        1834 :   I = Q_remove_denom(I, &den);
    5293        1834 :   if (DEBUGLEVEL) timer_printf(&tt, "mf1basis [prec]: Iden");
    5294        1834 :   return mkvec5(L, mf, M, v, mkvec2(I, den));
    5295             : }
    5296             : 
    5297             : /* lg(A) > 1, E a t_POL */
    5298             : static GEN
    5299         700 : mfmatsermul(GEN A, GEN E)
    5300             : {
    5301         700 :   long j, l = lg(A), r = nbrows(A);
    5302         700 :   GEN M = cgetg(l, t_MAT);
    5303         700 :   E = RgXn_red_shallow(E, r+1);
    5304        6328 :   for (j = 1; j < l; j++)
    5305             :   {
    5306        5628 :     GEN c = RgV_to_RgX(gel(A,j), 0);
    5307        5628 :     gel(M, j) = RgX_to_RgC(RgXn_mul(c, E, r+1), r);
    5308             :   }
    5309         700 :   return M;
    5310             : }
    5311             : /* lg(Ap) > 1, Ep an Flxn */
    5312             : static GEN
    5313        1141 : mfmatsermul_Fl(GEN Ap, GEN Ep, ulong p)
    5314             : {
    5315        1141 :   long j, l = lg(Ap), r = nbrows(Ap);
    5316        1141 :   GEN M = cgetg(l, t_MAT);
    5317       42630 :   for (j = 1; j < l; j++)
    5318             :   {
    5319       41489 :     GEN c = Flv_to_Flx(gel(Ap,j), 0);
    5320       41489 :     gel(M,j) = Flx_to_Flv(Flxn_mul(c, Ep, r+1, p), r);
    5321             :   }
    5322        1141 :   return M;
    5323             : }
    5324             : 
    5325             : /* CHI mod F | N, return mfchar of modulus N.
    5326             :  * FIXME: wasteful, G should be precomputed  */
    5327             : static GEN
    5328       13048 : mfcharinduce(GEN CHI, long N)
    5329             : {
    5330             :   GEN G, chi;
    5331       13048 :   if (mfcharmodulus(CHI) == N) return CHI;
    5332        1463 :   G = znstar0(utoipos(N), 1);
    5333        1463 :   chi = zncharinduce(gel(CHI,1), gel(CHI,2), G);
    5334        1463 :   CHI = leafcopy(CHI);
    5335        1463 :   gel(CHI,1) = G;
    5336        1463 :   gel(CHI,2) = chi; return CHI;
    5337             : }
    5338             : 
    5339             : static GEN
    5340        3983 : gmfcharno(GEN CHI)
    5341             : {
    5342        3983 :   GEN G = gel(CHI,1), chi = gel(CHI,2);
    5343        3983 :   return mkintmod(znconreyexp(G, chi), znstar_get_N(G));
    5344             : }
    5345             : static long
    5346       13699 : mfcharno(GEN CHI)
    5347             : {
    5348       13699 :   GEN n = znconreyexp(gel(CHI,1), gel(CHI,2));
    5349       13699 :   return itou(n);
    5350             : }
    5351             : 
    5352             : /* return k such that minimal mfcharacter in Galois orbit of CHI is CHI^k */
    5353             : static long
    5354       12138 : mfconreyminimize(GEN CHI)
    5355             : {
    5356       12138 :   GEN G = gel(CHI,1), cyc, chi;
    5357       12138 :   cyc = ZV_to_zv(znstar_get_cyc(G));
    5358       12138 :   chi = ZV_to_zv(znconreychar(G, gel(CHI,2)));
    5359       12138 :   return zv_cyc_minimize(cyc, chi, coprimes_zv(mfcharorder(CHI)));
    5360             : }
    5361             : 
    5362             : /* find scalar c such that first nonzero entry of c*v is 1; return c*v */
    5363             : static GEN
    5364        2065 : RgV_normalize(GEN v, GEN *pc)
    5365             : {
    5366        2065 :   long i, l = lg(v);
    5367        5313 :   for (i = 1; i < l; i++)
    5368             :   {
    5369        5313 :     GEN c = gel(v,i);
    5370        5313 :     if (!gequal0(c))
    5371             :     {
    5372        2065 :       if (gequal1(c)) break;
    5373         679 :       *pc = ginv(c); return RgV_Rg_mul(v, *pc);
    5374             :     }
    5375             :   }
    5376        1386 :   *pc = gen_1; return v;
    5377             : }
    5378             : /* pS != NULL; dim > 0 */
    5379             : static GEN
    5380         784 : mftreatdihedral(long N, GEN DIH, GEN POLCYC, long ordchi, GEN *pS)
    5381             : {
    5382         784 :   long l = lg(DIH), lim = mfsturmNk(N, 1), i;
    5383         784 :   GEN Minv, C = cgetg(l, t_VEC), M = cgetg(l, t_MAT);
    5384        2436 :   for (i = 1; i < l; i++)
    5385             :   {
    5386        1652 :     GEN c, v = mfcoefs_i(gel(DIH,i), lim, 1);
    5387        1652 :     gel(M,i) = RgV_normalize(v, &c);
    5388        1652 :     gel(C,i) = Rg_col_ei(c, l-1, i);
    5389             :   }
    5390         784 :   Minv = gel(mfclean(M,POLCYC,ordchi,0),2);
    5391         784 :   M = RgM_Minv_mul(M, Minv);
    5392         784 :   C = RgM_Minv_mul(C, Minv);
    5393         784 :   *pS = vecmflinear(DIH, C); return M;
    5394             : }
    5395             : 
    5396             : /* same mode a maximal ideal above q */
    5397             : static GEN
    5398        2408 : Tpmod(GEN Ap, GEN A, ulong chip, long p, ulong q)
    5399             : {
    5400        2408 :   GEN B = leafcopy(Ap);
    5401        2408 :   long i, ip, l = lg(B);
    5402       86345 :   for (i = 1, ip = p; ip < l; i++, ip += p)
    5403       83937 :     B[ip] = Fl_add(B[ip], Fl_mul(A[i], chip, q), q);
    5404        2408 :   return B;
    5405             : }
    5406             : /* Tp(f_1), ..., Tp(f_d) mod q */
    5407             : static GEN
    5408         301 : matTpmod(GEN xp, GEN x, ulong chip, long p, ulong q)
    5409        2709 : { pari_APPLY_same(Tpmod(gel(xp,i), gel(x,i), chip, p, q)); }
    5410             : 
    5411             : /* Ap[i] = a_{p*i}(F), A[i] = a_i(F), i = 1..lim
    5412             :  * Tp(f)[n] = a_{p*n}(f) + chi(p) a_{n/p}(f) * 1_{p | n} */
    5413             : static GEN
    5414         469 : Tp(GEN Ap, GEN A, GEN chip, long p)
    5415             : {
    5416         469 :   GEN B = leafcopy(Ap);
    5417         469 :   long i, ip, l = lg(B);
    5418       12915 :   for (i = 1, ip = p; ip < l; i++, ip += p)
    5419       12446 :     gel(B,ip) = gadd(gel(B,ip), gmul(gel(A,i), chip));
    5420         469 :   return B;
    5421             : }
    5422             : /* Tp(f_1), ..., Tp(f_d) */
    5423             : static GEN
    5424          56 : matTp(GEN xp, GEN x, GEN chip, long p)
    5425         525 : { pari_APPLY_same(Tp(gel(xp,i), gel(x,i), chip, p)); }
    5426             : 
    5427             : static GEN
    5428         378 : _RgXQM_mul(GEN x, GEN y, GEN T)
    5429         378 : { return T? RgXQM_mul(x, y, T): RgM_mul(x, y); }
    5430             : /* largest T-stable Q(CHI)-subspace of Q(CHI)-vector space spanned by columns
    5431             :  * of A */
    5432             : static GEN
    5433          28 : mfstabiter(GEN *pC, GEN A0, GEN chip, GEN TMP, GEN P, long ordchi)
    5434             : {
    5435          28 :   GEN A, Ap, vp = gel(TMP,4), C = NULL;
    5436          28 :   long i, lA, lim1 = gel(TMP,1)[3], p = gel(TMP,1)[4];
    5437             :   pari_timer tt;
    5438             : 
    5439          28 :   Ap = rowpermute(A0, vp);
    5440          28 :   A = rowslice(A0, 2, nbrows(Ap)+1); /* remove a0 */
    5441             :   for(;;)
    5442          28 :   {
    5443          56 :     GEN R = shallowconcat(matTp(Ap, A, chip, p), A);
    5444          56 :     GEN B = QabM_ker(R, P, ordchi);
    5445          56 :     long lB = lg(B);
    5446          56 :     if (DEBUGLEVEL)
    5447           0 :       timer_printf(&tt, "mf1basis: Hecke intersection (dim %ld)", lB-1);
    5448          56 :     if (lB == 1) return NULL;
    5449          56 :     lA = lg(A); if (lB == lA) break;
    5450          28 :     B = rowslice(B, 1, lA-1);
    5451          28 :     Ap = _RgXQM_mul(Ap, B, P);
    5452          28 :     A = _RgXQM_mul(A, B, P);
    5453          28 :     C = C? _RgXQM_mul(C, B, P): B;
    5454             :   }
    5455          28 :   if (nbrows(A) < lim1)
    5456             :   {
    5457          14 :     A0 = rowslice(A0, 2, lim1);
    5458          14 :     A = C? _RgXQM_mul(A0, C, P): A0;
    5459             :   }
    5460             :   else /* all needed coefs computed */
    5461          14 :     A = rowslice(A, 1, lim1-1);
    5462          28 :   if (*pC) C = C? _RgXQM_mul(*pC, C, P): *pC;
    5463             :   /* put back a0 */
    5464         119 :   for (i = 1; i < lA; i++) gel(A,i) = vec_prepend(gel(A,i), gen_0);
    5465          28 :   *pC = C; return A;
    5466             : }
    5467             : 
    5468             : static long
    5469         252 : mfstabitermod(GEN A, GEN vp, ulong chip, long p, ulong q)
    5470             : {
    5471         252 :   GEN Ap, C = NULL;
    5472         252 :   Ap = rowpermute(A, vp);
    5473         252 :   A = rowslice(A, 2, nbrows(Ap)+1);
    5474             :   while (1)
    5475          49 :   {
    5476         301 :     GEN Rp = shallowconcat(matTpmod(Ap, A, chip, p, q), A);
    5477         301 :     GEN B = Flm_ker(Rp, q);
    5478         301 :     long lA = lg(A), lB = lg(B);
    5479         301 :     if (lB == 1) return 0;
    5480         266 :     if (lB == lA) return lA-1;
    5481          49 :     B = rowslice(B, 1, lA-1);
    5482          49 :     Ap = Flm_mul(Ap, B, q);
    5483          49 :     A = Flm_mul(A, B, q);
    5484          49 :     C = C? Flm_mul(C, B, q): B;
    5485             :   }
    5486             : }
    5487             : 
    5488             : static GEN
    5489         595 : mfcharinv_i(GEN CHI)
    5490             : {
    5491         595 :   GEN G = gel(CHI,1);
    5492         595 :   CHI = leafcopy(CHI); gel(CHI,2) =  zncharconj(G, gel(CHI,2)); return CHI;
    5493             : }
    5494             : 
    5495             : /* upper bound dim S_1(Gamma_0(N),chi) performing the linear algebra mod p */
    5496             : static long
    5497         595 : mf1dimmod(GEN E1, GEN E, GEN chip, long ordchi, long dih, GEN TMP)
    5498             : {
    5499         595 :   GEN E1i, A, vp, mf, C = NULL;
    5500         595 :   ulong q, r = QabM_init(ordchi, &q);
    5501             :   long lim, LIM, p;
    5502             : 
    5503         595 :   LIM = gel(TMP,1)[2]; lim = gel(TMP,1)[1];
    5504         595 :   mf= gel(TMP,2);
    5505         595 :   A = gel(TMP,3);
    5506         595 :   A = QabM_to_Flm(A, r, q);
    5507         595 :   E1 = QabX_to_Flx(E1, r, q);
    5508         595 :   E1i = Flxn_inv(E1, nbrows(A), q);
    5509         595 :   if (E)
    5510             :   {
    5511         574 :     GEN Iden = gel(TMP,5), I = gel(Iden,1), den = gel(Iden,2);
    5512         574 :     GEN Mindex = MF_get_Mindex(mf), F = rowslice(A, 1, LIM);
    5513         574 :     GEN E1ip = Flxn_red(E1i, LIM);
    5514         574 :     ulong d = den? umodiu(den, q): 1;
    5515         574 :     long i, nE = lg(E) - 1;
    5516             :     pari_sp av;
    5517             : 
    5518         574 :     I = ZM_to_Flm(I, q);
    5519         574 :     if (d != 1) I = Flm_Fl_mul(I, Fl_inv(d, q), q);
    5520         574 :     av = avma;
    5521        1120 :     for (i = 1; i <= nE; i++)
    5522             :     {
    5523         889 :       GEN e = Flxn_mul(E1ip, QabX_to_Flx(gel(E,i), r, q), LIM, q);
    5524         889 :       GEN B = mfmatsermul_Fl(F, e, q), z;
    5525         889 :       GEN B2 = Flm_mul(I, rowpermute(B, Mindex), q);
    5526         889 :       B = rowslice(B, lim+1,LIM);
    5527         889 :       z = Flm_ker(Flm_sub(B2, B, q), q);
    5528         889 :       if (lg(z)-1 == dih) return dih;
    5529         546 :       C = C? Flm_mul(C, z, q): z;
    5530         546 :       F = Flm_mul(F, z, q);
    5531         546 :       (void)gc_all(av, 2, &F,&C);
    5532             :     }
    5533         231 :     A = F;
    5534             :   }
    5535             :   /* use Schaeffer */
    5536         252 :   p = gel(TMP,1)[4]; vp = gel(TMP,4);
    5537         252 :   A = mfmatsermul_Fl(A, E1i, q);
    5538         252 :   return mfstabitermod(A, vp, Qab_to_Fl(chip, r, q), p, q);
    5539             : }
    5540             : 
    5541             : static GEN
    5542         224 : mf1intermat(GEN A, GEN Mindex, GEN e, GEN Iden, long lim, GEN POLCYC)
    5543             : {
    5544         224 :   long j, l = lg(A), LIM = nbrows(A);
    5545         224 :   GEN I = gel(Iden,1), den = gel(Iden,2), B = cgetg(l, t_MAT);
    5546             : 
    5547        5257 :   for (j = 1; j < l; j++)
    5548             :   {
    5549        5033 :     pari_sp av = avma;
    5550        5033 :     GEN c = RgV_to_RgX(gel(A,j), 0), c1, c2;
    5551        5033 :     c = RgX_to_RgC(RgXn_mul(c, e, LIM), LIM);
    5552        5033 :     if (POLCYC) c = liftpol_shallow(c);
    5553        5033 :     c1 = vecslice(c, lim+1, LIM);
    5554        5033 :     if (den) c1 = RgC_Rg_mul(c1, den);
    5555        5033 :     c2 = RgM_RgC_mul(I, vecpermute(c, Mindex));
    5556        5033 :     gel(B, j) = gc_upto(av, RgC_sub(c2, c1));
    5557             :   }
    5558         224 :   return B;
    5559             : }
    5560             : /* Compute the full S_1(\G_0(N),\chi); return NULL if space is empty; else
    5561             :  * if pS is NULL, return stoi(dim), where dim is the dimension; else *pS is
    5562             :  * set to a vector of forms making up a basis, and return the matrix of their
    5563             :  * Fourier expansions. pdih gives the dimension of the subspace generated by
    5564             :  * dihedral forms; TMP is from mf1_pre or NULL. */
    5565             : static GEN
    5566       11284 : mf1basis(long N, GEN CHI, GEN TMP, GEN vSP, GEN *pS, long *pdih)
    5567             : {
    5568       11284 :   GEN E = NULL, EB, E1, E1i, dE1i, mf, A, C, POLCYC, DIH, Minv, chip;
    5569       11284 :   long nE = 0, p, LIM, lim, lim1, i, lA, dimp, ordchi, dih;
    5570             :   pari_timer tt;
    5571             :   pari_sp av;
    5572             : 
    5573       11284 :   if (pdih) *pdih = 0;
    5574       11284 :   if (pS) *pS = NULL;
    5575       11284 :   if (wt1empty(N) || mfcharparity(CHI) != -1) return NULL;
    5576       10990 :   ordchi = mfcharorder(CHI);
    5577       10990 :   if (uisprime(N) && ordchi > 4) return NULL;
    5578       10962 :   if (pS)
    5579             :   {
    5580        3857 :     DIH = mfdihedralcusp(N, CHI, vSP);
    5581        3857 :     dih = lg(DIH) - 1;
    5582             :   }
    5583             :   else
    5584             :   {
    5585        7105 :     DIH = NULL;
    5586        7105 :     dih = mfdihedralcuspdim(N, CHI, vSP);
    5587             :   }
    5588       10962 :   POLCYC = (ordchi <= 2)? NULL: mfcharpol(CHI);
    5589       10962 :   if (pdih) *pdih = dih;
    5590       10962 :   if (N <= 600) switch(N)
    5591             :   {
    5592             :     long m;
    5593         126 :     case 219: case 273: case 283: case 331: case 333: case 344: case 416:
    5594             :     case 438: case 468: case 491: case 504: case 546: case 553: case 563:
    5595             :     case 566: case 581: case 592:
    5596         126 :       break; /* one chi with both exotic and dihedral forms */
    5597        9499 :     default: /* only dihedral forms */
    5598        9499 :       if (!dih) return NULL;
    5599             :       /* fall through */
    5600             :     case 124: case 133: case 148: case 171: case 201: case 209: case 224:
    5601             :     case 229: case 248: case 261: case 266: case 288: case 296: case 301:
    5602             :     case 309: case 325: case 342: case 371: case 372: case 380: case 399:
    5603             :     case 402: case 403: case 404: case 408: case 418: case 432: case 444:
    5604             :     case 448: case 451: case 453: case 458: case 496: case 497: case 513:
    5605             :     case 522: case 527: case 532: case 576: case 579:
    5606             :       /* no chi with both exotic and dihedral; one chi with exotic forms */
    5607        3248 :       if (dih)
    5608             :       {
    5609        2338 :         if (!pS) return utoipos(dih);
    5610         728 :         return mftreatdihedral(N, DIH, POLCYC, ordchi, pS) ;
    5611             :       }
    5612         910 :       m = mfcharno(mfcharinduce(CHI,N));
    5613         910 :       if (N == 124 && (m != 67 && m != 87)) return NULL;
    5614         784 :       if (N == 133 && (m != 83 && m !=125)) return NULL;
    5615         490 :       if (N == 148 && (m !=105 && m !=117)) return NULL;
    5616         364 :       if (N == 171 && (m != 94 && m !=151)) return NULL;
    5617         364 :       if (N == 201 && (m != 29 && m !=104)) return NULL;
    5618         364 :       if (N == 209 && (m != 87 && m !=197)) return NULL;
    5619         364 :       if (N == 224 && (m != 95 && m !=191)) return NULL;
    5620         364 :       if (N == 229 && (m !=107 && m !=122)) return NULL;
    5621         364 :       if (N == 248 && (m != 87 && m !=191)) return NULL;
    5622         273 :       if (N == 261 && (m != 46 && m !=244)) return NULL;
    5623         273 :       if (N == 266 && (m != 83 && m !=125)) return NULL;
    5624         273 :       if (N == 288 && (m != 31 && m !=223)) return NULL;
    5625         273 :       if (N == 296 && (m !=105 && m !=265)) return NULL;
    5626             :   }
    5627         595 :   if (DEBUGLEVEL)
    5628           0 :     err_printf("mf1basis: start character %Ps, conductor = %ld, order = %ld\n",
    5629             :                gmfcharno(CHI), mfcharconductor(CHI), ordchi);
    5630         595 :   if (!TMP) TMP = mf1_pre(N);
    5631         595 :   lim = gel(TMP,1)[1]; LIM = gel(TMP,1)[2]; lim1 = gel(TMP,1)[3];
    5632         595 :   p = gel(TMP,1)[4];
    5633         595 :   mf  = gel(TMP,2);
    5634         595 :   A   = gel(TMP,3);
    5635         595 :   EB = mfeisensteinbasis(N, 1, mfcharinv_i(CHI));
    5636         595 :   nE = lg(EB) - 1;
    5637         595 :   E1 = RgV_to_RgX(mftocol(gel(EB,1), LIM-1, 1), 0); /* + O(x^LIM) */
    5638         595 :   if (--nE)
    5639         574 :     E = RgM_to_RgXV(mfvectomat(vecslice(EB, 2, nE+1), LIM-1, 1), 0);
    5640         595 :   chip = mfchareval(CHI, p); /* != 0 */
    5641         595 :   if (DEBUGLEVEL) timer_start(&tt);
    5642         595 :   av = avma; dimp = mf1dimmod(E1, E, chip, ordchi, dih, TMP);
    5643         595 :   set_avma(av);
    5644         595 :   if (DEBUGLEVEL) timer_printf(&tt, "mf1basis: dim mod p is %ld", dimp);
    5645         595 :   if (!dimp) return NULL;
    5646         280 :   if (!pS) return utoi(dimp);
    5647         224 :   if (dimp == dih) return mftreatdihedral(N, DIH, POLCYC, ordchi, pS);
    5648         168 :   E1i = RgXn_inv(E1, LIM); /* E[1] does not vanish at oo */
    5649         168 :   if (POLCYC) E1i = liftpol_shallow(E1i);
    5650         168 :   E1i = Q_remove_denom(E1i, &dE1i);
    5651         168 :   if (DEBUGLEVEL)
    5652             :   {
    5653           0 :     GEN a0 = gel(E1,2);
    5654           0 :     if (typ(a0) == t_POLMOD) a0 = gnorm(a0);
    5655           0 :     a0 = Q_abs_shallow(a0);
    5656           0 :     timer_printf(&tt, "mf1basis: invert E; norm(a0(E)) = %Ps", a0);
    5657             :   }
    5658         168 :   C = NULL;
    5659         168 :   if (nE)
    5660             :   { /* mf attached to S2(N), fi = mfbasis(mf)
    5661             :      * M = coefs(f1,...,fd) up to LIM
    5662             :      * F = coefs(F1,...,FD) = M * C, for some matrix C over Q(chi),
    5663             :      * initially 1, eventually giving \cap_E S2 / E; D <= d.
    5664             :      * B = coefs(E/E1 F1, .., E/E1 FD); we want X in Q(CHI)^d and
    5665             :      * Y in Q(CHI)^D such that
    5666             :      *   B * X = M * Y, i.e. Minv * rowpermute(B, Mindex * X) = Y
    5667             :      *(B  - I * rowpermute(B, Mindex)) * X = 0.
    5668             :      * where I = M * Minv. Rows of (B - I * ...) are 0 up to lim so
    5669             :      * are not included */
    5670         154 :     GEN Mindex = MF_get_Mindex(mf), Iden  = gel(TMP,5);
    5671             :     pari_timer TT;
    5672         154 :     pari_sp av = avma;
    5673         154 :     if (DEBUGLEVEL) timer_start(&TT);
    5674         238 :     for (i = 1; i <= nE; i++)
    5675             :     {
    5676         224 :       pari_sp av2 = avma;
    5677             :       GEN e, z, B;
    5678             : 
    5679         224 :       e = Q_primpart(RgXn_mul(E1i, gel(E,i), LIM));
    5680         224 :       if (DEBUGLEVEL) timer_printf(&TT, "mf1basis: E[%ld] / E[1]", i+1);
    5681             :       /* the first time A is over Z and it is more efficient to lift than
    5682             :          * to let RgXn_mul use Kronecker's trick */
    5683         224 :       if (POLCYC && i == 1) e = liftpol_shallow(e);
    5684         224 :       B = mf1intermat(A, Mindex, e, Iden, lim, i == 1? NULL: POLCYC);
    5685         224 :       if (DEBUGLEVEL) timer_printf(&TT, "mf1basis: ... intermat");
    5686         224 :       z = gc_upto(av2, QabM_ker(B, POLCYC, ordchi));
    5687         224 :       if (DEBUGLEVEL)
    5688           0 :         timer_printf(&TT, "mf1basis: ... kernel (dim %ld)",lg(z)-1);
    5689         224 :       if (lg(z) == 1) return NULL;
    5690         224 :       if (lg(z) == lg(A)) { set_avma(av2); continue; } /* no progress */
    5691         224 :       C = C? _RgXQM_mul(C, z, POLCYC): z;
    5692         224 :       A = _RgXQM_mul(A, z, POLCYC);
    5693         224 :       if (DEBUGLEVEL) timer_printf(&TT, "mf1basis: ... updates");
    5694         224 :       if (lg(z)-1 == dimp) break;
    5695          84 :       if (gc_needed(av, 1))
    5696             :       {
    5697           0 :         if (DEBUGMEM > 1) pari_warn(warnmem,"mf1basis i = %ld", i);
    5698           0 :         (void)gc_all(av, 2, &A, &C);
    5699             :       }
    5700             :     }
    5701         154 :     if (DEBUGLEVEL) timer_printf(&tt, "mf1basis: intersection [total]");
    5702             :   }
    5703         168 :   lA = lg(A);
    5704         168 :   if (lA-1 == dimp)
    5705             :   {
    5706         140 :     A = mfmatsermul(rowslice(A, 1, lim1), E1i);
    5707         140 :     if (POLCYC) A = RgXQM_red(A, POLCYC);
    5708         140 :     if (DEBUGLEVEL) timer_printf(&tt, "mf1basis: matsermul [1]");
    5709             :   }
    5710             :   else
    5711             :   {
    5712          28 :     A = mfmatsermul(A, E1i);
    5713          28 :     if (POLCYC) A = RgXQM_red(A, POLCYC);
    5714          28 :     if (DEBUGLEVEL) timer_printf(&tt, "mf1basis: matsermul [2]");
    5715          28 :     A = mfstabiter(&C, A, chip, TMP, POLCYC, ordchi);
    5716          28 :     if (DEBUGLEVEL) timer_printf(&tt, "mf1basis: Hecke stability");
    5717          28 :     if (!A) return NULL;
    5718             :   }
    5719         168 :   if (dE1i) C = RgM_Rg_mul(C, dE1i);
    5720         168 :   if (POLCYC)
    5721             :   {
    5722         147 :     A = QXQM_to_mod_shallow(A, POLCYC);
    5723         147 :     C = QXQM_to_mod_shallow(C, POLCYC);
    5724             :   }
    5725         168 :   lA = lg(A);
    5726         581 :   for (i = 1; i < lA; i++)
    5727             :   {
    5728         413 :     GEN c, v = gel(A,i);
    5729         413 :     gel(A,i) = RgV_normalize(v, &c);
    5730         413 :     gel(C,i) = RgC_Rg_mul(gel(C,i), c);
    5731             :   }
    5732         168 :   Minv = gel(mfclean(A, POLCYC, ordchi, 0), 2);
    5733         168 :   A = RgM_Minv_mul(A, Minv);
    5734         168 :   C = RgM_Minv_mul(C, Minv);
    5735         168 :   *pS = vecmflineardiv0(MF_get_S(mf), C, gel(EB,1));
    5736         168 :   return A;
    5737             : }
    5738             : 
    5739             : static void
    5740         413 : MF_set_space(GEN mf, long x) { gmael(mf,1,4) = utoi(x); }
    5741             : static GEN
    5742         252 : mf1_cusptonew(GEN mf, GEN vSP)
    5743             : {
    5744         252 :   const long vy = 1;
    5745             :   long i, lP, dSnew, ct;
    5746         252 :   GEN vP, F, S, Snew, vF, v = split_ii(mf, 0, 0, vSP, &i);
    5747             : 
    5748         252 :   F = gel(v,1);
    5749         252 :   vP= gel(v,2); lP = lg(vP);
    5750         252 :   if (lP == 1) { obj_insert(mf, MF_SPLIT, v); return NULL; }
    5751         238 :   MF_set_space(mf, mf_NEW);
    5752         238 :   S = MF_get_S(mf);
    5753         238 :   dSnew = dim_sum(v);
    5754         238 :   Snew = cgetg(dSnew + 1, t_VEC); ct = 0;
    5755         238 :   vF = cgetg(lP, t_MAT);
    5756         546 :   for (i = 1; i < lP; i++)
    5757             :   {
    5758         308 :     GEN V, P = gel(vP,i), f = liftpol_shallow(gel(F,i));
    5759         308 :     long j, d = degpol(P);
    5760         308 :     gel(vF,i) = V = zerocol(dSnew);
    5761         308 :     if (d == 1)
    5762             :     {
    5763         140 :       gel(Snew, ct+1) = mflineardiv_linear(S, f, 0);
    5764         140 :       gel(V, ct+1) = gen_1;
    5765             :     }
    5766             :     else
    5767             :     {
    5768         168 :       f = RgXV_to_RgM(f,d);
    5769         511 :       for (j = 1; j <= d; j++)
    5770             :       {
    5771         343 :         gel(Snew, ct+j) = mflineardiv_linear(S, row(f,j), 0);
    5772         343 :         gel(V, ct+j) = mkpolmod(pol_xn(j-1,vy), P);
    5773             :       }
    5774             :     }
    5775         308 :     ct += d;
    5776             :   }
    5777         238 :   obj_insert(mf, MF_SPLIT, mkvec2(vF, vP));
    5778         238 :   gel(mf,3) = Snew; return mf;
    5779             : }
    5780             : static GEN
    5781        3969 : mf1init(long N, GEN CHI, GEN TMP, GEN vSP, long space, long flraw)
    5782             : {
    5783        3969 :   GEN mf, mf1, S, M = mf1basis(N, CHI, TMP, vSP, &S, NULL);
    5784        3969 :   if (!M) return NULL;
    5785         952 :   mf1 = mkvec4(stoi(N), gen_1, CHI, utoi(mf_CUSP));
    5786         952 :   mf = mkmf(mf1, cgetg(1,t_VEC), S, gen_0, NULL);
    5787         952 :   if (space == mf_NEW)
    5788             :   {
    5789         252 :     gel(mf,5) = mfcleanCHI(M,CHI, 0);
    5790         252 :     mf = mf1_cusptonew(mf, vSP); if (!mf) return NULL;
    5791         238 :     if (!flraw) M = mfcoefs_mf(mf, mfsturmNk(N,1)+1, 1);
    5792             :   }
    5793         938 :   gel(mf,5) = flraw? zerovec(3): mfcleanCHI(M, CHI, 0);
    5794         938 :   return mf;
    5795             : }
    5796             : 
    5797             : static GEN
    5798        1029 : mfEMPTY(GEN mf1)
    5799             : {
    5800        1029 :   GEN Minv = mkMinv(cgetg(1,t_MAT), NULL,NULL,NULL);
    5801        1029 :   GEN M = mkvec3(cgetg(1,t_VECSMALL), Minv, cgetg(1,t_MAT));
    5802        1029 :   return mkmf(mf1, cgetg(1,t_VEC), cgetg(1,t_VEC), cgetg(1,t_VEC), M);
    5803             : }
    5804             : static GEN
    5805         616 : mfEMPTYall(long N, GEN gk, GEN vCHI, long space)
    5806             : {
    5807             :   long i, l;
    5808             :   GEN v, gN, gs;
    5809         616 :   if (!vCHI) return cgetg(1, t_VEC);
    5810          14 :   gN = utoipos(N); gs = utoi(space);
    5811          14 :   l = lg(vCHI); v = cgetg(l, t_VEC);
    5812          42 :   for (i = 1; i < l; i++) gel(v,i) = mfEMPTY(mkvec4(gN,gk,gel(vCHI,i),gs));
    5813          14 :   return v;
    5814             : }
    5815             : 
    5816             : static GEN
    5817        3983 : fmt_dim(GEN CHI, long d, long dih)
    5818        3983 : { return mkvec4(gmfcharorder(CHI), gmfcharno(CHI), utoi(d), stoi(dih)); }
    5819             : /* merge two vector of fmt_dim's for the same vector of characters. If CHI
    5820             :  * is not NULL, remove dim-0 spaces and add character from CHI */
    5821             : static GEN
    5822           7 : merge_dims(GEN V, GEN W, GEN CHI)
    5823             : {
    5824           7 :   long i, j, id, l = lg(V);
    5825           7 :   GEN A = cgetg(l, t_VEC);
    5826           7 :   if (l == 1) return A;
    5827           7 :   id = CHI? 1: 3;
    5828          21 :   for (i = j = 1; i < l; i++)
    5829             :   {
    5830          14 :     GEN v = gel(V,i), w = gel(W,i);
    5831          14 :     long dv = itou(gel(v,id)), dvh = itou(gel(v,id+1)), d;
    5832          14 :     long dw = itou(gel(w,id)), dwh = itou(gel(w,id+1));
    5833          14 :     d = dv + dw;
    5834          14 :     if (d || CHI)
    5835          14 :       gel(A,j++) = CHI? fmt_dim(gel(CHI,i),d, dvh+dwh)
    5836          14 :                       : mkvec2s(d,dvh+dwh);
    5837             :   }
    5838           7 :   setlg(A, j); return A;
    5839             : }
    5840             : static GEN
    5841        3010 : mfdim0all(GEN w)
    5842             : {
    5843        3038 :   if (w) retconst_vec(lg(w)-1, zerovec(2));
    5844        3003 :   return cgetg(1,t_VEC);
    5845             : }
    5846             : static long
    5847        7315 : mf1cuspdim_i(long N, GEN CHI, GEN TMP, GEN vSP, long *dih)
    5848             : {
    5849        7315 :   pari_sp av = avma;
    5850        7315 :   GEN b = mf1basis(N, CHI, TMP, vSP, NULL, dih);
    5851        7315 :   return gc_long(av, b? itou(b): 0);
    5852             : }
    5853             : 
    5854             : static long
    5855         476 : mf1cuspdim(long N, GEN CHI, GEN vSP)
    5856             : {
    5857         476 :   if (!vSP) vSP = get_vDIH(N, divisorsNF(N, mfcharconductor(CHI)));
    5858         476 :   return mf1cuspdim_i(N, CHI, NULL, vSP, NULL);
    5859             : }
    5860             : static GEN
    5861        4144 : mf1cuspdimall(long N, GEN vCHI)
    5862             : {
    5863             :   GEN z, TMP, w, vSP;
    5864             :   long i, j, l;
    5865        4144 :   if (wt1empty(N)) return mfdim0all(vCHI);
    5866        1141 :   w = mf1chars(N,vCHI);
    5867        1141 :   l = lg(w); if (l == 1) return cgetg(1,t_VEC);
    5868        1141 :   z = cgetg(l, t_VEC);
    5869        1141 :   TMP = mf1_pre(N); vSP = get_vDIH(N, NULL);
    5870        7861 :   for (i = j = 1; i < l; i++)
    5871             :   {
    5872        6720 :     GEN CHI = gel(w,i);
    5873        6720 :     long dih, d = mf1cuspdim_i(N, CHI, TMP, vSP, &dih);
    5874        6720 :     if (vCHI)
    5875          42 :       gel(z,j++) = mkvec2s(d, dih);
    5876        6678 :     else if (d)
    5877        1428 :       gel(z,j++) = fmt_dim(CHI, d, dih);
    5878             :   }
    5879        1141 :   setlg(z,j); return z;
    5880             : }
    5881             : 
    5882             : /* dimension of S_1(Gamma_1(N)) */
    5883             : static long
    5884        4123 : mf1cuspdimsum(long N)
    5885             : {
    5886        4123 :   pari_sp av = avma;
    5887        4123 :   GEN v = mf1cuspdimall(N, NULL);
    5888        4123 :   long i, ct = 0, l = lg(v);
    5889        5544 :   for (i = 1; i < l; i++)
    5890             :   {
    5891        1421 :     GEN w = gel(v,i); /* [ord(CHI),*,dim,*] */
    5892        1421 :     ct += itou(gel(w,3))*myeulerphiu(itou(gel(w,1)));
    5893             :   }
    5894        4123 :   return gc_long(av,ct);
    5895             : }
    5896             : 
    5897             : static GEN
    5898          56 : mf1newdimall(long N, GEN vCHI)
    5899             : {
    5900             :   GEN z, w, vTMP, vSP, fa, P, E;
    5901             :   long i, c, l, lw, P1;
    5902          56 :   if (wt1empty(N)) return mfdim0all(vCHI);
    5903          56 :   w = mf1chars(N,vCHI);
    5904          56 :   lw = lg(w); if (lw == 1) return cgetg(1,t_VEC);
    5905          56 :   vTMP = const_vec(N, NULL);
    5906          56 :   vSP = get_vDIH(N, NULL);
    5907          56 :   gel(vTMP,N) = mf1_pre(N);
    5908             :   /* if p || N and p \nmid F(CHI), S_1^new(G0(N),chi) = 0 */
    5909          56 :   fa = znstar_get_faN(gmael(w,1,1));
    5910          56 :   P = gel(fa,1); l = lg(P);
    5911          56 :   E = gel(fa,2);
    5912         154 :   for (i = P1 = 1; i < l; i++)
    5913          98 :     if (E[i] == 1) P1 *= itou(gel(P,i));
    5914             :   /* P1 = \prod_{v_p(N) = 1} p */
    5915          56 :   z = cgetg(lw, t_VEC);
    5916         182 :   for (i = c = 1; i < lw; i++)
    5917             :   {
    5918             :     long S, j, l, F, dihnew;
    5919         126 :     GEN D, CHI = gel(w,i), CHIP = mfchartoprimitive(CHI,&F);
    5920             : 
    5921         126 :     S = F % P1? 0: mf1cuspdim_i(N, CHI, gel(vTMP,N), vSP, &dihnew);
    5922         126 :     if (!S)
    5923             :     {
    5924          56 :       if (vCHI) gel(z, c++) = zerovec(2);
    5925          56 :       continue;
    5926             :     }
    5927          70 :     D = mydivisorsu(N/F); l = lg(D);
    5928          77 :     for (j = l-2; j > 0; j--) /* skip last M = N */
    5929             :     {
    5930           7 :       long M = D[j]*F, m, s, dih;
    5931           7 :       GEN TMP = gel(vTMP,M);
    5932           7 :       if (wt1empty(M) || !(m = mubeta(D[l-j]))) continue; /*m = mubeta(N/M)*/
    5933           7 :       if (!TMP) gel(vTMP,M) = TMP = mf1_pre(M);
    5934           7 :       s = mf1cuspdim_i(M, CHIP, TMP, vSP, &dih);
    5935           7 :       if (s) { S += m * s; dihnew += m * dih; }
    5936             :     }
    5937          70 :     if (vCHI)
    5938          63 :       gel(z,c++) = mkvec2s(S, dihnew);
    5939           7 :     else if (S)
    5940           7 :       gel(z, c++) = fmt_dim(CHI, S, dihnew);
    5941             :   }
    5942          56 :   setlg(z,c); return z;
    5943             : }
    5944             : 
    5945             : static GEN
    5946          28 : mf1olddimall(long N, GEN vCHI)
    5947             : {
    5948             :   long i, j, l;
    5949             :   GEN z, w;
    5950          28 :   if (wt1empty(N)) return mfdim0all(vCHI);
    5951          28 :   w = mf1chars(N,vCHI);
    5952          28 :   l = lg(w); z = cgetg(l, t_VEC);
    5953          84 :   for (i = j = 1; i < l; i++)
    5954             :   {
    5955          56 :     GEN CHI = gel(w,i);
    5956          56 :     long d = mfolddim(N, 1, CHI);
    5957          56 :     if (vCHI)
    5958          28 :       gel(z,j++) = mkvec2s(d,d?-1:0);
    5959          28 :     else if (d)
    5960           7 :       gel(z, j++) = fmt_dim(CHI, d, -1);
    5961             :   }
    5962          28 :   setlg(z,j); return z;
    5963             : }
    5964             : 
    5965             : static long
    5966         469 : mf1olddimsum(long N)
    5967             : {
    5968             :   GEN D;
    5969         469 :   long N2, i, l, S = 0;
    5970         469 :   newd_params(N, &N2); /* will ensure mubeta != 0 */
    5971         469 :   D = mydivisorsu(N/N2); l = lg(D);
    5972        2485 :   for (i = 2; i < l; i++)
    5973             :   {
    5974        2016 :     long M = D[l-i]*N2, d = mf1cuspdimsum(M);
    5975        2016 :     if (d) S -= mubeta(D[i]) * d;
    5976             :   }
    5977         469 :   return S;
    5978             : }
    5979             : static long
    5980        1050 : mf1newdimsum(long N)
    5981             : {
    5982        1050 :   long S = mf1cuspdimsum(N);
    5983        1050 :   return S? S - mf1olddimsum(N): 0;
    5984             : }
    5985             : 
    5986             : /* return the automorphism of a degree-2 nf */
    5987             : static GEN
    5988        5768 : nf2_get_conj(GEN nf)
    5989             : {
    5990        5768 :   GEN pol = nf_get_pol(nf);
    5991        5768 :   return deg1pol_shallow(gen_m1, negi(gel(pol,3)), varn(pol));
    5992             : }
    5993             : static int
    5994          42 : foo_stable(GEN foo)
    5995          42 : { return lg(foo) != 3 || equalii(gel(foo,1), gel(foo,2)); }
    5996             : 
    5997             : static long
    5998         224 : mfisdihedral(GEN vF, GEN DIH)
    5999             : {
    6000         224 :   GEN vG = gel(DIH,1), M = gel(DIH,2), v, G, bnr, w, gen, D, f, nf, tau;
    6001         224 :   GEN bnr0 = NULL, f0, f0b, xin, foo;
    6002             :   long i, l, e, j, L, n;
    6003         224 :   if (lg(M) == 1) return 0;
    6004          42 :   v = RgM_RgC_invimage(M, vF);
    6005          42 :   if (!v) return 0;
    6006          42 :   l = lg(v);
    6007          42 :   for (i = 1; i < l; i++)
    6008          42 :     if (!gequal0(gel(v,i))) break;
    6009          42 :   if (i == l) return 0;
    6010          42 :   G = gel(vG,i);
    6011          42 :   bnr = gel(G,2); D = cyc_get_expo(bnr_get_cyc(bnr));
    6012          42 :   w = gel(G,3);
    6013          42 :   f = bnr_get_mod(bnr);
    6014          42 :   nf = bnr_get_nf(bnr);
    6015          42 :   tau = nf2_get_conj(nf);
    6016          42 :   f0 = gel(f,1); foo = gel(f,2);
    6017          42 :   f0b = galoisapply(nf, tau, f0);
    6018          42 :   xin = zv_to_ZV(gel(w,2)); /* xi(bnr.gen[i]) = e(xin[i] / D) */
    6019          42 :   if (!foo_stable(foo)) { foo = mkvec2(gen_1, gen_1); bnr0 = bnr; }
    6020          42 :   if (!gequal(f0, f0b))
    6021             :   {
    6022          21 :     f0 = idealmul(nf, f0, idealdivexact(nf, f0b, idealadd(nf, f0, f0b)));
    6023          21 :     bnr0 = bnr;
    6024             :   }
    6025          42 :   if (bnr0)
    6026             :   { /* conductor not ambiguous */
    6027             :     GEN S;
    6028          28 :     bnr = Buchray(bnr_get_bnf(bnr), mkvec2(f0, foo), nf_INIT | nf_GEN);
    6029          28 :     S = bnrsurjection(bnr, bnr0);
    6030          28 :     xin = FpV_red(RgV_RgM_mul(xin, gel(S,1)), D);
    6031             :     /* still xi(gen[i]) = e(xin[i] / D), for the new generators; D stays
    6032             :      * the same, not exponent(bnr.cyc) ! */
    6033             :   }
    6034          42 :   gen = bnr_get_gen(bnr); L = lg(gen);
    6035          77 :   for (j = 1, e = itou(D); j < L; j++)
    6036             :   {
    6037          63 :     GEN Ng = idealnorm(nf, gel(gen,j));
    6038          63 :     GEN a = shifti(gel(xin,j), 1); /* xi(g_j^2) = e(a/D) */
    6039          63 :     GEN b = FpV_dotproduct(xin, isprincipalray(bnr,Ng), D);
    6040          63 :     GEN m = Fp_sub(a, b, D); /* xi(g_j/g_j^\tau) = e(m/D) */
    6041          63 :     e = ugcd(e, itou(m)); if (e == 1) break;
    6042             :   }
    6043          42 :   n = itou(D) / e;
    6044          42 :   return n == 1? 4: 2*n;
    6045             : }
    6046             : 
    6047             : static ulong
    6048         119 : myradicalu(ulong n) { return zv_prod(gel(myfactoru(n),1)); }
    6049             : 
    6050             : /* list of fundamental discriminants unramified outside N, with sign s
    6051             :  * [s = 0 => no sign condition] */
    6052             : static GEN
    6053         119 : mfunram(long N, long s)
    6054             : {
    6055         119 :   long cN = myradicalu(N >> vals(N)), p = 1, m = 1, l, c, i;
    6056         119 :   GEN D = mydivisorsu(cN), res;
    6057         119 :   l = lg(D);
    6058         119 :   if (s == 1) m = 0; else if (s == -1) p = 0;
    6059         119 :   res = cgetg(6*l - 5, t_VECSMALL);
    6060         119 :   c = 1;
    6061         119 :   if (!odd(N))
    6062             :   { /* d = 1 */
    6063          56 :     if (p) res[c++] = 8;
    6064          56 :     if (m) { res[c++] =-8; res[c++] =-4; }
    6065             :   }
    6066         364 :   for (i = 2; i < l; i++)
    6067             :   { /* skip d = 1, done above */
    6068         245 :     long d = D[i], d4 = d & 3L; /* d odd, squarefree, d4 = 1 or 3 */
    6069         245 :     if (d4 == 1) { if (p) res[c++] = d; }
    6070         182 :     else         { if (m) res[c++] =-d; }
    6071         245 :     if (!odd(N))
    6072             :     {
    6073          56 :       if (p) { res[c++] = 8*d; if (d4 == 3) res[c++] = 4*d; }
    6074          56 :       if (m) { res[c++] =-8*d; if (d4 == 1) res[c++] =-4*d; }
    6075             :     }
    6076             :   }
    6077         119 :   setlg(res, c); return res;
    6078             : }
    6079             : 
    6080             : /* Return 1 if F is definitely not S4 type; return 0 on failure. */
    6081             : static long
    6082         105 : mfisnotS4(long N, GEN w)
    6083             : {
    6084         105 :   GEN D = mfunram(N, 0);
    6085         105 :   long i, lD = lg(D), lw = lg(w);
    6086         616 :   for (i = 1; i < lD; i++)
    6087             :   {
    6088         511 :     long p, d = D[i], ok = 0;
    6089        1442 :     for (p = 2; p < lw; p++)
    6090        1442 :       if (w[p] && kross(d,p) == -1) { ok = 1; break; }
    6091         511 :     if (!ok) return 0;
    6092             :   }
    6093         105 :   return 1;
    6094             : }
    6095             : 
    6096             : /* Return 1 if Q(sqrt(5)) \not\subset Q(F), i.e. F is definitely not A5 type;
    6097             :  * return 0 on failure. */
    6098             : static long
    6099         105 : mfisnotA5(GEN F)
    6100             : {
    6101         105 :   GEN CHI = mf_get_CHI(F), P = mfcharpol(CHI), T, Q;
    6102             : 
    6103         105 :   if (mfcharorder(CHI) % 5 == 0) return 0;
    6104         105 :   T = mf_get_field(F); if (degpol(T) == 1) return 1;
    6105         105 :   if (degpol(P) > 1) T = rnfequation(P,T);
    6106         105 :   Q = gsubgs(pol_xn(2,varn(T)), 5);
    6107         105 :   return (typ(nfisincl(Q, T)) == t_INT);
    6108             : }
    6109             : 
    6110             : /* v[p+1]^2 / chi(p) - 2 = z + 1/z with z primitive root of unity of order n,
    6111             :  * return n */
    6112             : static long
    6113        6741 : mffindrootof1(GEN v, long p, GEN CHI)
    6114             : {
    6115        6741 :   GEN ap = gel(v,p+1), u0, u1, u1k, u2;
    6116        6741 :   long c = 1;
    6117        6741 :   if (gequal0(ap)) return 2;
    6118        5033 :   u0 = gen_2; u1k = u1 = gsubgs(gdiv(gsqr(ap), mfchareval(CHI, p)), 2);
    6119       14812 :   while (!gequalsg(2, liftpol_shallow(u1))) /* u1 = z^c + z^-c */
    6120             :   {
    6121        9779 :     u2 = gsub(gmul(u1k, u1), u0);
    6122        9779 :     u0 = u1; u1 = u2; c++;
    6123             :   }
    6124        5033 :   return c;
    6125             : }
    6126             : 
    6127             : /* we known that F is not dihedral */
    6128             : static long
    6129         182 : mfgaloistype_i(long N, GEN CHI, GEN F, GEN v)
    6130             : {
    6131             :   forprime_t iter;
    6132         182 :   long lim = lg(v)-2;
    6133         182 :   GEN w = zero_zv(lim);
    6134             :   pari_sp av;
    6135             :   ulong p;
    6136         182 :   u_forprime_init(&iter, 2, lim);
    6137         182 :   av = avma;
    6138        5292 :   while((p = u_forprime_next(&iter))) if (N%p) switch(mffindrootof1(v, p, CHI))
    6139             :   {
    6140        1400 :     case 1: case 2: continue;
    6141        3451 :     case 3: w[p] = 1; break;
    6142          70 :     case 4: return -24; /* S4 */
    6143           0 :     case 5: return -60; /* A5 */
    6144           7 :     default: pari_err_DOMAIN("mfgaloistype", "form", "not a",
    6145             :                              strtoGENstr("cuspidal eigenform"), F);
    6146           0 :     set_avma(av);
    6147             :   }
    6148         105 :   if (mfisnotS4(N,w) && mfisnotA5(F)) return -12; /* A4 */
    6149           0 :   return 0; /* FAILURE */
    6150             : }
    6151             : 
    6152             : static GEN
    6153         224 : mfgaloistype0(long N, GEN CHI, GEN F, GEN DIH, long lim)
    6154             : {
    6155         224 :   pari_sp av = avma;
    6156         224 :   GEN vF = mftocol(F, lim, 1);
    6157         224 :   long t = mfisdihedral(vF, DIH), bound;
    6158         224 :   if (t) return gc_stoi(av,t);
    6159         182 :   bound = maxss(200, 5*expu(N)*expu(N));
    6160             :   for(;;)
    6161             :   {
    6162         182 :     t = mfgaloistype_i(N, CHI, F, vF);
    6163         175 :     set_avma(av); if (t) return stoi(t);
    6164           0 :     if (lim > bound) return gen_0;
    6165           0 :     lim += lim >> 1;
    6166           0 :     vF = mfcoefs_i(F,lim,1);
    6167             :   }
    6168             : }
    6169             : 
    6170             : /* If f is NULL, give all the galoistypes, otherwise just for f */
    6171             : /* Return 0 to indicate failure; in this case the type is either -12 or -60,
    6172             :  * most likely -12. FIXME using the Galois representation. */
    6173             : GEN
    6174         231 : mfgaloistype(GEN NK, GEN f)
    6175             : {
    6176         231 :   pari_sp av = avma;
    6177         231 :   GEN CHI, T, F, DIH, SP, mf = checkMF_i(NK);
    6178             :   long N, k, lL, i, lim, SB;
    6179             : 
    6180         231 :   if (f && !checkmf_i(f)) pari_err_TYPE("mfgaloistype", f);
    6181         224 :   if (mf)
    6182             :   {
    6183         189 :     N = MF_get_N(mf);
    6184         189 :     k = MF_get_k(mf);
    6185         189 :     CHI = MF_get_CHI(mf);
    6186             :   }
    6187             :   else
    6188             :   {
    6189          35 :     checkNK(NK, &N, &k, &CHI, 0);
    6190          35 :     mf = f? NULL: mfinit_i(NK, mf_NEW);
    6191             :   }
    6192         224 :   if (k != 1) pari_err_DOMAIN("mfgaloistype", "k", "!=", gen_1, stoi(k));
    6193         224 :   SB = mf? mfsturm_mf(mf): mfsturmNk(N,1);
    6194         224 :   SP = get_DIH(N);
    6195         224 :   DIH = mfdihedralnew(N, CHI, SP);
    6196         224 :   lim = lg(DIH) == 1? 200: SB;
    6197         224 :   DIH = mkvec2(DIH, mfvectomat(DIH,SB,1));
    6198         224 :   if (f) return gc_INT(av, mfgaloistype0(N,CHI, f, DIH, lim));
    6199         126 :   F = mfeigenbasis(mf); lL = lg(F);
    6200         126 :   T = cgetg(lL, t_VEC);
    6201         252 :   for (i=1; i < lL; i++) gel(T,i) = mfgaloistype0(N, CHI, gel(F,i), DIH, lim);
    6202         126 :   return gc_upto(av, T);
    6203             : }
    6204             : 
    6205             : /******************************************************************/
    6206             : /*                   Find all dihedral forms.                     */
    6207             : /******************************************************************/
    6208             : /* lim >= 2 */
    6209             : static void
    6210          14 : consttabdihedral(long lim) { cache_set(cache_DIH, mfdihedralall(lim)); }
    6211             : 
    6212             : /* a ideal coprime to bnr modulus */
    6213             : static long
    6214      107611 : mfdiheval(GEN bnr, GEN w, GEN a)
    6215             : {
    6216      107611 :   GEN L, cycn = gel(w,1), chin = gel(w,2);
    6217      107611 :   long ordmax = cycn[1];
    6218      107611 :   L = ZV_to_Flv(isprincipalray(bnr,a), ordmax);
    6219      107611 :   return Flv_dotproduct(chin, L, ordmax);
    6220             : }
    6221             : 
    6222             : /* x(t^k) mod T = polcyclo(m), 0 <= k < m */
    6223             : static GEN
    6224       30331 : Galois(GEN x, long k, GEN T, long m)
    6225             : {
    6226             :   GEN B;
    6227             :   long i, ik, d;
    6228       30331 :   if (typ(x) != t_POL) return x;
    6229        7455 :   if (varn(x) != varn(T)) pari_APPLY_pol_normalized(Galois(gel(x,i), k, T, m));
    6230        7420 :   if ((d = degpol(x)) <= 0) return x;
    6231        7063 :   B = cgetg(m + 2, t_POL); B[1] = x[1]; gel(B,2) = gel(x,2);
    6232       61565 :   for (i = 1; i < m; i++) gel(B, i+2) = gen_0;
    6233       23940 :   for (i = 1, ik = k; i <= d; i++, ik = Fl_add(ik, k, m))
    6234       16877 :     gel(B, ik + 2) = gel(x, i+2);
    6235        7063 :   return QX_ZX_rem(normalizepol(B), T);
    6236             : }
    6237             : static GEN
    6238        1022 : vecGalois(GEN x, long k, GEN T, long m)
    6239       31332 : { pari_APPLY_same(Galois(gel(x,i), k, T, m)); }
    6240             : 
    6241             : static GEN
    6242      234178 : fix_pol(GEN S, GEN Pn, int *trace)
    6243             : {
    6244      234178 :   if (typ(S) != t_POL) return S;
    6245      118069 :   S = RgX_rem(S, Pn);
    6246      118069 :   if (typ(S) == t_POL)
    6247             :   {
    6248      118069 :     switch(lg(S))
    6249             :     {
    6250       45108 :       case 2: return gen_0;
    6251       20517 :       case 3: return gel(S,2);
    6252             :     }
    6253       52444 :     *trace = 1;
    6254             :   }
    6255       52444 :   return S;
    6256             : }
    6257             : 
    6258             : static GEN
    6259       13573 : dihan(GEN bnr, GEN w, GEN k0j, long m, ulong lim)
    6260             : {
    6261       13573 :   GEN nf = bnr_get_nf(bnr), f = bid_get_ideal(bnr_get_bid(bnr));
    6262       13573 :   GEN v = zerovec(lim+1), cycn = gel(w,1), Tinit = gel(w,3);
    6263       13573 :   GEN Pn = gel(Tinit,lg(Tinit)==4? 2: 1);
    6264       13573 :   long j, ordmax = cycn[1];
    6265       13573 :   long D = itos(nf_get_disc(nf)), vt = varn(Pn);
    6266       13573 :   int trace = 0;
    6267             :   ulong p, n;
    6268             :   forprime_t T;
    6269             : 
    6270       13573 :   if (!lim) return v;
    6271       13363 :   gel(v,2) = gen_1;
    6272       13363 :   u_forprime_init(&T, 2, lim);
    6273             :   /* fill in prime powers first */
    6274      116207 :   while ((p = u_forprime_next(&T)))
    6275             :   {
    6276             :     GEN vP, vchiP, S;
    6277             :     long k, lP;
    6278             :     ulong q, qk;
    6279      102844 :     if (kross(D,p) >= 0) q = p;
    6280       45192 :     else if (!(q = umuluu_le(p,p,lim))) continue;
    6281             :     /* q = Norm P */
    6282       65856 :     vP = idealprimedec(nf, utoipos(p));
    6283       65856 :     lP = lg(vP);
    6284       65856 :     vchiP = cgetg(lP, t_VECSMALL);
    6285      179081 :     for (j = k = 1; j < lP; j++)
    6286             :     {
    6287      113225 :       GEN P = gel(vP,j);
    6288      113225 :       if (!idealval(nf, f, P)) vchiP[k++] = mfdiheval(bnr,w,P);
    6289             :     }
    6290       65856 :     if (k == 1) continue;
    6291       62188 :     setlg(vchiP, k); lP = k;
    6292       62188 :     if (lP == 2)
    6293             :     { /* one prime above p not dividing f */
    6294       16765 :       long s, s0 = vchiP[1];
    6295       27069 :       for (qk=q, s = s0;; s = Fl_add(s,s0,ordmax))
    6296             :       {
    6297       27069 :         S = Qab_zeta(s, ordmax, vt);
    6298       27069 :         gel(v, qk+1) = fix_pol(S, Pn, &trace);
    6299       27069 :         if (!(qk = umuluu_le(qk,q,lim))) break;
    6300             :       }
    6301             :     }
    6302             :     else /* two primes above p not dividing f */
    6303             :     {
    6304       45423 :       long s, s0 = vchiP[1], s1 = vchiP[2];
    6305       45423 :       for (qk=q, k = 1;; k++)
    6306       18424 :       { /* sum over a,b s.t. Norm( P1^a P2^b ) = q^k, i.e. a+b = k */
    6307             :         long a;
    6308       63847 :         GEN S = gen_0;
    6309      220752 :         for (a = 0; a <= k; a++)
    6310             :         {
    6311      156905 :           s = Fl_add(Fl_mul(a, s0, ordmax), Fl_mul(k-a, s1, ordmax), ordmax);
    6312      156905 :           S = gadd(S, Qab_zeta(s, ordmax, vt));
    6313             :         }
    6314       63847 :         gel(v, qk+1) = fix_pol(S, Pn, &trace);
    6315       63847 :         if (!(qk = umuluu_le(qk,q,lim))) break;
    6316             :       }
    6317             :     }
    6318             :   }
    6319             :   /* complete with nonprime powers */
    6320      308098 :   for (n = 2; n <= lim; n++)
    6321             :   {
    6322      294735 :     GEN S, fa = myfactoru(n), P = gel(fa, 1), E = gel(fa, 2);
    6323             :     long q;
    6324      294735 :     if (lg(P) == 2) continue;
    6325             :     /* not a prime power */
    6326      143262 :     q = upowuu(P[1],E[1]);
    6327      143262 :     S = gmul(gel(v, q + 1), gel(v, n/q + 1));
    6328      143262 :     gel(v, n+1) = fix_pol(S, Pn, &trace);
    6329             :   }
    6330       13363 :   if (trace)
    6331             :   {
    6332        7154 :     long k0 = k0j[1], jdeg = k0j[2];
    6333        7154 :     v = QabV_tracerel(Tinit, jdeg, v); /* Apply Galois Mod(k0, ordw) */
    6334        7154 :     if (k0 > 1) v = vecGalois(v, k0, gel(Tinit,1), m);
    6335             :   }
    6336       13363 :   return v;
    6337             : }
    6338             : 
    6339             : /* as cyc_normalize for t_VECSMALL cyc */
    6340             : static GEN
    6341       26810 : cyc_normalize_zv(GEN cyc)
    6342             : {
    6343       26810 :   long i, o = cyc[1], l = lg(cyc); /* > 1 */
    6344       26810 :   GEN D = cgetg(l, t_VECSMALL);
    6345       31185 :   D[1] = o; for (i = 2; i < l; i++) D[i] = o / cyc[i];
    6346       26810 :   return D;
    6347             : }
    6348             : /* as char_normalize for t_VECSMALLs */
    6349             : static GEN
    6350      118517 : char_normalize_zv(GEN chi, GEN ncyc)
    6351             : {
    6352      118517 :   long i, l = lg(chi);
    6353      118517 :   GEN c = cgetg(l, t_VECSMALL);
    6354      118517 :   if (l > 1) {
    6355      118517 :     c[1] = chi[1];
    6356      160454 :     for (i = 2; i < l; i++) c[i] = chi[i] * ncyc[i];
    6357             :   }
    6358      118517 :   return c;
    6359             : }
    6360             : 
    6361             : static GEN
    6362        9331 : dihan_bnf(long D)
    6363             : {
    6364        9331 :   GEN c = getrand(), bnf;
    6365        9331 :   setrand(gen_1);
    6366        9331 :   bnf = Buchall(quadpoly_i(stoi(D)), nf_FORCE, LOWDEFAULTPREC);
    6367        9331 :   setrand(c);
    6368        9331 :   return bnf;
    6369             : }
    6370             : static GEN
    6371       37758 : dihan_bnr(GEN bnf, GEN A)
    6372             : {
    6373       37758 :   GEN c = getrand(), bnr;
    6374       37758 :   setrand(gen_1);
    6375       37758 :   bnr = Buchray(bnf, A, nf_INIT|nf_GEN);
    6376       37758 :   setrand(c);
    6377       37758 :   return bnr;
    6378             : }
    6379             : /* Hecke xi * (D/.) = Dirichlet chi, return v in Q^r st chi(g_i) = e(v[i]).
    6380             :  * cycn = cyc_normalize_zv(bnr.cyc), chin = char_normalize_zv(chi,cyc) */
    6381             : static GEN
    6382       34489 : bnrchartwist2conrey(GEN chin, GEN cycn, GEN bnrconreyN, GEN kroconreyN)
    6383             : {
    6384       34489 :   long l = lg(bnrconreyN), c1 = cycn[1], i;
    6385       34489 :   GEN v = cgetg(l, t_COL);
    6386      125363 :   for (i = 1; i < l; i++)
    6387             :   {
    6388       90874 :     GEN d = sstoQ(zv_dotproduct(chin, gel(bnrconreyN,i)), c1);
    6389       90874 :     if (kroconreyN[i] < 0) d = gadd(d, ghalf);
    6390       90874 :     gel(v,i) = d;
    6391             :   }
    6392       34489 :   return v;
    6393             : }
    6394             : 
    6395             : /* chi(g_i) = e(v[i]) denormalize wrt Conrey generators orders */
    6396             : static GEN
    6397       34489 : conreydenormalize(GEN znN, GEN v)
    6398             : {
    6399       34489 :   GEN gcyc = znstar_get_conreycyc(znN), w;
    6400       34489 :   long l = lg(v), i;
    6401       34489 :   w = cgetg(l, t_COL);
    6402      125363 :   for (i = 1; i < l; i++)
    6403       90874 :     gel(w,i) = modii(gmul(gel(v,i), gel(gcyc,i)), gel(gcyc,i));
    6404       34489 :   return w;
    6405             : }
    6406             : 
    6407             : static long
    6408       84028 : Miyake(GEN vchi, GEN gb, GEN cycn)
    6409             : {
    6410       84028 :   long i, e = cycn[1], lb = lg(gb);
    6411       84028 :   GEN v = char_normalize_zv(vchi, cycn);
    6412      124992 :   for (i = 1; i < lb; i++)
    6413      100268 :     if ((zv_dotproduct(v, gel(gb,i)) -  v[i]) % e) return 1;
    6414       24724 :   return 0;
    6415             : }
    6416             : 
    6417             : /* list of Hecke characters not induced by a Dirichlet character up to Galois
    6418             :  * conjugation, whose conductor is bnr.cond; cycn = cyc_normalize(bnr.cyc)*/
    6419             : static GEN
    6420       26810 : mklvchi(GEN bnr, GEN cycn, GEN gb)
    6421             : {
    6422       26810 :   GEN cyc = bnr_get_cyc(bnr), cycsmall = ZV_to_zv(cyc);
    6423       26810 :   GEN vchi = cyc2elts(cycsmall);
    6424       26810 :   long ordmax = cycsmall[1], c, i, l;
    6425       26810 :   l = lg(vchi);
    6426      304024 :   for (i = c = 1; i < l; i++)
    6427             :   {
    6428      277214 :     GEN chi = gel(vchi,i);
    6429      277214 :     if (!gb || Miyake(chi, gb, cycn)) gel(vchi, c++) = Flv_to_ZV(chi);
    6430             :   }
    6431       26810 :   setlg(vchi, c); l = c;
    6432      279300 :   for (i = 1; i < l; i++)
    6433             :   {
    6434      252490 :     GEN chi = gel(vchi,i);
    6435             :     long n;
    6436      252490 :     if (!chi) continue;
    6437     1055754 :     for (n = 2; n < ordmax; n++)
    6438      966476 :       if (ugcd(n, ordmax) == 1)
    6439             :       {
    6440      397670 :         GEN tmp = ZV_ZV_mod(gmulsg(n, chi), cyc);
    6441             :         long j;
    6442     7623539 :         for (j = i+1; j < l; j++)
    6443     7225869 :           if (gel(vchi,j) && gequal(gel(vchi,j), tmp)) gel(vchi,j) = NULL;
    6444             :       }
    6445             :   }
    6446      279300 :   for (i = c = 1; i < l; i++)
    6447             :   {
    6448      252490 :     GEN chi = gel(vchi,i);
    6449      252490 :     if (chi && bnrisconductor(bnr, chi)) gel(vchi, c++) = chi;
    6450             :   }
    6451       26810 :   setlg(vchi, c); return vchi;
    6452             : }
    6453             : 
    6454             : static GEN
    6455        7805 : get_gb(GEN bnr, GEN con)
    6456             : {
    6457        7805 :   GEN gb, g = bnr_get_gen(bnr), nf = bnr_get_nf(bnr);
    6458        7805 :   long i, l = lg(g);
    6459        7805 :   gb = cgetg(l, t_VEC);
    6460       18326 :   for (i = 1; i < l; i++)
    6461       10521 :     gel(gb,i) = ZV_to_zv(isprincipalray(bnr, galoisapply(nf, con, gel(g,i))));
    6462        7805 :   return gb;
    6463             : }
    6464             : static GEN
    6465       15862 : get_bnrconreyN(GEN bnr, GEN znN)
    6466             : {
    6467       15862 :   GEN z, g = znstar_get_conreygen(znN);
    6468       15862 :   long i, l = lg(g);
    6469       15862 :   z = cgetg(l, t_VEC);
    6470       57134 :   for (i = 1; i < l; i++) gel(z,i) = ZV_to_zv(isprincipalray(bnr,gel(g,i)));
    6471       15862 :   return z;
    6472             : }
    6473             : /* con = NULL if D > 0 or if D < 0 and id != idcon. */
    6474             : static GEN
    6475       33698 : mfdihedralcommon(GEN bnf, GEN id, GEN znN, GEN kroconreyN, long vt,
    6476             :                  long N, long D, GEN con)
    6477             : {
    6478       33698 :   GEN bnr = dihan_bnr(bnf, id), cyc = ZV_to_zv( bnr_get_cyc(bnr) );
    6479             :   GEN bnrconreyN, cycn, cycN, Lvchi, res, P, vT;
    6480             :   long j, ordmax, l, lc, deghecke;
    6481             : 
    6482       33698 :   lc = lg(cyc); if (lc == 1) return NULL;
    6483       26810 :   cycn = cyc_normalize_zv(cyc);
    6484       26810 :   Lvchi = mklvchi(bnr, cycn, con? get_gb(bnr, con): NULL);
    6485       26810 :   l = lg(Lvchi);
    6486       26810 :   if (l == 1) return NULL;
    6487             : 
    6488       15862 :   bnrconreyN = get_bnrconreyN(bnr, znN);
    6489       15862 :   cycN = ZV_to_zv(znstar_get_cyc(znN));
    6490       15862 :   ordmax = cyc[1];
    6491       15862 :   vT = const_vec(odd(ordmax)? ordmax << 1: ordmax, NULL);
    6492       15862 :   P = polcyclo(ordmax, vt);
    6493       15862 :   gel(vT,ordmax) = Qab_trace_init(ordmax, ordmax, P, P);
    6494       15862 :   deghecke = myeulerphiu(ordmax);
    6495       15862 :   res = cgetg(l, t_VEC);
    6496       50351 :   for (j = 1; j < l; j++)
    6497             :   {
    6498       34489 :     GEN T, v, vchi = ZV_to_zv(gel(Lvchi,j));
    6499       34489 :     GEN chi, chin = char_normalize_zv(vchi, cycn);
    6500             :     long o, vnum, k0, degrel;
    6501       34489 :     v = bnrchartwist2conrey(chin, cycn, bnrconreyN, kroconreyN);
    6502       34489 :     o = itou(Q_denom(v));
    6503       34489 :     T = gel(vT, o);
    6504       34489 :     if (!T) gel(vT,o) = T = Qab_trace_init(ordmax, o, P, polcyclo(o,vt));
    6505       34489 :     chi = conreydenormalize(znN, v);
    6506       34489 :     vnum = itou(znconreyexp(znN, chi));
    6507       34489 :     chi = ZV_to_zv(znconreychar(znN,chi));
    6508       34489 :     degrel = deghecke / degpol(gel(T,1));
    6509       34489 :     k0 = zv_cyc_minimize(cycN, chi, coprimes_zv(o));
    6510       34489 :     vnum = Fl_powu(vnum, k0, N);
    6511             :     /* encodes degrel forms: jdeg = 0..degrel-1 */
    6512       34489 :     gel(res,j) = mkvec3(mkvecsmalln(5, N, k0 % o, vnum, D, degrel),
    6513             :                         id, mkvec3(cycn,chin,T));
    6514             :   }
    6515       15862 :   return res;
    6516             : }
    6517             : 
    6518             : static long
    6519       49364 : is_cond(long D, long n)
    6520             : {
    6521       49364 :   if (D > 0) return n != 4 || (D&7L) == 1;
    6522       30114 :   return n != 2 && n != 3 && (n != 4 || (D&7L)!=1);
    6523             : }
    6524             : /* Append to v all dihedral weight 1 forms coming from D, if fundamental.
    6525             :  * level in [l1, l2] */
    6526             : static void
    6527       18718 : append_dihedral(GEN v, long D, long l1, long l2, long vt)
    6528             : {
    6529       18718 :   long Da = labs(D), no, i, numi, ct, min, max;
    6530             :   GEN bnf, con, vI, resall, arch1, arch2;
    6531             :   pari_sp av;
    6532             : 
    6533             :   /* min <= Nf <= max */
    6534       18718 :   max = l2 / Da;
    6535       18718 :   if (l1 == l2)
    6536             :   { /* assume Da | l2 */
    6537         140 :     min = max;
    6538         140 :     if (D > 0 && min < 3) return;
    6539             :   }
    6540             :   else /* assume l1 < l2 */
    6541       18578 :     min = (l1 + Da-1)/Da;
    6542       18718 :   if (!sisfundamental(D)) return;
    6543             : 
    6544        5726 :   av = avma;
    6545        5726 :   bnf = dihan_bnf(D);
    6546        5726 :   con = nf2_get_conj(bnf_get_nf(bnf));
    6547        5726 :   vI = ideallist(bnf, max);
    6548       55090 :   numi = 0; for (i = min; i <= max; i++) numi += lg(gel(vI, i)) - 1;
    6549        5726 :   if (D > 0)
    6550             :   {
    6551        1428 :     numi <<= 1;
    6552        1428 :     arch1 = mkvec2(gen_1,gen_0);
    6553        1428 :     arch2 = mkvec2(gen_0,gen_1);
    6554             :   }
    6555             :   else
    6556        4298 :     arch1 = arch2 = NULL;
    6557        5726 :   resall = cgetg(numi+1, t_VEC); ct = 1;
    6558       55090 :   for (no = min; no <= max; no++) if (is_cond(D, no))
    6559             :   {
    6560       44646 :     long N = Da*no, lc, lI;
    6561       44646 :     GEN I = gel(vI, no), znN = znstar0(utoipos(N), 1), conreyN, kroconreyN;
    6562             : 
    6563       44646 :     conreyN = znstar_get_conreygen(znN); lc = lg(conreyN);
    6564       44646 :     kroconreyN = cgetg(lc, t_VECSMALL);
    6565      166054 :     for (i = 1; i < lc; i++) kroconreyN[i] = krosi(D, gel(conreyN, i));
    6566       44646 :     lI = lg(I);
    6567       87822 :     for (i = 1; i < lI; i++)
    6568             :     {
    6569       43176 :       GEN id = gel(I, i), idcon, z;
    6570             :       long j;
    6571       43176 :       if (typ(id) == t_INT) continue;
    6572       28182 :       idcon = galoisapply(bnf, con, id);
    6573       51408 :       for (j = i; j < lI; j++)
    6574       51408 :         if (gequal(idcon, gel(I, j))) { gel(I, j) = gen_0; break; }
    6575       28182 :       if (D < 0)
    6576             :       {
    6577       17479 :         GEN conk = i == j ? con : NULL;
    6578       17479 :         z = mfdihedralcommon(bnf, id, znN, kroconreyN, vt, N, D, conk);
    6579       17479 :         if (z) gel(resall, ct++) = z;
    6580             :       }
    6581             :       else
    6582             :       {
    6583             :         GEN ide;
    6584       10703 :         ide = mkvec2(id, arch1);
    6585       10703 :         z = mfdihedralcommon(bnf, ide, znN, kroconreyN, vt, N, D, NULL);
    6586       10703 :         if (z) gel(resall, ct++) = z;
    6587       10703 :         if (gequal(idcon,id)) continue;
    6588        5516 :         ide = mkvec2(id, arch2);
    6589        5516 :         z = mfdihedralcommon(bnf, ide, znN, kroconreyN, vt, N, D, NULL);
    6590        5516 :         if (z) gel(resall, ct++) = z;
    6591             :       }
    6592             :     }
    6593             :   }
    6594        5726 :   if (ct == 1) set_avma(av);
    6595             :   else
    6596             :   {
    6597        4816 :     setlg(resall, ct);
    6598        4816 :     vectrunc_append(v, gc_GEN(av, shallowconcat1(resall)));
    6599             :   }
    6600             : }
    6601             : 
    6602             : static long
    6603       42042 : di_N(GEN a) { return gel(a,1)[1]; }
    6604             : static GEN
    6605          14 : mfdihedral(long N)
    6606             : {
    6607          14 :   GEN D = mydivisorsu(N), res = vectrunc_init(2*N);
    6608          14 :   long j, l = lg(D), vt = fetch_user_var("t");
    6609         105 :   for (j = 2; j < l; j++)
    6610             :   { /* skip d = 1 */
    6611          91 :     long d = D[j];
    6612          91 :     if (d == 2) continue;
    6613          84 :     append_dihedral(res, -d, N,N, vt);
    6614          84 :     if (d >= 5 && D[l-j] >= 3) append_dihedral(res, d, N,N, vt);/* Nf >= 3 */
    6615             :   }
    6616          14 :   if (lg(res) > 1) res = shallowconcat1(res);
    6617          14 :   return res;
    6618             : }
    6619             : /* All primitive dihedral weight 1 forms of leven in [1, N], N > 1 */
    6620             : static GEN
    6621          14 : mfdihedralall(long N)
    6622             : {
    6623          14 :   GEN res = vectrunc_init(2*N), z;
    6624          14 :   long D, ct, i, vt = fetch_user_var("t");
    6625             : 
    6626       13986 :   for (D = -3; D >= -N; D--) append_dihedral(res, D, 1,N, vt);
    6627             :   /* Nf >= 3 (GTM 193, prop 3.3.18) */
    6628        4620 :   for (D = N / 3; D >= 5; D--) append_dihedral(res, D, 1,N, vt);
    6629          14 :   ct = lg(res);
    6630          14 :   if (ct > 1)
    6631             :   { /* sort wrt N */
    6632          14 :     res = shallowconcat1(res);
    6633          14 :     res = vecpermute(res, indexvecsort(res, mkvecsmall(1)));
    6634          14 :     ct = lg(res);
    6635             :   }
    6636          14 :   z = const_vec(N, cgetg(1,t_VEC));
    6637        7658 :   for (i = 1; i < ct;)
    6638             :   { /* regroup result sharing the same N */
    6639        7644 :     long n = di_N(gel(res,i)), j = i+1, k;
    6640             :     GEN v;
    6641       34412 :     while (j < ct && di_N(gel(res,j)) == n) j++;
    6642        7644 :     gel(z, n) = v = cgetg(j-i+1, t_VEC);
    6643       42056 :     for (k = 1; i < j; k++,i++) gel(v,k) = gel(res,i);
    6644             :   }
    6645          14 :   return z;
    6646             : }
    6647             : 
    6648             : /* return [vF, index], where vecpermute(vF,index) generates dihedral forms
    6649             :  * for character CHI */
    6650             : static GEN
    6651       24969 : mfdihedralnew_i(long N, GEN CHI, GEN SP)
    6652             : {
    6653             :   GEN bnf, Tinit, Pm, vf, M, V, NK;
    6654             :   long Dold, d, ordw, i, SB, c, l, k0, k1, chino, chinoorig, lv;
    6655             : 
    6656       24969 :   lv = lg(SP); if (lv == 1) return NULL;
    6657       12138 :   CHI = mfcharinduce(CHI,N);
    6658       12138 :   ordw = mfcharorder(CHI);
    6659       12138 :   chinoorig = mfcharno(CHI);
    6660       12138 :   k0 = mfconreyminimize(CHI);
    6661       12138 :   chino = Fl_powu(chinoorig, k0, N);
    6662       12138 :   k1 = Fl_inv(k0 % ordw, ordw);
    6663       12138 :   V = cgetg(lv, t_VEC);
    6664       12138 :   d = 0;
    6665       39039 :   for (i = l = 1; i < lv; i++)
    6666             :   {
    6667       26901 :     GEN sp = gel(SP,i), T = gel(sp,1);
    6668       26901 :     if (T[3] != chino) continue;
    6669        4060 :     d += T[5];
    6670        4060 :     if (k1 != 1)
    6671             :     {
    6672          77 :       GEN t = leafcopy(T);
    6673          77 :       t[3] = chinoorig;
    6674          77 :       t[2] = (t[2]*k1) % ordw;
    6675          77 :       sp = mkvec4(t, gel(sp,2), gel(sp,3), gel(sp,4));
    6676             :     }
    6677        4060 :     gel(V, l++) = sp;
    6678             :   }
    6679       12138 :   setlg(V, l); /* dihedral forms of level N and character CHI */
    6680       12138 :   if (l == 1) return NULL;
    6681             : 
    6682        2555 :   SB = mfsturmNk(N,1) + 1;
    6683        2555 :   M = cgetg(d+1, t_MAT);
    6684        2555 :   vf = cgetg(d+1, t_VEC);
    6685        2555 :   NK = mkNK(N, 1, CHI);
    6686        2555 :   bnf = NULL; Dold = 0;
    6687        6615 :   for (i = c = 1; i < l; i++)
    6688             :   { /* T = [N, k0, conreyno, D, degrel] */
    6689        4060 :     GEN bnr, Vi = gel(V,i), T = gel(Vi,1), id = gel(Vi,2), w = gel(Vi,3);
    6690        4060 :     long jdeg, k0i = T[2], D = T[4], degrel = T[5];
    6691             : 
    6692        4060 :     if (D != Dold) { Dold = D; bnf = dihan_bnf(D); }
    6693        4060 :     bnr = dihan_bnr(bnf, id);
    6694       12054 :     for (jdeg = 0; jdeg < degrel; jdeg++,c++)
    6695             :     {
    6696        7994 :       GEN k0j = mkvecsmall2(k0i, jdeg), an = dihan(bnr, w, k0j, ordw, SB);
    6697        7994 :       settyp(an, t_COL); gel(M,c) = an;
    6698        7994 :       gel(vf,c) = tag3(t_MF_DIHEDRAL, NK, bnr, w, k0j);
    6699             :     }
    6700             :   }
    6701        2555 :   Tinit = gmael3(V,1,3,3); Pm = gel(Tinit,1);
    6702        2555 :   V = QabM_indexrank(M, degpol(Pm)==1? NULL: Pm, ordw);
    6703        2555 :   return mkvec2(vf,gel(V,2));
    6704             : }
    6705             : static long
    6706       16149 : mfdihedralnewdim(long N, GEN CHI, GEN SP)
    6707             : {
    6708       16149 :   pari_sp av = avma;
    6709       16149 :   GEN S = mfdihedralnew_i(N, CHI, SP);
    6710       16149 :   return gc_long(av, S? lg(gel(S,2))-1: 0);
    6711             : }
    6712             : static GEN
    6713        8820 : mfdihedralnew(long N, GEN CHI, GEN SP)
    6714             : {
    6715        8820 :   pari_sp av = avma;
    6716        8820 :   GEN S = mfdihedralnew_i(N, CHI, SP);
    6717        8820 :   if (!S) retgc_const(av, cgetg(1, t_VEC));
    6718         917 :   return vecpermute(gel(S,1), gel(S,2));
    6719             : }
    6720             : 
    6721             : static long
    6722        7105 : mfdihedralcuspdim(long N, GEN CHI, GEN vSP)
    6723             : {
    6724        7105 :   pari_sp av = avma;
    6725             :   GEN D, CHIP;
    6726             :   long F, i, lD, dim;
    6727             : 
    6728        7105 :   CHIP = mfchartoprimitive(CHI, &F);
    6729        7105 :   D = mydivisorsu(N/F); lD = lg(D);
    6730        7105 :   dim = mfdihedralnewdim(N, CHI, gel(vSP,N)); /* d = 1 */
    6731       16149 :   for (i = 2; i < lD; i++)
    6732             :   {
    6733        9044 :     long d = D[i], a = mfdihedralnewdim(N/d, CHIP, gel(vSP, N/d));
    6734        9044 :     if (a) dim += a * mynumdivu(d);
    6735             :   }
    6736        7105 :   return gc_long(av,dim);
    6737             : }
    6738             : 
    6739             : static GEN
    6740        7385 : mfbdall(GEN E, long N)
    6741             : {
    6742        7385 :   GEN v, D = mydivisorsu(N);
    6743        7385 :   long i, j, nD = lg(D) - 1, nE = lg(E) - 1;
    6744        7385 :   v = cgetg(nD*nE + 1, t_VEC);
    6745       10500 :   for (j = 1; j <= nE; j++)
    6746             :   {
    6747        3115 :     GEN Ej = gel(E, j);
    6748        9513 :     for (i = 0; i < nD; i++) gel(v, i*nE + j) = mfbd_i(Ej, D[i+1]);
    6749             :   }
    6750        7385 :   return v;
    6751             : }
    6752             : static GEN
    6753        3857 : mfdihedralcusp(long N, GEN CHI, GEN vSP)
    6754             : {
    6755        3857 :   pari_sp av = avma;
    6756             :   GEN D, CHIP, z;
    6757             :   long F, i, lD;
    6758             : 
    6759        3857 :   CHIP = mfchartoprimitive(CHI, &F);
    6760        3857 :   D = mydivisorsu(N/F); lD = lg(D);
    6761        3857 :   z = cgetg(lD, t_VEC);
    6762        3857 :   gel(z,1) = mfdihedralnew(N, CHI, gel(vSP,N));
    6763        8596 :   for (i = 2; i < lD; i++) /* skip 1 */
    6764             :   {
    6765        4739 :     GEN LF = mfdihedralnew(N / D[i], CHIP, gel(vSP, N / D[i]));
    6766        4739 :     gel(z,i) = mfbdall(LF, D[i]);
    6767             :   }
    6768        3857 :   return gc_GEN(av, shallowconcat1(z));
    6769             : }
    6770             : 
    6771             : /* used to decide between ratlift and comatrix for ZM_inv; ratlift is better
    6772             :  * when N has many divisors */
    6773             : static int
    6774        2604 : abundant(ulong N) { return mynumdivu(N) >= 8; }
    6775             : 
    6776             : /* CHI an mfchar */
    6777             : static int
    6778         371 : cmp_ord(void *E, GEN a, GEN b)
    6779             : {
    6780         371 :   GEN chia = MF_get_CHI(a), chib = MF_get_CHI(b);
    6781         371 :   (void)E; return cmpii(gmfcharorder(chia), gmfcharorder(chib));
    6782             : }
    6783             : /* mfinit structure.
    6784             : -- mf[1] contains [N,k,CHI,space],
    6785             : -- mf[2] contains vector of closures of Eisenstein series, empty if not
    6786             :    full space.
    6787             : -- mf[3] contains vector of closures, so #mf[3] = dimension of cusp/new space.
    6788             : -- mf[4] contains the corresponding indices: either j for T(j)tf if newspace,
    6789             :    or [M,j,d] for B(d)T(j)tf_M if cuspspace or oldspace.
    6790             : -- mf[5] contains the matrix M of first coefficients of basis, never cleaned.
    6791             :  * NK is either [N,k] or [N,k,CHI].
    6792             :  * mfinit does not do the splitting, only the basis generation. */
    6793             : 
    6794             : /* Set flraw to 1 if do not need mf[5]: no mftobasis etc..., only the
    6795             :    expansions of the basis elements are needed. */
    6796             : 
    6797             : static GEN
    6798        5075 : mfinit_Nkchi(long N, long k, GEN CHI, long space, long flraw)
    6799             : {
    6800        5075 :   GEN M = NULL, mf = NULL, mf1 = mkvec4(utoi(N), stoi(k), CHI, utoi(space));
    6801        5075 :   long sb = mfsturmNk(N, k);
    6802        5075 :   if (k < 0 || badchar(N, k, CHI)) return mfEMPTY(mf1);
    6803        5040 :   if (k == 0 || space == mf_EISEN) /*nothing*/;
    6804        4879 :   else if (k == 1)
    6805             :   {
    6806         364 :     switch (space)
    6807             :     {
    6808         350 :       case mf_NEW:
    6809             :       case mf_FULL:
    6810         350 :       case mf_CUSP: mf = mf1init(N, CHI, NULL, get_vDIH(N,NULL), space, flraw);
    6811         350 :                     break;
    6812           7 :       case mf_OLD: pari_err_IMPL("mfinit in weight 1 for old space");
    6813           7 :       default: pari_err_FLAG("mfinit");
    6814             :     }
    6815             :   }
    6816             :   else /* k >= 2 */
    6817             :   {
    6818        4515 :     long ord = mfcharorder(CHI);
    6819        4515 :     GEN z = NULL, P = (ord <= 2)? NULL: mfcharpol(CHI);
    6820             :     cachenew_t cache;
    6821        4515 :     switch(space)
    6822             :     {
    6823        1239 :       case mf_NEW:
    6824        1239 :         mf = mfnewinit(N, k, CHI, &cache, 1);
    6825        1239 :         if (mf && !flraw) { M = MF_get_M(mf); z = MF_get_Mindex(mf); }
    6826        1239 :         break;
    6827        3269 :       case mf_OLD:
    6828             :       case mf_CUSP:
    6829             :       case mf_FULL:
    6830        3269 :         if (!(mf = mfinitcusp(N, k, CHI, &cache, space))) break;
    6831        2961 :         if (!flraw)
    6832             :         {
    6833        2296 :           M = bhnmat_extend(M, sb+1, 1, MF_get_S(mf), &cache);
    6834        2296 :           if (space != mf_FULL) gel(mf,5) = mfcleanCHI(M, CHI, abundant(N));
    6835             :         }
    6836        2961 :         dbg_cachenew(&cache); break;
    6837           7 :       default: pari_err_FLAG("mfinit");
    6838             :     }
    6839        4508 :     if (z) gel(mf,5) = mfclean2(M, z, P, ord);
    6840             :   }
    6841        5019 :   if (!mf) mf = mfEMPTY(mf1);
    6842             :   else
    6843             :   {
    6844        4053 :     gel(mf,1) = mf1;
    6845        4053 :     if (flraw) gel(mf,5) = zerovec(3);
    6846             :   }
    6847        5019 :   if (!space_is_cusp(space))
    6848             :   {
    6849         861 :     GEN E = mfeisensteinbasis(N, k, CHI);
    6850         861 :     gel(mf,2) = E;
    6851         861 :     if (!flraw)
    6852             :     {
    6853         539 :       if (M)
    6854         231 :         M = shallowconcat(mfvectomat(E, sb+1, 1), M);
    6855             :       else
    6856         308 :         M = mfcoefs_mf(mf, sb+1, 1);
    6857         539 :       gel(mf,5) = mfcleanCHI(M, CHI, abundant(N));
    6858             :     }
    6859             :   }
    6860        5019 :   return mf;
    6861             : }
    6862             : 
    6863             : /* mfinit for k = nk/dk */
    6864             : static GEN
    6865        2765 : mfinit_Nndkchi(long N, long nk, long dk, GEN CHI, long space, long flraw)
    6866         273 : { return (dk == 2)? mf2init_Nkchi(N, nk >> 1, CHI, space, flraw)
    6867        3038 :                   : mfinit_Nkchi(N, nk, CHI, space, flraw); }
    6868             : static GEN
    6869        3430 : mfinit_i(GEN NK, long space)
    6870             : {
    6871             :   GEN CHI, mf;
    6872             :   long N, k, dk, joker;
    6873        3430 :   if (checkmf_i(NK))
    6874             :   {
    6875         161 :     N = mf_get_N(NK);
    6876         161 :     Qtoss(mf_get_gk(NK), &k, &dk);
    6877         161 :     CHI = mf_get_CHI(NK);
    6878             :   }
    6879        3269 :   else if ((mf = checkMF_i(NK)))
    6880             :   {
    6881          21 :     long s = MF_get_space(mf);
    6882          21 :     if (s == space) return mf;
    6883          21 :     Qtoss(MF_get_gk(mf), &k, &dk);
    6884          21 :     if (dk == 1 && k > 1 && space == mf_NEW && (s == mf_CUSP || s == mf_FULL))
    6885          21 :       return mfinittonew(mf);
    6886           0 :     N = MF_get_N(mf);
    6887           0 :     CHI = MF_get_CHI(mf);
    6888             :   }
    6889             :   else
    6890        3248 :     checkNK2(NK, &N, &k, &dk, &CHI, 1);
    6891        3388 :   joker = !CHI || typ(CHI) == t_COL;
    6892        3388 :   if (joker)
    6893             :   {
    6894        1162 :     GEN mf, vCHI = CHI;
    6895             :     long i, j, l;
    6896        1162 :     if (CHI && lg(CHI) == 1) return cgetg(1,t_VEC);
    6897        1155 :     if (k < 0) return mfEMPTYall(N, uutoQ(k,dk), CHI, space);
    6898        1141 :     if (k == 1 && dk == 1 && space != mf_EISEN)
    6899         504 :     {
    6900             :       GEN TMP, vSP, gN, gs;
    6901             :       pari_timer tt;
    6902        1106 :       if (space != mf_CUSP && space != mf_NEW)
    6903           0 :         pari_err_IMPL("mfinit([N,1,wildcard], space != cusp or new space)");
    6904        1106 :       if (wt1empty(N)) return mfEMPTYall(N, gen_1, CHI, space);
    6905         504 :       vCHI = mf1chars(N,vCHI);
    6906         504 :       l = lg(vCHI); mf = cgetg(l, t_VEC); if (l == 1) return mf;
    6907         504 :       TMP = mf1_pre(N); vSP = get_vDIH(N, NULL);
    6908         504 :       gN = utoipos(N); gs = utoi(space);
    6909         504 :       if (DEBUGLEVEL) timer_start(&tt);
    6910        4123 :       for (i = j = 1; i < l; i++)
    6911             :       {
    6912        3619 :         pari_sp av = avma;
    6913        3619 :         GEN c = gel(vCHI,i), z = mf1init(N, c, TMP, vSP, space, 0);
    6914        3619 :         if (z) z = gc_GEN(av, z);
    6915             :         else
    6916             :         {
    6917        2905 :           set_avma(av);
    6918        2905 :           if (CHI) z = mfEMPTY(mkvec4(gN,gen_1,c,gs));
    6919             :         }
    6920        3619 :         if (z) gel(mf, j++) = z;
    6921        3619 :         if (DEBUGLEVEL)
    6922           0 :           timer_printf(&tt, "mf1basis: character %ld / %ld (order = %ld)",
    6923             :                        i, l-1, mfcharorder(c));
    6924             :       }
    6925             :     }
    6926             :     else
    6927             :     {
    6928          35 :       vCHI = mfchars(N,k,dk,vCHI);
    6929          35 :       l = lg(vCHI); mf = cgetg(l, t_VEC);
    6930         119 :       for (i = j = 1; i < l; i++)
    6931             :       {
    6932          84 :         pari_sp av = avma;
    6933          84 :         GEN v = mfinit_Nndkchi(N, k, dk, gel(vCHI,i), space, 0);
    6934          84 :         if (MF_get_dim(v) || CHI) gel(mf, j++) = v; else set_avma(av);
    6935             :       }
    6936             :     }
    6937         539 :     setlg(mf,j);
    6938         539 :     if (!CHI) gen_sort_inplace(mf, NULL, &cmp_ord, NULL);
    6939         539 :     return mf;
    6940             :   }
    6941        2226 :   return mfinit_Nndkchi(N, k, dk, CHI, space, 0);
    6942             : }
    6943             : GEN
    6944        2450 : mfinit(GEN NK, long space)
    6945             : {
    6946        2450 :   pari_sp av = avma;
    6947        2450 :   return gc_GEN(av, mfinit_i(NK, space));
    6948             : }
    6949             : 
    6950             : /* UTILITY FUNCTIONS */
    6951             : static void
    6952         364 : cusp_canon(GEN cusp, long N, long *pA, long *pC)
    6953             : {
    6954         364 :   pari_sp av = avma;
    6955             :   long A, C, tc, cg;
    6956         364 :   if (N <= 0) pari_err_DOMAIN("mfcuspwidth","N","<=",gen_0,stoi(N));
    6957         357 :   if (!cusp || (tc = typ(cusp)) == t_INFINITY) { *pA = 1; *pC = N; return; }
    6958         350 :   if (tc != t_INT && tc != t_FRAC) pari_err_TYPE("checkcusp", cusp);
    6959         350 :   Qtoss(cusp, &A,&C);
    6960         350 :   if (N % C)
    6961             :   {
    6962             :     ulong uC;
    6963          14 :     long u = Fl_invgen((C-1)%N + 1, N, &uC);
    6964          14 :     A = Fl_mul(A, u, N);
    6965          14 :     C = (long)uC;
    6966             :   }
    6967         350 :   cg = ugcd(C, N/C);
    6968         420 :   while (ugcd(A, N) > 1) A += cg;
    6969         350 :   *pA = A % N; *pC = C; set_avma(av);
    6970             : }
    6971             : static long
    6972        1001 : mfcuspcanon_width(long N, long C)
    6973        1001 : { return (!C || C == N)? 1 : N / ugcd(N, Fl_sqr(umodsu(C,N),N)); }
    6974             : /* v = [a,c] a ZC, width of cusp (a:c) */
    6975             : static long
    6976        9975 : mfZC_width(long N, GEN v)
    6977             : {
    6978        9975 :   ulong C = umodiu(gel(v,2), N);
    6979        9975 :   return (C == 0)? 1: N / ugcd(N, Fl_sqr(C,N));
    6980             : }
    6981             : long
    6982         161 : mfcuspwidth(GEN gN, GEN cusp)
    6983             : {
    6984         161 :   long N = 0, A, C;
    6985             :   GEN mf;
    6986         161 :   if (typ(gN) == t_INT) N = itos(gN);
    6987          42 :   else if ((mf = checkMF_i(gN))) N = MF_get_N(mf);
    6988           0 :   else pari_err_TYPE("mfcuspwidth", gN);
    6989         161 :   cusp_canon(cusp, N, &A, &C);
    6990         154 :   return mfcuspcanon_width(N, C);
    6991             : }
    6992             : 
    6993             : /* Q a t_INT */
    6994             : static GEN
    6995          14 : findq(GEN al, GEN Q)
    6996             : {
    6997             :   long n;
    6998          14 :   if (typ(al) == t_FRAC && cmpii(gel(al,2), Q) <= 0)
    6999           0 :     return mkvec(mkvec2(gel(al,1), gel(al,2)));
    7000          14 :   n = 1 + (long)ceil(2.0781*gtodouble(glog(Q, LOWDEFAULTPREC)));
    7001          14 :   return contfracpnqn(gboundcf(al,n), n);
    7002             : }
    7003             : static GEN
    7004          91 : findqga(long N, GEN z)
    7005             : {
    7006          91 :   GEN Q, LDC, CK = NULL, DK = NULL, ma, x, y = imag_i(z);
    7007             :   long j, l;
    7008          91 :   if (gcmpgs(gmulsg(2*N, y), 1) >= 0) return NULL;
    7009          14 :   x = real_i(z);
    7010          14 :   Q = ground(ginv(gsqrt(gmulsg(N, y), LOWDEFAULTPREC)));
    7011          14 :   LDC = findq(gmulsg(-N,x), Q);
    7012          14 :   ma = gen_1; l = lg(LDC);
    7013          35 :   for (j = 1; j < l; j++)
    7014             :   {
    7015          21 :     GEN D, DC = gel(LDC,j), C1 = gel(DC,2);
    7016          21 :     if (cmpii(C1,Q) > 0) break;
    7017          21 :     D = gel(DC,1);
    7018          21 :     if (ugcdiu(D,N) == 1)
    7019             :     {
    7020           7 :       GEN C = mului(N, C1), den;
    7021           7 :       den = gadd(gsqr(gmul(C,y)), gsqr(gadd(D, gmul(C,x))));
    7022           7 :       if (gcmp(den, ma) < 0) { ma = den; CK = C; DK = D; }
    7023             :     }
    7024             :   }
    7025          14 :   return DK? mkvec2(CK, DK): NULL;
    7026             : }
    7027             : 
    7028             : static long
    7029         168 : valNC2(GEN P, GEN E, long e)
    7030             : {
    7031         168 :   long i, d = 1, l = lg(P);
    7032         504 :   for (i = 1; i < l; i++)
    7033             :   {
    7034         336 :     long v = u_lval(e, P[i]) << 1;
    7035         336 :     if (v == E[i] + 1) v--;
    7036         336 :     d *= upowuu(P[i], v);
    7037             :   }
    7038         168 :   return d;
    7039             : }
    7040             : 
    7041             : static GEN
    7042          49 : findqganew(long N, GEN z)
    7043             : {
    7044          49 :   GEN MI, DI, x = real_i(z), y = imag_i(z), Ck = gen_0, Dk = gen_1, fa, P, E;
    7045             :   long i;
    7046          49 :   MI = uutoQ(1,N);
    7047          49 :   DI = mydivisorsu(mysqrtu(N));
    7048          49 :   fa = myfactoru(N); P = gel(fa,1); E = gel(fa,2);
    7049         217 :   for (i = 1; i < lg(DI); i++)
    7050             :   {
    7051         168 :     long e = DI[i], g;
    7052             :     GEN U, C, D, m;
    7053         168 :     (void)cxredsl2(gmulsg(e, z), &U);
    7054         168 :     C = gcoeff(U,2,1); if (!signe(C)) continue;
    7055         168 :     D = gcoeff(U,2,2);
    7056         168 :     g = ugcdiu(D,e);
    7057         168 :     if (g > 1) { C = muliu(C,e/g); D = diviuexact(D,g); } else C = muliu(C,e);
    7058         168 :     m = gadd(gsqr(gadd(gmul(C, x), D)), gsqr(gmul(C, y)));
    7059         168 :     m = gdivgu(m, valNC2(P, E, e));
    7060         168 :     if (gcmp(m, MI) < 0) { MI = m; Ck = C; Dk = D; }
    7061             :   }
    7062          49 :   return signe(Ck)? mkvec2(Ck, Dk): NULL;
    7063             : }
    7064             : 
    7065             : /* Return z' and U = [a,b;c,d] \in SL_2(Z), z' = U*z,
    7066             :  * Im(z')/width(U.oo) > sqrt(3)/(2N). Set *pczd = c*z+d */
    7067             : static GEN
    7068         182 : cxredga0N(long N, GEN z, GEN *pU, GEN *pczd, long flag)
    7069             : {
    7070         182 :   GEN v = NULL, A, B, C, D;
    7071             :   long e;
    7072         182 :   if (N == 1) return cxredsl2_i(z, pU, pczd);
    7073         140 :   e = gexpo(gel(z,2));
    7074         140 :   if (e < 0) z = gprec_wensure(z, precision(z) + nbits2extraprec(-e));
    7075         140 :   v = flag? findqganew(N,z): findqga(N,z);
    7076         140 :   if (!v) { *pU = matid(2); *pczd = gen_1; return z; }
    7077          56 :   C = gel(v,1);
    7078          56 :   D = gel(v,2);
    7079          56 :   if (!is_pm1(bezout(C,D, &B,&A))) pari_err_BUG("cxredga0N [gcd > 1]");
    7080          56 :   B = negi(B);
    7081          56 :   *pU = mkmat2(mkcol2(A,C), mkcol2(B,D));
    7082          56 :   *pczd = gadd(gmul(C,z), D);
    7083          56 :   return gdiv(gadd(gmul(A,z), B), *pczd);
    7084             : }
    7085             : 
    7086             : static GEN
    7087         161 : lfunthetaall(GEN b, GEN vL, GEN t, long bitprec)
    7088             : {
    7089         161 :   long i, l = lg(vL);
    7090         161 :   GEN v = cgetg(l, t_VEC);
    7091         350 :   for (i = 1; i < l; i++)
    7092             :   {
    7093         189 :     GEN T, L = gel(vL,i), a0 = gel(L,1), ldata = gel(L,2);
    7094         189 :     GEN van = gel(ldata_get_an(ldata),2);
    7095         189 :     if (lg(van) == 1)
    7096             :     {
    7097           0 :       T = gmul(b, a0);
    7098           0 :       if (isexactzero(T)) { GEN z = real_0_bit(-bitprec); T = mkcomplex(z,z); }
    7099             :     }
    7100             :     else
    7101             :     {
    7102         189 :       T = gmul2n(lfuntheta(ldata, t, 0, bitprec), -1);
    7103         189 :       T = gmul(b, gadd(a0, T));
    7104             :     }
    7105         189 :     gel(v,i) = T;
    7106             :   }
    7107         161 :   return l == 2? gel(v,1): v;
    7108             : }
    7109             : 
    7110             : /* P in ZX, irreducible */
    7111             : static GEN
    7112         182 : ZX_roots(GEN P, long prec)
    7113             : {
    7114         182 :   long d = degpol(P);
    7115         182 :   if (d == 1) return mkvec(gen_0);
    7116         182 :   if (d == 2 && isint1(gel(P,2)) && isintzero(gel(P,3)) && isint1(gel(P,4)))
    7117           7 :     return mkvec2(powIs(3), gen_I()); /* order as polroots */
    7118         294 :   return (ZX_sturm_irred(P) == d)? ZX_realroots_irred(P, prec)
    7119         294 :                                  : QX_complex_roots(P, prec);
    7120             : }
    7121             : /* initializations for RgX_RgV_eval / RgC_embed */
    7122             : static GEN
    7123         217 : rootspowers(GEN v)
    7124             : {
    7125         217 :   long i, l = lg(v);
    7126         217 :   GEN w = cgetg(l, t_VEC);
    7127         868 :   for (i = 1; i < l; i++) gel(w,i) = gpowers(gel(v,i), l-2);
    7128         217 :   return w;
    7129             : }
    7130             : /* mf embeddings attached to Q(chi)/(T), chi attached to cyclotomic P */
    7131             : static GEN
    7132         938 : getembed(GEN P, GEN T, GEN zcyclo, long prec)
    7133             : {
    7134             :   long i, l;
    7135             :   GEN v;
    7136         938 :   if (degpol(P) == 1) P = NULL; /* mfcharpol for quadratic char */
    7137         938 :   if (degpol(T) == 1) T = NULL; /* dim 1 orbit */
    7138         938 :   if (T && P)
    7139          35 :   { /* K(y) / (T(y)), K = Q(t)/(P) cyclotomic */
    7140          35 :     GEN vr = RgX_is_ZX(T)? ZX_roots(T,prec): roots(RgX_embed1(T,zcyclo), prec);
    7141          35 :     v = rootspowers(vr); l = lg(v);
    7142         105 :     for (i = 1; i < l; i++) gel(v,i) = mkcol3(P,zcyclo,gel(v,i));
    7143             :   }
    7144         903 :   else if (T)
    7145             :   { /* Q(y) / (T(y)), T noncyclotomic */
    7146         182 :     GEN vr = ZX_roots(T, prec);
    7147         182 :     v = rootspowers(vr); l = lg(v);
    7148         763 :     for (i = 1; i < l; i++) gel(v,i) = mkcol2(T, gel(v,i));
    7149             :   }
    7150             :   else /* cyclotomic or rational */
    7151         721 :     v = mkvec(P? mkvec2(P, zcyclo): cgetg(1,t_VEC));
    7152         938 :   return v;
    7153             : }
    7154             : static GEN
    7155         791 : grootsof1_CHI(GEN CHI, long prec)
    7156         791 : { return grootsof1(mfcharorder(CHI), prec); }
    7157             : /* return the [Q(F):Q(chi)] embeddings of F */
    7158             : static GEN
    7159         623 : mfgetembed(GEN F, long prec)
    7160             : {
    7161         623 :   GEN T = mf_get_field(F), CHI = mf_get_CHI(F), P = mfcharpol(CHI);
    7162         623 :   return getembed(P, T, grootsof1_CHI(CHI, prec), prec);
    7163             : }
    7164             : static GEN
    7165           7 : mfchiembed(GEN mf, long prec)
    7166             : {
    7167           7 :   GEN CHI = MF_get_CHI(mf), P = mfcharpol(CHI);
    7168           7 :   return getembed(P, pol_x(0), grootsof1_CHI(CHI, prec), prec);
    7169             : }
    7170             : /* mfgetembed for the successive eigenforms in MF_get_newforms */
    7171             : static GEN
    7172         161 : mfeigenembed(GEN mf, long prec)
    7173             : {
    7174         161 :   GEN vP = MF_get_fields(mf), vF = MF_get_newforms(mf);
    7175         161 :   GEN zcyclo, vE, CHI = MF_get_CHI(mf), P = mfcharpol(CHI);
    7176         161 :   long i, l = lg(vP);
    7177         161 :   vF = Q_remove_denom(liftpol_shallow(vF), NULL);
    7178         161 :   prec += nbits2extraprec(gexpo(vF));
    7179         161 :   zcyclo = grootsof1_CHI(CHI, prec);
    7180         161 :   vE = cgetg(l, t_VEC);
    7181         469 :   for (i = 1; i < l; i++) gel(vE,i) = getembed(P, gel(vP,i), zcyclo, prec);
    7182         161 :   return vE;
    7183             : }
    7184             : 
    7185             : static int
    7186          28 : checkPv(GEN P, GEN v)
    7187          28 : { return typ(P) == t_POL && is_vec_t(typ(v)) && lg(v)-1 >= degpol(P); }
    7188             : static int
    7189          28 : checkemb_i(GEN E)
    7190             : {
    7191          28 :   long t = typ(E), l = lg(E);
    7192          28 :   if (t == t_VEC) return l == 1 || (l == 3 && checkPv(gel(E,1), gel(E,2)));
    7193          21 :   if (t != t_COL) return 0;
    7194          21 :   if (l == 3) return checkPv(gel(E,1), gel(E,2));
    7195          21 :   return l == 4 && is_vec_t(typ(gel(E,2))) && checkPv(gel(E,1), gel(E,3));
    7196             : }
    7197             : static GEN
    7198          28 : anyembed(GEN v, GEN E)
    7199             : {
    7200          28 :   switch(typ(v))
    7201             :   {
    7202          21 :     case t_VEC: case t_COL: return mfvecembed(E, v);
    7203           7 :     case t_MAT: return mfmatembed(E, v);
    7204             :   }
    7205           0 :   return mfembed(E, v);
    7206             : }
    7207             : GEN
    7208          49 : mfembed0(GEN E, GEN v, long prec)
    7209             : {
    7210          49 :   pari_sp av = avma;
    7211          49 :   GEN mf, vE = NULL;
    7212          49 :   if (checkmf_i(E)) vE = mfgetembed(E, prec);
    7213          35 :   else if ((mf = checkMF_i(E))) vE = mfchiembed(mf, prec);
    7214          49 :   if (vE)
    7215             :   {
    7216          21 :     long i, l = lg(vE);
    7217             :     GEN w;
    7218          21 :     if (!v) return gc_GEN(av, l == 2? gel(vE,1): vE);
    7219           0 :     w = cgetg(l, t_VEC);
    7220           0 :     for (i = 1; i < l; i++) gel(w,i) = anyembed(v, gel(vE,i));
    7221           0 :     return gc_GEN(av, l == 2? gel(w,1): w);
    7222             :   }
    7223          28 :   if (!checkemb_i(E) || !v) pari_err_TYPE("mfembed", E);
    7224          28 :   return gc_GEN(av, anyembed(v,E));
    7225             : }
    7226             : 
    7227             : /* dummy lfun create for theta evaluation */
    7228             : static GEN
    7229         980 : mfthetaancreate(GEN van, GEN N, GEN k)
    7230             : {
    7231         980 :   GEN L = zerovec(6);
    7232         980 :   gel(L,1) = lfuntag(t_LFUN_GENERIC, van);
    7233         980 :   gel(L,3) = mkvec2(gen_0, gen_1);
    7234         980 :   gel(L,4) = k;
    7235         980 :   gel(L,5) = N; return L;
    7236             : }
    7237             : /* destroy van and prepare to evaluate theta(sigma(van)), for all sigma in
    7238             :  * embeddings vector vE */
    7239             : static GEN
    7240         357 : van_embedall(GEN van, GEN vE, GEN gN, GEN gk)
    7241             : {
    7242         357 :   GEN a0 = gel(van,1), vL;
    7243         357 :   long i, lE = lg(vE), l = lg(van);
    7244         357 :   van++; van[0] = evaltyp(t_VEC) | _evallg(l-1); /* remove a0 */
    7245         357 :   vL = cgetg(lE, t_VEC);
    7246         945 :   for (i = 1; i < lE; i++)
    7247             :   {
    7248         588 :     GEN E = gel(vE,i), v = mfvecembed(E, van);
    7249         588 :     gel(vL,i) = mkvec2(mfembed(E,a0), mfthetaancreate(v, gN, gk));
    7250             :   }
    7251         357 :   return vL;
    7252             : }
    7253             : 
    7254             : static int
    7255        1134 : cusp_AC(GEN cusp, long *A, long *C)
    7256             : {
    7257        1134 :   switch(typ(cusp))
    7258             :   {
    7259         140 :     case t_INFINITY: *A = 1; *C = 0; break;
    7260         301 :     case t_INT:  *A = itos(cusp); *C = 1; break;
    7261         448 :     case t_FRAC: *A = itos(gel(cusp, 1)); *C = itos(gel(cusp, 2)); break;
    7262         245 :     case t_REAL: case t_COMPLEX:
    7263         245 :       *A = 0; *C = 0;
    7264         245 :       if (gsigne(imag_i(cusp)) <= 0)
    7265           7 :         pari_err_DOMAIN("mfeval","imag(tau)","<=",gen_0,cusp);
    7266         238 :       return 0;
    7267           0 :     default: pari_err_TYPE("cusp_AC", cusp);
    7268             :   }
    7269         889 :   return 1;
    7270             : }
    7271             : static GEN
    7272         518 : cusp2mat(long A, long C)
    7273             : { long B, D;
    7274         518 :   cbezout(A, C, &D, &B);
    7275         518 :   return mkmat22s(A, -B, C, D);
    7276             : }
    7277             : static GEN
    7278          21 : mkS(void) { return mkmat22s(0,-1,1,0); }
    7279             : 
    7280             : /* if t is a cusp, return F(t), else NULL */
    7281             : static GEN
    7282         364 : evalcusp(GEN mf, GEN F, GEN t, long prec)
    7283             : {
    7284             :   long A, C;
    7285             :   GEN R;
    7286         364 :   if (!cusp_AC(t, &A,&C)) return NULL;
    7287         196 :   if (C % mf_get_N(F) == 0) return gel(mfcoefs_i(F, 0, 1), 1);
    7288         175 :   R = mfgaexpansion(mf, F, cusp2mat(A,C), 0, prec);
    7289         175 :   return gequal0(gel(R,1))? gmael(R,3,1): gen_0;
    7290             : }
    7291             : /* Evaluate an mf closure numerically, i.e., in the usual sense, either for a
    7292             :  * single tau or a vector of tau; for each, return a vector of results
    7293             :  * corresponding to all complex embeddings of F. If flag is nonzero, allow
    7294             :  * replacing F by F | gamma to increase imag(gamma^(-1).tau) [ expensive if
    7295             :  * MF_EISENSPACE not present ] */
    7296             : static GEN
    7297         168 : mfeval_i(GEN mf, GEN F, GEN vtau, long flag, long bitprec)
    7298             : {
    7299             :   GEN L0, vL, vb, sqN, vczd, vTAU, vs, van, vE;
    7300         168 :   long N = MF_get_N(mf), N0, ta, lv, i, prec = nbits2prec(bitprec);
    7301         168 :   GEN gN = utoipos(N), gk = mf_get_gk(F), gk1 = gsubgs(gk,1), vgk;
    7302         168 :   long flscal = 0;
    7303             : 
    7304             :   /* gen_0 is ignored, second component assumes Ramanujan-Petersson in
    7305             :    * 1/2-integer weight */
    7306         168 :   vgk = mkvec2(gen_0, mfiscuspidal(mf,F)? gmul2n(gk1,-1): gk1);
    7307         168 :   ta = typ(vtau);
    7308         168 :   if (!is_vec_t(ta)) { flscal = 1; vtau = mkvec(vtau); ta = t_VEC; }
    7309         168 :   lv = lg(vtau);
    7310         168 :   sqN = sqrtr_abs(utor(N, prec));
    7311         168 :   vs = const_vec(lv-1, NULL);
    7312         168 :   vb = const_vec(lv-1, NULL);
    7313         168 :   vL = cgetg(lv, t_VEC);
    7314         168 :   vTAU = cgetg(lv, t_VEC);
    7315         168 :   vczd = cgetg(lv, t_VEC);
    7316         168 :   L0 = mfthetaancreate(NULL, gN, vgk); /* only for thetacost */
    7317         168 :   vE = mfgetembed(F, prec);
    7318         168 :   N0 = 0;
    7319         357 :   for (i = 1; i < lv; i++)
    7320             :   {
    7321         196 :     GEN z = gel(vtau,i), tau, U;
    7322             :     long w, n;
    7323             : 
    7324         196 :     gel(vs,i) = evalcusp(mf, F, z, prec);
    7325         189 :     if (gel(vs,i)) continue;
    7326         161 :     tau = cxredga0N(N, z, &U, &gel(vczd,i), flag);
    7327         161 :     if (!flag) w = 0; else { w = mfZC_width(N, gel(U,1)); tau = gdivgu(tau,w); }
    7328         161 :     gel(vTAU,i) = mulcxmI(gmul(tau, sqN));
    7329         161 :     n = lfunthetacost(L0, real_i(gel(vTAU,i)), 0, bitprec, NULL);
    7330         161 :     if (N0 < n) N0 = n;
    7331         161 :     if (flag)
    7332             :     {
    7333          49 :       GEN A, al, v = mfslashexpansion(mf, F, ZM_inv(U,NULL), n, 0, &A, prec);
    7334          49 :       gel(vL,i) = van_embedall(v, vE, gN, vgk);
    7335          49 :       al = gel(A,1);
    7336          49 :       if (!gequal0(al))
    7337           7 :         gel(vb,i) = gexp(gmul(gmul(gmulsg(w,al),PiI2(prec)), tau), prec);
    7338             :     }
    7339             :   }
    7340         161 :   if (!flag)
    7341             :   {
    7342         112 :     van = mfcoefs_i(F, N0, 1);
    7343         112 :     vL = const_vec(lv-1, van_embedall(van, vE, gN, vgk));
    7344             :   }
    7345         350 :   for (i = 1; i < lv; i++)
    7346             :   {
    7347             :     GEN T;
    7348         189 :     if (gel(vs,i)) continue;
    7349         161 :     T = gpow(gel(vczd,i), gneg(gk), prec);
    7350         161 :     if (flag && gel(vb,i)) T = gmul(T, gel(vb,i));
    7351         161 :     gel(vs,i) = lfunthetaall(T, gel(vL,i), gel(vTAU,i), bitprec);
    7352             :   }
    7353         161 :   return flscal? gel(vs,1): vs;
    7354             : }
    7355             : 
    7356             : static long
    7357        1372 : mfistrivial(GEN F)
    7358             : {
    7359        1372 :   switch(mf_get_type(F))
    7360             :   {
    7361           7 :     case t_MF_CONST: return lg(gel(F,2)) == 1;
    7362         287 :     case t_MF_LINEAR: case t_MF_LINEAR_BHN: return gequal0(gel(F,3));
    7363        1078 :     default: return 0;
    7364             :   }
    7365             : }
    7366             : 
    7367             : static long
    7368        1190 : mf_same_k(GEN mf, GEN f) { return gequal(MF_get_gk(mf), mf_get_gk(f)); }
    7369             : static long
    7370        1148 : mf_same_CHI(GEN mf, GEN f)
    7371             : {
    7372        1148 :   GEN F1, F2, chi1, chi2, CHI1 = MF_get_CHI(mf), CHI2 = mf_get_CHI(f);
    7373             :   /* are the primitive chars attached to CHI1 and CHI2 equal ? */
    7374        1148 :   F1 = znconreyconductor(gel(CHI1,1), gel(CHI1,2), &chi1);
    7375        1148 :   if (typ(F1) == t_VEC) F1 = gel(F1,1);
    7376        1148 :   F2 = znconreyconductor(gel(CHI2,1), gel(CHI2,2), &chi2);
    7377        1148 :   if (typ(F2) == t_VEC) F2 = gel(F2,1);
    7378        1148 :   return equalii(F1,F2) && ZV_equal(chi1,chi2);
    7379             : }
    7380             : /* check k and CHI rigorously, but not coefficients nor N */
    7381             : static long
    7382         259 : mfisinspace_i(GEN mf, GEN F)
    7383             : {
    7384         259 :   return mfistrivial(F) || (mf_same_k(mf,F) && mf_same_CHI(mf,F));
    7385             : }
    7386             : static void
    7387           7 : err_space(GEN F)
    7388           7 : { pari_err_DOMAIN("mftobasis", "form", "does not belong to",
    7389           0 :                   strtoGENstr("space"), F); }
    7390             : 
    7391             : static long
    7392         154 : mfcheapeisen(GEN mf)
    7393             : {
    7394         154 :   long k, L, N = MF_get_N(mf);
    7395             :   GEN P;
    7396         154 :   if (N <= 70) return 1;
    7397          84 :   k = itos(gceil(MF_get_gk(mf)));
    7398          84 :   if (odd(k)) k--;
    7399          84 :   switch (k)
    7400             :   {
    7401           0 :     case 2:  L = 190; break;
    7402          14 :     case 4:  L = 162; break;
    7403          70 :     case 6:
    7404          70 :     case 8:  L = 88; break;
    7405           0 :     case 10: L = 78; break;
    7406           0 :     default: L = 66; break;
    7407             :   }
    7408          84 :   P = gel(myfactoru(N), 1);
    7409          84 :   return P[lg(P)-1] <= L;
    7410             : }
    7411             : 
    7412             : static GEN
    7413         189 : myimag_i(GEN x)
    7414             : {
    7415         189 :   long tc = typ(x);
    7416         189 :   if (tc == t_INFINITY || tc == t_INT || tc == t_FRAC) return gen_1;
    7417         196 :   if (tc == t_VEC) pari_APPLY_same(myimag_i(gel(x,i)));
    7418         154 :   return imag_i(x);
    7419             : }
    7420             : 
    7421             : static GEN
    7422         154 : mintau(GEN vtau)
    7423             : {
    7424         154 :   if (!is_vec_t(typ(vtau))) return myimag_i(vtau);
    7425           7 :   return (lg(vtau) == 1)? gen_1: vecmin(myimag_i(vtau));
    7426             : }
    7427             : 
    7428             : /* initialization for mfgaexpansion: what does not depend on cusp */
    7429             : static GEN
    7430        1218 : mf_eisendec(GEN mf, GEN F, long prec)
    7431             : {
    7432        1218 :   GEN B = liftpol_shallow(mfeisensteindec(mf, F)), v = variables_vecsmall(B);
    7433        1218 :   GEN Mvecj = obj_check(mf, MF_EISENSPACE);
    7434        1218 :   long l = lg(v), i, ord;
    7435        1218 :   if (lg(Mvecj) < 5) Mvecj = gel(Mvecj,1);
    7436        1218 :   ord = itou(gel(Mvecj,4));
    7437        1274 :   for (i = 1; i < l; i++)
    7438         924 :     if (v[i] != 1)
    7439             :     {
    7440             :       GEN d;
    7441             :       long e;
    7442         868 :       B = Q_remove_denom(B, &d);
    7443         868 :       e = gexpo(B);
    7444         868 :       if (e > 0) prec += nbits2prec(e);
    7445         868 :       B = gsubst(B, v[i], rootsof1u_cx(ord, prec));
    7446         868 :       if (d) B = gdiv(B, d);
    7447         868 :       break;
    7448             :     }
    7449        1218 :   return B;
    7450             : }
    7451             : 
    7452             : GEN
    7453         168 : mfeval(GEN mf0, GEN F, GEN vtau, long bitprec)
    7454             : {
    7455         168 :   pari_sp av = avma;
    7456         168 :   long flnew = 1;
    7457         168 :   GEN mf = checkMF_i(mf0);
    7458         168 :   if (!mf) pari_err_TYPE("mfeval", mf0);
    7459         168 :   if (!checkmf_i(F)) pari_err_TYPE("mfeval", F);
    7460         168 :   if (!mfisinspace_i(mf, F)) err_space(F);
    7461         168 :   if (!obj_check(mf, MF_EISENSPACE)) flnew = mfcheapeisen(mf);
    7462         168 :   if (flnew && gcmpgs(gmulsg(2*MF_get_N(mf), mintau(vtau)), 1) >= 0) flnew = 0;
    7463         168 :   return gc_GEN(av, mfeval_i(mf, F, vtau, flnew, bitprec));
    7464             : }
    7465             : 
    7466             : static long
    7467         189 : val(GEN v, long bit)
    7468             : {
    7469         189 :   long c, l = lg(v);
    7470         392 :   for (c = 1; c < l; c++)
    7471         378 :     if (gexpo(gel(v,c)) > -bit) return c-1;
    7472          14 :   return -1;
    7473             : }
    7474             : GEN
    7475         203 : mfcuspval(GEN mf, GEN F, GEN cusp, long bitprec)
    7476             : {
    7477         203 :   pari_sp av = avma;
    7478         203 :   long lvE, w, N, sb, n, A, C, prec = nbits2prec(bitprec);
    7479             :   GEN ga, gk, vE;
    7480         203 :   mf = checkMF(mf);
    7481         203 :   if (!checkmf_i(F)) pari_err_TYPE("mfcuspval",F);
    7482         203 :   N = MF_get_N(mf);
    7483         203 :   cusp_canon(cusp, N, &A, &C);
    7484         203 :   gk = mf_get_gk(F);
    7485         203 :   if (typ(gk) != t_INT)
    7486             :   {
    7487          42 :     GEN FT = mfmultheta(F), mf2 = obj_checkbuild(mf, MF_MF2INIT, &mf2init);
    7488          42 :     GEN r = mfcuspval(mf2, FT, cusp, bitprec);
    7489          42 :     if ((C & 3L) == 2)
    7490             :     {
    7491          14 :       GEN z = uutoQ(1,4);
    7492          14 :       r = gsub(r, typ(r) == t_VEC? const_vec(lg(r)-1, z): z);
    7493             :     }
    7494          42 :     return gc_upto(av, r);
    7495             :   }
    7496         161 :   vE = mfgetembed(F, prec);
    7497         161 :   lvE = lg(vE);
    7498         161 :   w = mfcuspcanon_width(N, C);
    7499         161 :   sb = w * mfsturmNk(N, itos(gk));
    7500         161 :   ga = cusp2mat(A,C);
    7501         168 :   for (n = 8;; n = minss(sb, n << 1))
    7502           7 :   {
    7503         168 :     GEN R = mfgaexpansion(mf, F, ga, n, prec), res = liftpol_shallow(gel(R,3));
    7504         168 :     GEN v = cgetg(lvE-1, t_VECSMALL);
    7505         168 :     long j, ok = 1;
    7506         168 :     res = RgC_embedall(res, vE);
    7507         357 :     for (j = 1; j < lvE; j++)
    7508             :     {
    7509         189 :       v[j] = val(gel(res,j), bitprec/2);
    7510         189 :       if (v[j] < 0) ok = 0;
    7511             :     }
    7512         168 :     if (ok)
    7513             :     {
    7514         154 :       res = cgetg(lvE, t_VEC);
    7515         329 :       for (j = 1; j < lvE; j++) gel(res,j) = gadd(gel(R,1), uutoQ(v[j], w));
    7516         154 :       return gc_GEN(av, lvE==2? gel(res,1): res);
    7517             :     }
    7518          14 :     if (n == sb) return lvE==2? mkoo(): const_vec(lvE-1, mkoo()); /* 0 */
    7519             :   }
    7520             : }
    7521             : 
    7522             : long
    7523         238 : mfiscuspidal(GEN mf, GEN F)
    7524             : {
    7525         238 :   pari_sp av = avma;
    7526             :   GEN mf2;
    7527         238 :   if (space_is_cusp(MF_get_space(mf))) return 1;
    7528         105 :   if (typ(mf_get_gk(F)) == t_INT)
    7529             :   {
    7530          63 :     GEN v = mftobasis(mf,F,0), vE;
    7531          63 :     if (lg(v)==1) return gc_long(av, 0);
    7532          63 :     vE = vecslice(v, 1, lg(MF_get_E(mf))-1);
    7533          63 :     return gc_long(av, gequal0(vE));
    7534             :   }
    7535          42 :   if (!gequal0(mfak_i(F, 0))) return 0;
    7536          21 :   mf2 = obj_checkbuild(mf, MF_MF2INIT, &mf2init);
    7537          21 :   return mfiscuspidal(mf2, mfmultheta(F));
    7538             : }
    7539             : 
    7540             : /* F = vector of newforms in mftobasis format */
    7541             : static GEN
    7542         119 : mffrickeeigen_i(GEN mf, GEN F, GEN vE, long prec)
    7543             : {
    7544         119 :   GEN M, Z, L0, gN = MF_get_gN(mf), gk = MF_get_gk(mf);
    7545         119 :   long N0, i, lM, bit = prec2nbits(prec), k = itou(gk);
    7546         119 :   long LIM = 5; /* Sturm bound is enough */
    7547             : 
    7548         119 :   L0 = mfthetaancreate(NULL, gN, gk); /* only for thetacost */
    7549         119 : START:
    7550         119 :   N0 = lfunthetacost(L0, gen_1, LIM, bit, NULL);
    7551         119 :   M = mfcoefs_mf(mf, N0, 1);
    7552         119 :   lM = lg(F);
    7553         119 :   Z = cgetg(lM, t_VEC);
    7554         315 :   for (i = 1; i < lM; i++)
    7555             :   { /* expansion of D * F[i] */
    7556         196 :     GEN D, z, van = RgM_RgC_mul(M, Q_remove_denom(gel(F,i), &D));
    7557         196 :     GEN L = van_embedall(van, gel(vE,i), gN, gk);
    7558         196 :     long l = lg(L), j, bit_add = D? expi(D): 0;
    7559         196 :     gel(Z,i) = z = cgetg(l, t_VEC);
    7560         595 :     for (j = 1; j < l; j++)
    7561             :     {
    7562             :       GEN v, C, C0;
    7563             :       long m, e;
    7564         546 :       for (m = 0; m <= LIM; m++)
    7565             :       {
    7566         546 :         v = lfuntheta(gmael(L,j,2), gen_1, m, bit);
    7567         546 :         if (gexpo(v) > bit_add - bit/2) break;
    7568             :       }
    7569         399 :       if (m > LIM) { LIM <<= 1; goto START; }
    7570         399 :       C = mulcxpowIs(gdiv(v,conj_i(v)), 2*m - k);
    7571         399 :       C0 = grndtoi(C, &e); if (e < 5-prec2nbits(precision(C))) C = C0;
    7572         399 :       gel(z,j) = C;
    7573             :     }
    7574             :   }
    7575         119 :   return Z;
    7576             : }
    7577             : static GEN
    7578          84 : mffrickeeigen(GEN mf, GEN vE, long prec)
    7579             : {
    7580          84 :   GEN D = obj_check(mf, MF_FRICKE);
    7581          84 :   if (D) { long p = gprecision(D); if (!p || p >= prec) return D; }
    7582          77 :   D = mffrickeeigen_i(mf, MF_get_newforms(mf), vE, prec);
    7583          77 :   return obj_insert(mf, MF_FRICKE, D);
    7584             : }
    7585             : 
    7586             : /* integral weight, new space for primitive quadratic character CHIP;
    7587             :  * MF = vector of embedded eigenforms coefs on mfbasis, by orbit.
    7588             :  * Assume N > Q > 1 and (Q,f(CHIP)) = 1 */
    7589             : static GEN
    7590          56 : mfatkineigenquad(GEN mf, GEN CHIP, long Q, GEN MF, long bitprec)
    7591             : {
    7592             :   GEN L0, la2, S, F, vP, tau, wtau, Z, va, vb, den, coe, sqrtQ, sqrtN;
    7593          56 :   GEN M, gN, gk = MF_get_gk(mf);
    7594          56 :   long N0, x, yq, i, j, lF, dim, muQ, prec = nbits2prec(bitprec);
    7595          56 :   long N = MF_get_N(mf), k = itos(gk), NQ = N / Q;
    7596             : 
    7597             :   /* Q coprime to FC */
    7598          56 :   F = MF_get_newforms(mf);
    7599          56 :   vP = MF_get_fields(mf);
    7600          56 :   lF = lg(F);
    7601          56 :   Z = cgetg(lF, t_VEC);
    7602          56 :   S = MF_get_S(mf); dim = lg(S) - 1;
    7603          56 :   muQ = mymoebiusu(Q);
    7604          56 :   if (muQ)
    7605             :   {
    7606          42 :     GEN SQ = cgetg(dim+1,t_VEC), Qk = gpow(stoi(Q), sstoQ(k-2, 2), prec);
    7607          42 :     long i, bit2 = bitprec >> 1;
    7608         154 :     for (j = 1; j <= dim; j++) gel(SQ,j) = mfak_i(gel(S,j), Q);
    7609          84 :     for (i = 1; i < lF; i++)
    7610             :     {
    7611          42 :       GEN S = RgV_dotproduct(gel(F,i), SQ), T = gel(vP,i);
    7612             :       long e;
    7613          42 :       if (degpol(T) > 1 && typ(S) != t_POLMOD) S = gmodulo(S, T);
    7614          42 :       S = grndtoi(gdiv(conjvec(S, prec), Qk), &e);
    7615          42 :       if (e > -bit2) pari_err_PREC("mfatkineigenquad");
    7616          42 :       if (muQ == -1) S = gneg(S);
    7617          42 :       gel(Z,i) = S;
    7618             :     }
    7619          42 :     return Z;
    7620             :   }
    7621          14 :   la2 = mfchareval(CHIP, Q); /* 1 or -1 */
    7622          14 :   (void)cbezout(Q, NQ, &x, &yq);
    7623          14 :   sqrtQ = sqrtr_abs(utor(Q,prec));
    7624          14 :   tau = mkcomplex(gadd(sstoQ(-1, NQ), uutoQ(1, 1000)),
    7625             :                   divru(sqrtQ, N));
    7626          14 :   den = gaddgs(gmulsg(NQ, tau), 1);
    7627          14 :   wtau = gdiv(gsub(gmulsg(x, tau), sstoQ(yq, Q)), den);
    7628          14 :   coe = gpowgs(gmul(sqrtQ, den), k);
    7629             : 
    7630          14 :   sqrtN = sqrtr_abs(utor(N,prec));
    7631          14 :   tau  = mulcxmI(gmul(tau,  sqrtN));
    7632          14 :   wtau = mulcxmI(gmul(wtau, sqrtN));
    7633          14 :   gN = utoipos(N);
    7634          14 :   L0 = mfthetaancreate(NULL, gN, gk); /* only for thetacost */
    7635          14 :   N0 = maxss(lfunthetacost(L0,real_i(tau), 0,bitprec, NULL),
    7636             :              lfunthetacost(L0,real_i(wtau),0,bitprec, NULL));
    7637          14 :   M = mfcoefs_mf(mf, N0, 1);
    7638          14 :   va = cgetg(dim+1, t_VEC);
    7639          14 :   vb = cgetg(dim+1, t_VEC);
    7640         105 :   for (j = 1; j <= dim; j++)
    7641             :   {
    7642          91 :     GEN L, v = vecslice(gel(M,j), 2, N0+1); /* remove a0 */
    7643          91 :     settyp(v, t_VEC); L = mfthetaancreate(v, gN, gk);
    7644          91 :     gel(va,j) = lfuntheta(L, tau,0,bitprec);
    7645          91 :     gel(vb,j) = lfuntheta(L,wtau,0,bitprec);
    7646             :   }
    7647          84 :   for (i = 1; i < lF; i++)
    7648             :   {
    7649          70 :     GEN z, FE = gel(MF,i);
    7650          70 :     long l = lg(FE);
    7651          70 :     z = cgetg(l, t_VEC);
    7652          70 :     for (j = 1; j < l; j++)
    7653             :     {
    7654          70 :       GEN f = gel(FE,j), a = RgV_dotproduct(va,f), b = RgV_dotproduct(vb,f);
    7655          70 :       GEN la = ground( gdiv(b, gmul(a,coe)) );
    7656          70 :       if (!gequal(gsqr(la), la2)) pari_err_PREC("mfatkineigenquad");
    7657          70 :       if (typ(la) == t_INT)
    7658             :       {
    7659          70 :         if (j != 1) pari_err_BUG("mfatkineigenquad");
    7660          70 :         z = const_vec(l-1, la); break;
    7661             :       }
    7662           0 :       gel(z,j) = la;
    7663             :     }
    7664          70 :     gel(Z,i) = z;
    7665             :   }
    7666          14 :   return Z;
    7667             : }
    7668             : 
    7669             : static GEN
    7670          84 : myusqrt(ulong a, long prec)
    7671             : {
    7672          84 :   if (a == 1UL) return gen_1;
    7673          70 :   if (uissquareall(a, &a)) return utoipos(a);
    7674          49 :   return sqrtr_abs(utor(a, prec));
    7675             : }
    7676             : /* Assume mf is a nontrivial new space, rational primitive character CHIP
    7677             :  * and (Q,FC) = 1 */
    7678             : static GEN
    7679         112 : mfatkinmatnewquad(GEN mf, GEN CHIP, long Q, long flag, long PREC)
    7680             : {
    7681         112 :   GEN cM, M, D, MF, den, vE, F = MF_get_newforms(mf);
    7682         112 :   long i, c, e, prec, bitprec, lF = lg(F), N = MF_get_N(mf), k = MF_get_k(mf);
    7683             : 
    7684         112 :   if (Q == 1) return mkvec4(gen_0, matid(MF_get_dim(mf)), gen_1, mf);
    7685         112 :   den = gel(MF_get_Minv(mf), 2);
    7686         112 :   bitprec = expi(den) + 64;
    7687         112 :   if (!flag) bitprec = maxss(bitprec, prec2nbits(PREC));
    7688             : 
    7689          35 : START:
    7690         112 :   prec = nbits2prec(bitprec);
    7691         112 :   vE = mfeigenembed(mf, prec);
    7692         112 :   M = cgetg(lF, t_VEC);
    7693         294 :   for (i = 1; i < lF; i++) gel(M,i) = RgC_embedall(gel(F,i), gel(vE,i));
    7694         112 :   if (Q != N)
    7695             :   {
    7696          56 :     D = mfatkineigenquad(mf, CHIP, Q, M, bitprec);
    7697          56 :     c = odd(k)? Q: 1;
    7698             :   }
    7699             :   else
    7700             :   {
    7701          56 :     D = mffrickeeigen(mf, vE, prec);
    7702          56 :     c = mfcharmodulus(CHIP); if (odd(k)) c = -Q/c;
    7703             :   }
    7704         112 :   D = shallowconcat1(D);
    7705         112 :   if (vec_isconst(D)) { MF = diagonal_shallow(D); flag = 0; }
    7706             :   else
    7707             :   {
    7708          63 :     M = shallowconcat1(M);
    7709          63 :     MF = RgM_mul(matmuldiagonal(M,D), ginv(M));
    7710             :   }
    7711         112 :   if (!flag) return mkvec4(gen_0, MF, gen_1, mf);
    7712             : 
    7713          21 :   if (c > 0)
    7714          21 :     cM = myusqrt(c, PREC);
    7715             :   else
    7716             :   {
    7717           0 :     MF = imag_i(MF); c = -c;
    7718           0 :     cM = mkcomplex(gen_0, myusqrt(c,PREC));
    7719             :   }
    7720          21 :   if (c != 1) MF = RgM_Rg_mul(MF, myusqrt(c,prec));
    7721          21 :   MF = grndtoi(RgM_Rg_mul(MF,den), &e);
    7722          21 :   if (e > -32) { bitprec <<= 1; goto START; }
    7723          21 :   MF = RgM_Rg_div(MF, den);
    7724          21 :   if (is_rational_t(typ(cM)) && !isint1(cM))
    7725           0 :   { MF = RgM_Rg_div(MF, cM); cM = gen_1; }
    7726          21 :   return mkvec4(gen_0, MF, cM, mf);
    7727             : }
    7728             : 
    7729             : /* let CHI mod N, Q || N, return \bar{CHI_Q} * CHI_{N/Q} */
    7730             : static GEN
    7731         112 : mfcharAL(GEN CHI, long Q)
    7732             : {
    7733         112 :   GEN G = gel(CHI,1), c = gel(CHI,2), cycc, d, P, E, F;
    7734         112 :   long l = lg(c), N = mfcharmodulus(CHI), i;
    7735         112 :   if (N == Q) return mfcharconj(CHI);
    7736          56 :   if (N == 1) return CHI;
    7737          42 :   CHI = leafcopy(CHI);
    7738          42 :   gel(CHI,2) = d = leafcopy(c);
    7739          42 :   F = znstar_get_faN(G);
    7740          42 :   P = gel(F,1);
    7741          42 :   E = gel(F,2);
    7742          42 :   cycc = znstar_get_conreycyc(G);
    7743          42 :   if (!odd(Q) && equaliu(gel(P,1), 2) && E[1] >= 3)
    7744          14 :     gel(d,2) = Fp_neg(gel(d,2), gel(cycc,2));
    7745          56 :   else for (i = 1; i < l; i++)
    7746          28 :     if (!umodui(Q, gel(P,i))) gel(d,i) = Fp_neg(gel(d,i), gel(cycc,i));
    7747          42 :   return CHI;
    7748             : }
    7749             : static long
    7750         245 : atkin_get_NQ(long N, long Q, const char *f)
    7751             : {
    7752         245 :   long NQ = N / Q;
    7753         245 :   if (N % Q) pari_err_DOMAIN(f,"N % Q","!=",gen_0,utoi(Q));
    7754         245 :   if (ugcd(NQ, Q) > 1) pari_err_DOMAIN(f,"gcd(Q,N/Q)","!=",gen_1,utoi(Q));
    7755         245 :   return NQ;
    7756             : }
    7757             : 
    7758             : /* transform mf to new_NEW if possible */
    7759             : static GEN
    7760        1589 : MF_set_new(GEN mf)
    7761             : {
    7762        1589 :   GEN vMjd, vj, gk = MF_get_gk(mf);
    7763             :   long l, j;
    7764        1589 :   if (MF_get_space(mf) != mf_CUSP
    7765        1589 :       || typ(gk) != t_INT || itou(gk) == 1) return mf;
    7766         182 :   vMjd = MFcusp_get_vMjd(mf); l = lg(vMjd);
    7767         182 :   if (l > 1 && gel(vMjd,1)[1] != MF_get_N(mf)) return mf; /* oldspace != 0 */
    7768         175 :   mf = shallowcopy(mf);
    7769         175 :   gel(mf,1) = shallowcopy(gel(mf,1));
    7770         175 :   MF_set_space(mf, mf_NEW);
    7771         175 :   vj = cgetg(l, t_VECSMALL);
    7772         938 :   for (j = 1; j < l; j++) vj[j] = gel(vMjd, j)[2];
    7773         175 :   gel(mf,4) = vj; return mf;
    7774             : }
    7775             : 
    7776             : /* if flag = 1, rationalize, else don't */
    7777             : static GEN
    7778         224 : mfatkininit_i(GEN mf, long Q, long flag, long prec)
    7779             : {
    7780             :   GEN M, B, C, CHI, CHIAL, G, chi, P, z, g, mfB, s, Mindex, Minv;
    7781         224 :   long j, l, lim, ord, FC, NQ, cQ, nk, dk, N = MF_get_N(mf);
    7782             : 
    7783         224 :   B = MF_get_basis(mf); l = lg(B);
    7784         224 :   M = cgetg(l, t_MAT); if (l == 1) return mkvec4(gen_0,M,gen_1,mf);
    7785         224 :   Qtoss(MF_get_gk(mf), &nk,&dk);
    7786         224 :   Q = labs(Q);
    7787         224 :   NQ = atkin_get_NQ(N, Q, "mfatkininit");
    7788         224 :   CHI = MF_get_CHI(mf);
    7789         224 :   CHI = mfchartoprimitive(CHI, &FC);
    7790         224 :   ord = mfcharorder(CHI);
    7791         224 :   mf = MF_set_new(mf);
    7792         224 :   if (MF_get_space(mf) == mf_NEW && ord <= 2 && NQ % FC == 0 && dk == 1)
    7793         112 :     return mfatkinmatnewquad(mf, CHI, Q, flag, prec);
    7794             :   /* now flag != 0 */
    7795         112 :   G   = gel(CHI,1);
    7796         112 :   chi = gel(CHI,2);
    7797         112 :   if (Q == N) { g = mkmat22s(0, -1, N, 0); cQ = NQ; } /* Fricke */
    7798             :   else
    7799             :   {
    7800          28 :     GEN F, gQP = utoi(ugcd(Q, FC));
    7801             :     long t, v;
    7802          28 :     chi = znchardecompose(G, chi, gQP);
    7803          28 :     F = znconreyconductor(G, chi, &chi);
    7804          28 :     G = znstar0(F,1);
    7805          28 :     (void)cbezout(Q, NQ, &t, &v);
    7806          28 :     g = mkmat22s(Q*t, 1, -N*v, Q);
    7807          28 :     cQ = -NQ*v;
    7808             :   }
    7809         112 :   C = s = gen_1;
    7810             :   /* N.B. G,chi are G_Q,chi_Q [primitive] at this point */
    7811         112 :   if (lg(chi) != 1) C = ginv( znchargauss(G, chi, gen_1, prec2nbits(prec)) );
    7812         112 :   if (dk == 1)
    7813          91 :   { if (odd(nk)) s = myusqrt(Q,prec); }
    7814             :   else
    7815             :   {
    7816          21 :     long r = nk >> 1; /* k-1/2 */
    7817          21 :     s = gpow(utoipos(Q), mkfracss(odd(r)? 1: 3, 4), prec);
    7818          21 :     if (odd(cQ))
    7819             :     {
    7820          21 :       long t = r + ((cQ-1) >> 1);
    7821          21 :       s = mkcomplex(s, odd(t)? gneg(s): s);
    7822             :     }
    7823             :   }
    7824         112 :   if (!isint1(s)) C = gmul(C, s);
    7825         112 :   CHIAL = mfcharAL(CHI, Q);
    7826         112 :   if (dk == 2)
    7827             :   {
    7828          21 :     ulong q = odd(Q)? Q << 2: Q, Nq = ulcm(q, mfcharmodulus(CHIAL));
    7829          21 :     CHIAL = induceN(Nq, CHIAL);
    7830          21 :     CHIAL = mfcharmul(CHIAL, induce(gel(CHIAL,1), utoipos(q)));
    7831             :   }
    7832         112 :   CHIAL = mfchartoprimitive(CHIAL,NULL);
    7833         112 :   mfB = gequal(CHIAL,CHI)? mf: mfinit_Nndkchi(N,nk,dk,CHIAL,MF_get_space(mf),0);
    7834         112 :   Mindex = MF_get_Mindex(mfB);
    7835         112 :   Minv = MF_get_Minv(mfB);
    7836         112 :   P = z = NULL;
    7837         112 :   if (ord > 2) { P = mfcharpol(CHI); z = rootsof1u_cx(ord, prec); }
    7838         112 :   lim = maxss(mfsturm(mfB), mfsturm(mf)) + 1;
    7839         567 :   for (j = 1; j < l; j++)
    7840             :   {
    7841         455 :     GEN v = mfslashexpansion(mf, gel(B,j), g, lim, 0, NULL, prec+EXTRAPREC64);
    7842             :     long junk;
    7843         455 :     if (!isint1(C)) v = RgV_Rg_mul(v, C);
    7844         455 :     v = bestapprnf(v, P, z, prec);
    7845         455 :     v = vecpermute_partial(v, Mindex, &junk);
    7846         455 :     v = Minv_RgC_mul(Minv, v); /* cf mftobasis_i */
    7847         455 :     gel(M, j) = v;
    7848             :   }
    7849         112 :   if (is_rational_t(typ(C)) && !gequal1(C)) { M = gdiv(M, C); C = gen_1; }
    7850         112 :   if (mfB == mf) mfB = gen_0;
    7851         112 :   return mkvec4(mfB, M, C, mf);
    7852             : }
    7853             : GEN
    7854          98 : mfatkininit(GEN mf, long Q, long prec)
    7855             : {
    7856          98 :   pari_sp av = avma;
    7857          98 :   mf = checkMF(mf); return gc_GEN(av, mfatkininit_i(mf, Q, 1, prec));
    7858             : }
    7859             : static void
    7860          63 : checkmfa(GEN z)
    7861             : {
    7862          63 :   if (typ(z) != t_VEC || lg(z) != 5 || typ(gel(z,2)) != t_MAT
    7863          63 :       || !checkMF_i(gel(z,4))
    7864          63 :       || (!isintzero(gel(z,1)) && !checkMF_i(gel(z,1))))
    7865           0 :     pari_err_TYPE("mfatkin [please apply mfatkininit()]",z);
    7866          63 : }
    7867             : 
    7868             : /* Apply atkin Q to closure F */
    7869             : GEN
    7870          63 : mfatkin(GEN mfa, GEN F)
    7871             : {
    7872          63 :   pari_sp av = avma;
    7873             :   GEN z, mfB, MQ, mf;
    7874          63 :   checkmfa(mfa);
    7875          63 :   mfB= gel(mfa,1);
    7876          63 :   MQ = gel(mfa,2);
    7877          63 :   mf = gel(mfa,4);
    7878          63 :   if (typ(mfB) == t_INT) mfB = mf;
    7879          63 :   z = RgM_RgC_mul(MQ, mftobasis_i(mf,F));
    7880          63 :   return gc_upto(av, mflinear(mfB, z));
    7881             : }
    7882             : 
    7883             : GEN
    7884          49 : mfatkineigenvalues(GEN mf, long Q, long prec)
    7885             : {
    7886          49 :   pari_sp av = avma;
    7887             :   GEN vF, L, CHI, M, mfatk, C, MQ, vE, mfB;
    7888             :   long N, NQ, l, i;
    7889             : 
    7890          49 :   mf = checkMF(mf); N = MF_get_N(mf);
    7891          49 :   vF = MF_get_newforms(mf); l = lg(vF);
    7892             :   /* N.B. k is integral */
    7893          49 :   if (l == 1) retgc_const(av, cgetg(1, t_VEC));
    7894          49 :   L = cgetg(l, t_VEC);
    7895          49 :   if (Q == 1)
    7896             :   {
    7897           7 :     GEN vP = MF_get_fields(mf);
    7898          21 :     for (i = 1; i < l; i++) gel(L,i) = const_vec(degpol(gel(vP,i)), gen_1);
    7899           7 :     return L;
    7900             :   }
    7901          42 :   vE = mfeigenembed(mf,prec);
    7902          42 :   if (Q == N) return gc_upto(av, mffrickeeigen(mf, vE, prec));
    7903          21 :   Q = labs(Q);
    7904          21 :   NQ = atkin_get_NQ(N, Q, "mfatkineigenvalues"); /* != 1 */
    7905          21 :   mfatk = mfatkininit(mf, Q, prec);
    7906          21 :   mfB= gel(mfatk,1); if (typ(mfB) != t_VEC) mfB = mf;
    7907          21 :   MQ = gel(mfatk,2);
    7908          21 :   C  = gel(mfatk,3);
    7909          21 :   M = row(mfcoefs_mf(mfB,1,1), 2); /* vec of a_1(b_i) for mfbasis functions */
    7910          56 :   for (i = 1; i < l; i++)
    7911             :   {
    7912          35 :     GEN c = RgV_dotproduct(RgM_RgC_mul(MQ,gel(vF,i)), M); /* C * eigen_i */
    7913          35 :     gel(L,i) = Rg_embedall_i(c, gel(vE,i));
    7914             :   }
    7915          21 :   if (!gequal1(C)) L = gdiv(L, C);
    7916          21 :   CHI = MF_get_CHI(mf);
    7917          21 :   if (mfcharorder(CHI) <= 2 && NQ % mfcharconductor(CHI) == 0) L = ground(L);
    7918          21 :   return gc_GEN(av, L);
    7919             : }
    7920             : 
    7921             : /* expand B_d V, keeping same length */
    7922             : static GEN
    7923       14168 : bdexpand(GEN V, long d)
    7924             : {
    7925             :   GEN W;
    7926             :   long N, n;
    7927       14168 :   if (d == 1) return V;
    7928        2730 :   N = lg(V)-1; W = zerovec(N);
    7929       47768 :   for (n = 0; n <= (N-1)/d; n++) gel(W, n*d+1) = gel(V, n+1);
    7930        2730 :   return W;
    7931             : }
    7932             : /* expand B_d V, increasing length up to lim */
    7933             : static GEN
    7934         343 : bdexpandall(GEN V, long d, long lim)
    7935             : {
    7936             :   GEN W;
    7937             :   long N, n;
    7938         343 :   if (d == 1) return V;
    7939          49 :   N = lg(V)-1; W = zerovec(lim);
    7940         301 :   for (n = 0; n <= N-1 && n*d <= lim; n++) gel(W, n*d+1) = gel(V, n+1);
    7941          49 :   return W;
    7942             : }
    7943             : 
    7944             : static void
    7945       15491 : parse_vecj(GEN T, GEN *E1, GEN *E2)
    7946             : {
    7947       15491 :   if (lg(T)==3) { *E1 = gel(T,1); *E2 = gel(T,2); }
    7948        5600 :   else { *E1 = T; *E2 = NULL; }
    7949       15491 : }
    7950             : 
    7951             : /* g in M_2(Z) ? */
    7952             : static int
    7953        3486 : check_M2Z(GEN g)
    7954        3486 : {  return typ(g) == t_MAT && lg(g) == 3 && lgcols(g) == 3 && RgM_is_ZM(g); }
    7955             : /* g in SL_2(Z) ? */
    7956             : static int
    7957        2058 : check_SL2Z(GEN g) { return check_M2Z(g) && equali1(ZM_det(g)); }
    7958             : 
    7959             : static GEN
    7960        9513 : mfcharcxeval(GEN CHI, long n, long prec)
    7961             : {
    7962        9513 :   ulong ord, N = mfcharmodulus(CHI);
    7963             :   GEN ordg;
    7964        9513 :   if (N == 1) return gen_1;
    7965        3696 :   if (ugcd(N, labs(n)) > 1) return gen_0;
    7966        3696 :   ordg = gmfcharorder(CHI);
    7967        3696 :   ord = itou(ordg);
    7968        3696 :   return rootsof1q_cx(znchareval_i(CHI,n,ordg), ord, prec);
    7969             : }
    7970             : 
    7971             : static GEN
    7972       11039 : RgV_shift(GEN V, GEN gn)
    7973             : {
    7974             :   long i, n, l;
    7975             :   GEN W;
    7976       11039 :   if (typ(gn) != t_INT) pari_err_BUG("RgV_shift [n not integral]");
    7977       11039 :   n = itos(gn);
    7978       11039 :   if (n < 0) pari_err_BUG("RgV_shift [n negative]");
    7979       11039 :   if (!n) return V;
    7980         112 :   W = cgetg_copy(V, &l); if (n > l-1) n = l-1;
    7981         308 :   for (i=1; i <= n; i++) gel(W,i) = gen_0;
    7982        4900 :   for (    ; i < l; i++) gel(W,i) = gel(V, i-n);
    7983         112 :   return W;
    7984             : }
    7985             : static GEN
    7986       19236 : hash_eisengacx(hashtable *H, void *E, long w, GEN ga, long n, long prec)
    7987             : {
    7988       19236 :   ulong h = H->hash(E);
    7989       19236 :   hashentry *e = hash_search2(H, E, h);
    7990             :   GEN v;
    7991       19236 :   if (e) v = (GEN)e->val;
    7992             :   else
    7993             :   {
    7994       12971 :     v = mfeisensteingacx((GEN)E, w, ga, n, prec);
    7995       12971 :     hash_insert2(H, E, (void*)v, h);
    7996             :   }
    7997       19236 :   return v;
    7998             : }
    7999             : static GEN
    8000       11039 : vecj_expand(GEN B, hashtable *H, long w, GEN ga, long n, long prec)
    8001             : {
    8002             :   GEN E1, E2, v;
    8003       11039 :   parse_vecj(B, &E1, &E2);
    8004       11039 :   v = hash_eisengacx(H, (void*)E1, w, ga, n, prec);
    8005       11039 :   if (E2)
    8006             :   {
    8007        8141 :     GEN u = hash_eisengacx(H, (void*)E2, w, ga, n, prec);
    8008        8141 :     GEN a = gadd(gel(v,1), gel(u,1));
    8009        8141 :     GEN b = RgV_mul_RgXn(gel(v,2), gel(u,2));
    8010        8141 :     v = mkvec2(a,b);
    8011             :   }
    8012       11039 :   return v;
    8013             : }
    8014             : static GEN
    8015        1288 : shift_M(GEN M, GEN Valpha, long w)
    8016             : {
    8017        1288 :   long i, l = lg(Valpha);
    8018        1288 :   GEN almin = vecmin(Valpha);
    8019       12327 :   for (i = 1; i < l; i++)
    8020             :   {
    8021       11039 :     GEN alpha = gel(Valpha, i), gsh = gmulsg(w, gsub(alpha,almin));
    8022       11039 :     gel(M,i) = RgV_shift(gel(M,i), gsh);
    8023             :   }
    8024        1288 :   return almin;
    8025             : }
    8026             : static GEN mfeisensteinspaceinit(GEN NK);
    8027             : #if 0
    8028             : /* ga in M_2^+(Z)), n >= 0 */
    8029             : static GEN
    8030             : mfgaexpansion_init(GEN mf, GEN ga, long n, long prec)
    8031             : {
    8032             :   GEN M, Mvecj, vecj, almin, Valpha;
    8033             :   long i, w, l, N = MF_get_N(mf), c = itos(gcoeff(ga,2,1));
    8034             :   hashtable *H;
    8035             : 
    8036             :   if (c % N == 0)
    8037             :   { /* ga in G_0(N), trivial case; w = 1 */
    8038             :     GEN chid = mfcharcxeval(MF_get_CHI(mf), itos(gcoeff(ga,2,2)), prec);
    8039             :     return mkvec2(chid, utoi(n));
    8040             :   }
    8041             : 
    8042             :   Mvecj = obj_checkbuild(mf, MF_EISENSPACE, &mfeisensteinspaceinit);
    8043             :   if (lg(Mvecj) < 5) pari_err_IMPL("mfgaexpansion_init in this case");
    8044             :   w = mfcuspcanon_width(N, c);
    8045             :   vecj = gel(Mvecj, 3);
    8046             :   l = lg(vecj);
    8047             :   M = cgetg(l, t_VEC);
    8048             :   Valpha = cgetg(l, t_VEC);
    8049             :   H = hash_create_GEN(l, 1);
    8050             :   for (i = 1; i < l; i++)
    8051             :   {
    8052             :     GEN v = vecj_expand(gel(vecj,i), H, w, ga, n, prec);
    8053             :     gel(Valpha,i) = gel(v,1);
    8054             :     gel(M,i) = gel(v,2);
    8055             :   }
    8056             :   almin = shift_M(M, Valpha, w);
    8057             :   return mkvec3(almin, utoi(w), M);
    8058             : }
    8059             : /* half-integer weight not supported; vF = [F,eisendec(F)].
    8060             :  * Minit = mfgaexpansion_init(mf, ga, n, prec) */
    8061             : static GEN
    8062             : mfgaexpansion_with_init(GEN Minit, GEN vF)
    8063             : {
    8064             :   GEN v;
    8065             :   if (lg(Minit) == 3)
    8066             :   { /* ga in G_0(N) */
    8067             :     GEN chid = gel(Minit,1), gn = gel(Minit,2);
    8068             :     v = mfcoefs_i(gel(vF,1), itou(gn), 1);
    8069             :     v = mkvec3(gen_0, gen_1, RgV_Rg_mul(v,chid));
    8070             :   }
    8071             :   else
    8072             :   {
    8073             :     GEN V = RgM_RgC_mul(gel(Minit,3), gel(vF,2));
    8074             :     v = mkvec3(gel(Minit,1), gel(Minit,2), V);
    8075             :   }
    8076             :   return v;
    8077             : }
    8078             : #endif
    8079             : 
    8080             : /* B = mfeisensteindec(F) already embedded, ga in M_2^+(Z)), n >= 0 */
    8081             : static GEN
    8082        1288 : mfgaexpansion_i(GEN mf, GEN B0, GEN ga, long n, long prec)
    8083             : {
    8084        1288 :   GEN M, Mvecj, vecj, almin, Valpha, B, E = NULL;
    8085        1288 :   long i, j, w, nw, l, N = MF_get_N(mf), bit = prec2nbits(prec) / 2;
    8086             :   hashtable *H;
    8087             : 
    8088        1288 :   Mvecj = obj_check(mf, MF_EISENSPACE);
    8089        1288 :   if (lg(Mvecj) < 5) { E = gel(Mvecj, 2); Mvecj = gel(Mvecj, 1); }
    8090        1288 :   vecj = gel(Mvecj, 3);
    8091        1288 :   l = lg(vecj);
    8092        1288 :   B = cgetg(l, t_COL);
    8093        1288 :   M = cgetg(l, t_VEC);
    8094        1288 :   Valpha = cgetg(l, t_VEC);
    8095        1288 :   w = mfZC_width(N, gel(ga,1));
    8096        1288 :   nw = E ? n + w : n;
    8097        1288 :   H = hash_create_GEN(l, 1);
    8098       15673 :   for (i = j = 1; i < l; i++)
    8099             :   {
    8100             :     GEN v;
    8101       14385 :     if (gequal0(gel(B0,i))) continue;
    8102       11039 :     v = vecj_expand(gel(vecj,i), H, w, ga, nw, prec);
    8103       11039 :     gel(B,j) = gel(B0,i);
    8104       11039 :     gel(Valpha,j) = gel(v,1);
    8105       11039 :     gel(M,j) = gel(v,2); j++;
    8106             :   }
    8107        1288 :   setlg(Valpha, j);
    8108        1288 :   setlg(B, j);
    8109        1288 :   setlg(M, j); l = j;
    8110        1288 :   if (l == 1) return mkvec3(gen_0, utoi(w), zerovec(n+1));
    8111        1288 :   almin = shift_M(M, Valpha, w);
    8112        1288 :   B = RgM_RgC_mul(M, B); l = lg(B);
    8113      158347 :   for (i = 1; i < l; i++)
    8114      157059 :     if (gexpo(gel(B,i)) < -bit) gel(B,i) = gen_0;
    8115        1288 :   settyp(B, t_VEC);
    8116        1288 :   if (E)
    8117             :   {
    8118             :     GEN v, e;
    8119          56 :     long ell = 0, vB, ve;
    8120         126 :     for (i = 1; i < l; i++)
    8121         126 :       if (!gequal0(gel(B,i))) break;
    8122          56 :     vB = i-1;
    8123          56 :     v = hash_eisengacx(H, (void*)E, w, ga, n + vB, prec);
    8124          56 :     e = gel(v,2); l = lg(e);
    8125          56 :     for (i = 1; i < l; i++)
    8126          56 :       if (!gequal0(gel(e,i))) break;
    8127          56 :     ve = i-1;
    8128          56 :     almin = gsub(almin, gel(v,1));
    8129          56 :     if (gsigne(almin) < 0)
    8130             :     {
    8131           0 :       GEN gell = gceil(gmulsg(-w, almin));
    8132           0 :       ell = itos(gell);
    8133           0 :       almin = gadd(almin, gdivgu(gell, w));
    8134           0 :       if (nw < ell) pari_err_IMPL("alpha < 0 in mfgaexpansion");
    8135             :     }
    8136          56 :     if (ve) { ell += ve; e = vecslice(e, ve+1, l-1); }
    8137          56 :     B = vecslice(B, ell + 1, minss(n + ell + 1, lg(B)-1));
    8138          56 :     B = RgV_div_RgXn(B, e);
    8139             :   }
    8140        1288 :   return mkvec3(almin, utoi(w), B);
    8141             : }
    8142             : 
    8143             : /* Theta multiplier: assume 4 | C, (C,D)=1 */
    8144             : static GEN
    8145         343 : mfthetamultiplier(GEN C, GEN D)
    8146             : {
    8147         343 :   long s = kronecker(C, D);
    8148         343 :   if (Mod4(D) == 1) return s > 0 ? gen_1: gen_m1;
    8149          84 :   return s > 0? powIs(3): gen_I();
    8150             : }
    8151             : /* theta | [*,*;C,D] defined over Q(i) [else over Q] */
    8152             : static int
    8153          56 : mfthetaI(long C, long D) { return odd(C) || (D & 3) == 3; }
    8154             : /* (theta | M) [0..n], assume (C,D) = 1 */
    8155             : static GEN
    8156         343 : mfthetaexpansion(GEN M, long n)
    8157             : {
    8158         343 :   GEN w, s, al, sla, E, V = zerovec(n+1), C = gcoeff(M,2,1), D = gcoeff(M,2,2);
    8159         343 :   long lim, la, f, C4 = Mod4(C);
    8160         343 :   switch (C4)
    8161             :   {
    8162          70 :     case 0: al = gen_0; w = gen_1;
    8163          70 :       s = mfthetamultiplier(C,D);
    8164          70 :       lim = usqrt(n); gel(V, 1) = s;
    8165          70 :       s = gmul2n(s, 1);
    8166         756 :       for (f = 1; f <= lim; f++) gel(V, f*f + 1) = s;
    8167          70 :       break;
    8168         105 :     case 2: al = uutoQ(1,4); w = gen_1;
    8169         105 :       E = subii(C, shifti(D,1)); /* (E, D) = 1 */
    8170         105 :       s = gmul2n(mfthetamultiplier(E, D), 1);
    8171         105 :       if ((!signe(E) && equalim1(D)) || (signe(E) > 0 && signe(C) < 0))
    8172          14 :         s = gneg(s);
    8173         105 :       lim = (usqrt(n << 2) - 1) >> 1;
    8174         966 :       for (f = 0; f <= lim; f++) gel(V, f*(f+1) + 1) = s;
    8175         105 :       break;
    8176         168 :     default: al = gen_0; w = utoipos(4);
    8177         168 :       la = (-Mod4(D)*C4) & 3L;
    8178         168 :       E = negi(addii(D, mului(la, C)));
    8179         168 :       s = mfthetamultiplier(E, C); /* (E,C) = 1 */
    8180         168 :       if (signe(C) < 0 && signe(E) >= 0) s = gneg(s);
    8181         168 :       s = gsub(s, mulcxI(s));
    8182         168 :       sla = gmul(s, powIs(-la));
    8183         168 :       lim = usqrt(n); gel(V, 1) = gmul2n(s, -1);
    8184        1708 :       for (f = 1; f <= lim; f++) gel(V, f*f + 1) = odd(f) ? sla : s;
    8185         168 :       break;
    8186             :   }
    8187         343 :   return mkvec3(al, w, V);
    8188             : }
    8189             : 
    8190             : /* F 1/2 integral weight */
    8191             : static GEN
    8192         343 : mf2gaexpansion(GEN mf2, GEN F, GEN ga, long n, long prec)
    8193             : {
    8194         343 :   GEN FT = mfmultheta(F), mf = obj_checkbuild(mf2, MF_MF2INIT, &mf2init);
    8195         343 :   GEN res, V1, Tres, V2, al, V, gsh, C = gcoeff(ga,2,1);
    8196         343 :   long w2, N = MF_get_N(mf), w = mfcuspcanon_width(N, umodiu(C,N));
    8197         343 :   long ext = (Mod4(C) != 2)? 0: (w+3) >> 2;
    8198         343 :   long prec2 = prec + nbits2extraprec((long)M_PI/(2*M_LN2)*sqrt(n + ext));
    8199         343 :   res = mfgaexpansion(mf, FT, ga, n + ext, prec2);
    8200         343 :   Tres = mfthetaexpansion(ga, n + ext);
    8201         343 :   V1 = gel(res,3);
    8202         343 :   V2 = gel(Tres,3);
    8203         343 :   al = gsub(gel(res,1), gel(Tres,1));
    8204         343 :   w2 = itos(gel(Tres,2));
    8205         343 :   if (w != itos(gel(res,2)) || w % w2)
    8206           0 :     pari_err_BUG("mf2gaexpansion [incorrect w2 or w]");
    8207         343 :   if (w2 != w) V2 = bdexpand(V2, w/w2);
    8208         343 :   V = RgV_div_RgXn(V1, V2);
    8209         343 :   gsh = gfloor(gmulsg(w, al));
    8210         343 :   if (!gequal0(gsh))
    8211             :   {
    8212          35 :     al = gsub(al, gdivgu(gsh, w));
    8213          35 :     if (gsigne(gsh) > 0)
    8214             :     {
    8215           0 :       V = RgV_shift(V, gsh);
    8216           0 :       V = vecslice(V, 1, n + 1);
    8217             :     }
    8218             :     else
    8219             :     {
    8220          35 :       long sh = -itos(gsh), i;
    8221          35 :       if (sh > ext) pari_err_BUG("mf2gaexpansion [incorrect sh]");
    8222         154 :       for (i = 1; i <= sh; i++)
    8223         119 :         if (!gequal0(gel(V,i))) pari_err_BUG("mf2gaexpansion [sh too large]");
    8224          35 :       V = vecslice(V, sh+1, n + sh+1);
    8225             :     }
    8226             :   }
    8227         343 :   obj_free(mf); return mkvec3(al, stoi(w), gprec_wtrunc(V, prec));
    8228             : }
    8229             : 
    8230             : static GEN
    8231          77 : mfgaexpansionatkin(GEN mf, GEN F, GEN C, GEN D, long Q, long n, long prec)
    8232             : {
    8233          77 :   GEN mfa = mfatkininit_i(mf, Q, 0, prec), MQ = gel(mfa,2);
    8234          77 :   long i, FC, k = MF_get_k(mf);
    8235          77 :   GEN x, v, V, z, s, CHI = mfchartoprimitive(MF_get_CHI(mf), &FC);
    8236             : 
    8237             :   /* V = mfcoefs(F | w_Q, n), can't use mfatkin because MQ nonrational */
    8238          77 :   V = RgM_RgC_mul(mfcoefs_mf(mf,n,1), RgM_RgC_mul(MQ, mftobasis_i(mf,F)));
    8239          77 :   (void)bezout(utoipos(Q), C, &x, &v);
    8240          77 :   s = mfchareval(CHI, (umodiu(x, FC) * umodiu(D, FC)) % FC);
    8241          77 :   s = gdiv(s, gpow(utoipos(Q), uutoQ(k,2), prec));
    8242          77 :   V = RgV_Rg_mul(V, s);
    8243          77 :   z = rootsof1powinit(umodiu(D,Q)*umodiu(v,Q) % Q, Q, prec);
    8244       11613 :   for (i = 1; i <= n+1; i++) gel(V,i) = gmul(gel(V,i), rootsof1pow(z, i-1));
    8245          77 :   return mkvec3(gen_0, utoipos(Q), V);
    8246             : }
    8247             : 
    8248             : static long
    8249          70 : inveis_extraprec(long N, GEN ga, GEN Mvecj, long n)
    8250             : {
    8251          70 :   long e, w = mfZC_width(N, gel(ga,1));
    8252          70 :   GEN f, E = gel(Mvecj,2), v = mfeisensteingacx(E, w, ga, n, DEFAULTPREC);
    8253          70 :   v = gel(v,2);
    8254          70 :   f = RgV_to_RgX(v,0); n -= RgX_valrem(f, &f);
    8255          70 :   e = gexpo(RgXn_inv(f, n+1));
    8256          70 :   return (e > 0)? nbits2extraprec(e): 0;
    8257             : }
    8258             : /* allow F of the form [F, mf_eisendec(F)]~ */
    8259             : static GEN
    8260        2051 : mfgaexpansion(GEN mf, GEN F, GEN ga, long n, long prec)
    8261             : {
    8262        2051 :   GEN v, EF = NULL, res, Mvecj, c, d;
    8263             :   long precnew, N;
    8264             : 
    8265        2051 :   if (n < 0) pari_err_DOMAIN("mfgaexpansion", "n", "<", gen_0, stoi(n));
    8266        2051 :   if (typ(F) == t_COL && lg(F) == 3) { EF = gel(F,2); F = gel(F,1); }
    8267        2051 :   if (!checkmf_i(F)) pari_err_TYPE("mfgaexpansion", F);
    8268        2051 :   if (!check_SL2Z(ga)) pari_err_TYPE("mfgaexpansion",ga);
    8269        2051 :   if (typ(mf_get_gk(F)) != t_INT) return mf2gaexpansion(mf, F, ga, n, prec);
    8270        1708 :   c = gcoeff(ga,2,1);
    8271        1708 :   d = gcoeff(ga,2,2);
    8272        1708 :   N = MF_get_N(mf);
    8273        1708 :   if (!umodiu(c, mf_get_N(F)))
    8274             :   { /* trivial case: ga in Gamma_0(N) */
    8275         343 :     long w = mfcuspcanon_width(N, umodiu(c,N));
    8276         343 :     GEN CHI = mf_get_CHI(F);
    8277         343 :     GEN chid = mfcharcxeval(CHI, umodiu(d,mfcharmodulus(CHI)), prec);
    8278         343 :     v = mfcoefs_i(F, n/w, 1); if (!isint1(chid)) v = RgV_Rg_mul(v,chid);
    8279         343 :     return mkvec3(gen_0, stoi(w), bdexpandall(v,w,n+1));
    8280             :   }
    8281        1365 :   mf = MF_set_new(mf);
    8282        1365 :   if (MF_get_space(mf) == mf_NEW)
    8283             :   {
    8284         483 :     long cN = umodiu(c,N), g = ugcd(cN,N), Q = N/g;
    8285         483 :     GEN CHI = MF_get_CHI(mf);
    8286         483 :     if (ugcd(cN, Q)==1 && mfcharorder(CHI) <= 2
    8287         231 :                        && g % mfcharconductor(CHI) == 0
    8288         119 :                        && degpol(mf_get_field(F)) == 1)
    8289          77 :       return mfgaexpansionatkin(mf, F, c, d, Q, n, prec);
    8290             :   }
    8291        1288 :   Mvecj = obj_checkbuild(mf, MF_EISENSPACE, &mfeisensteinspaceinit);
    8292        1288 :   precnew = prec;
    8293        1288 :   if (lg(Mvecj) < 5) precnew += inveis_extraprec(N, ga, Mvecj, n);
    8294        1288 :   if (!EF) EF = mf_eisendec(mf, F, precnew);
    8295        1288 :   res = mfgaexpansion_i(mf, EF, ga, n, precnew);
    8296        1288 :   return precnew == prec ? res : gprec_wtrunc(res, prec);
    8297             : }
    8298             : 
    8299             : /* parity = -1 or +1 */
    8300             : static GEN
    8301         217 : findd(long N, long parity)
    8302             : {
    8303         217 :   GEN L, D = mydivisorsu(N);
    8304         217 :   long i, j, l = lg(D);
    8305         217 :   L = cgetg(l, t_VEC);
    8306        1218 :   for (i = j = 1; i < l; i++)
    8307             :   {
    8308        1001 :     long d = D[i];
    8309        1001 :     if (parity == -1) d = -d;
    8310        1001 :     if (sisfundamental(d)) gel(L,j++) = stoi(d);
    8311             :   }
    8312         217 :   setlg(L,j); return L;
    8313             : }
    8314             : /* does ND contain a divisor of N ? */
    8315             : static int
    8316         413 : seenD(long N, GEN ND)
    8317             : {
    8318         413 :   long j, l = lg(ND);
    8319         427 :   for (j = 1; j < l; j++)
    8320          14 :     if (N % ND[j] == 0) return 1;
    8321         413 :   return 0;
    8322             : }
    8323             : static GEN
    8324          63 : search_levels(GEN vN, const char *f)
    8325             : {
    8326          63 :   switch(typ(vN))
    8327             :   {
    8328          28 :     case t_INT: vN = mkvecsmall(itos(vN)); break;
    8329          35 :     case t_VEC: case t_COL: vN = ZV_to_zv(vN); break;
    8330           0 :     case t_VECSMALL: vN = leafcopy(vN); break;
    8331           0 :     default: pari_err_TYPE(f, vN);
    8332             :   }
    8333          63 :   vecsmall_sort(vN); return vN;
    8334             : }
    8335             : GEN
    8336          28 : mfsearch(GEN NK, GEN V, long space)
    8337             : {
    8338          28 :   pari_sp av = avma;
    8339             :   GEN F, gk, NbyD, vN;
    8340             :   long n, nk, dk, parity, nV, i, lvN;
    8341             : 
    8342          28 :   if (typ(NK) != t_VEC || lg(NK) != 3) pari_err_TYPE("mfsearch", NK);
    8343          28 :   gk = gel(NK,2);
    8344          28 :   if (typ(gmul2n(gk, 1)) != t_INT) pari_err_TYPE("mfsearch [k]", gk);
    8345          28 :   switch(typ(V))
    8346             :   {
    8347          28 :     case t_VEC: V = shallowtrans(V);
    8348          28 :     case t_COL: break;
    8349           0 :     default: pari_err_TYPE("mfsearch [V]", V);
    8350             :   }
    8351          28 :   vN = search_levels(gel(NK,1), "mfsearch [N]");
    8352          28 :   if (gequal0(V)) { set_avma(av); retmkvec(mftrivial()); }
    8353          14 :   lvN = lg(vN);
    8354             : 
    8355          14 :   Qtoss(gk, &nk,&dk);
    8356          14 :   parity = (dk == 1 && odd(nk)) ? -1 : 1;
    8357          14 :   nV = lg(V)-2;
    8358          14 :   F = cgetg(1, t_VEC);
    8359          14 :   NbyD = const_vec(vN[lvN-1], cgetg(1,t_VECSMALL));
    8360         231 :   for (n = 1; n < lvN; n++)
    8361             :   {
    8362         217 :     long N = vN[n];
    8363             :     GEN L;
    8364         217 :     if (N <= 0 || (dk == 2 && (N & 3))) continue;
    8365         217 :     L = findd(N, parity);
    8366         630 :     for (i = 1; i < lg(L); i++)
    8367             :     {
    8368         413 :       GEN mf, M, CO, gD = gel(L,i);
    8369         413 :       GEN *ND = (GEN*)NbyD + itou(gD); /* points to NbyD[|D|] */
    8370             : 
    8371         413 :       if (seenD(N, *ND)) continue;
    8372         413 :       mf = mfinit_Nndkchi(N, nk, dk, get_mfchar(gD), space, 1);
    8373         413 :       M = mfcoefs_mf(mf, nV, 1);
    8374         413 :       CO = inverseimage(M, V); if (lg(CO) == 1) continue;
    8375             : 
    8376          42 :       F = vec_append(F, mflinear(mf,CO));
    8377          42 :       *ND = vecsmall_append(*ND, N); /* add to NbyD[|D|] */
    8378             :     }
    8379             :   }
    8380          14 :   return gc_GEN(av, F);
    8381             : }
    8382             : 
    8383             : static GEN
    8384         889 : search_from_split(GEN mf, GEN vap, GEN vlp)
    8385             : {
    8386         889 :   pari_sp av = avma;
    8387         889 :   long lvlp = lg(vlp), j, jv, l1;
    8388         889 :   GEN v, NK, S1, S, M = NULL;
    8389             : 
    8390         889 :   S1 = gel(split_i(mf, 1, 0), 1); /* rational newforms */
    8391         889 :   l1 = lg(S1);
    8392         889 :   if (l1 == 1) return gc_NULL(av);
    8393         455 :   v = cgetg(l1, t_VEC);
    8394         455 :   S = MF_get_S(mf);
    8395         455 :   NK = mf_get_NK(gel(S,1));
    8396         455 :   if (lvlp > 1) M = rowpermute(mfcoefs_mf(mf, vlp[lvlp-1], 1), vlp);
    8397         980 :   for (j = jv = 1; j < l1; j++)
    8398             :   {
    8399         525 :     GEN vF = gel(S1,j);
    8400             :     long t;
    8401         658 :     for (t = lvlp-1; t > 0; t--)
    8402             :     { /* lhs = vlp[j]-th coefficient of eigenform */
    8403         595 :       GEN rhs = gel(vap,t), lhs = RgMrow_RgC_mul(M, vF, t);
    8404         595 :       if (!gequal(lhs, rhs)) break;
    8405             :     }
    8406         525 :     if (!t) gel(v,jv++) = mflinear_i(NK,S,vF);
    8407             :   }
    8408         455 :   if (jv == 1) return gc_NULL(av);
    8409          63 :   setlg(v,jv); return v;
    8410             : }
    8411             : GEN
    8412          35 : mfeigensearch(GEN NK, GEN AP)
    8413             : {
    8414          35 :   pari_sp av = avma;
    8415          35 :   GEN k, vN, vap, vlp, vres = cgetg(1, t_VEC), D;
    8416             :   long n, lvN, i, l, even;
    8417             : 
    8418          35 :   if (!AP) l = 1;
    8419             :   else
    8420             :   {
    8421          28 :     l = lg(AP);
    8422          28 :     if (typ(AP) != t_VEC) pari_err_TYPE("mfeigensearch",AP);
    8423             :   }
    8424          35 :   vap = cgetg(l, t_VEC);
    8425          35 :   vlp = cgetg(l, t_VECSMALL);
    8426          35 :   if (l > 1)
    8427             :   {
    8428          28 :     GEN perm = indexvecsort(AP, mkvecsmall(1));
    8429          77 :     for (i = 1; i < l; i++)
    8430             :     {
    8431          49 :       GEN v = gel(AP,perm[i]), gp, ap;
    8432          49 :       if (typ(v) != t_VEC || lg(v) != 3) pari_err_TYPE("mfeigensearch", AP);
    8433          49 :       gp = gel(v,1);
    8434          49 :       ap = gel(v,2);
    8435          49 :       if (typ(gp) != t_INT || (typ(ap) != t_INT && typ(ap) != t_INTMOD))
    8436           0 :         pari_err_TYPE("mfeigensearch", AP);
    8437          49 :       gel(vap,i) = ap;
    8438          49 :       vlp[i] = itos(gp)+1; if (vlp[i] < 0) pari_err_TYPE("mfeigensearch", AP);
    8439             :     }
    8440             :   }
    8441          35 :   l = lg(NK);
    8442          35 :   if (typ(NK) != t_VEC || l != 3) pari_err_TYPE("mfeigensearch",NK);
    8443          35 :   k = gel(NK,2);
    8444          35 :   vN = search_levels(gel(NK,1), "mfeigensearch [N]");
    8445          35 :   lvN = lg(vN);
    8446          35 :   vecsmall_sort(vlp);
    8447          35 :   even = !mpodd(k);
    8448         980 :   for (n = 1; n < lvN; n++)
    8449             :   {
    8450         945 :     pari_sp av2 = avma;
    8451             :     GEN mf, L;
    8452         945 :     long N = vN[n];
    8453         945 :     if (even) D = gen_1;
    8454             :     else
    8455             :     {
    8456         112 :       long r = (N&3L);
    8457         112 :       if (r == 1 || r == 2) continue;
    8458          56 :       D = stoi( corediscs(-N, NULL) ); /* < 0 */
    8459             :     }
    8460         889 :     mf = mfinit_i(mkvec3(utoipos(N), k, D), mf_NEW);
    8461         889 :     L = search_from_split(mf, vap, vlp);
    8462         889 :     if (L) vres = shallowconcat(vres, L); else set_avma(av2);
    8463             :   }
    8464          35 :   return gc_GEN(av, vres);
    8465             : }
    8466             : 
    8467             : /* tf_{N,k}(n) */
    8468             : static GEN
    8469     4646243 : mfnewtracecache(long N, long k, long n, cachenew_t *cache)
    8470             : {
    8471     4646243 :   GEN C = NULL, S;
    8472             :   long lcache;
    8473     4646243 :   if (!n) return gen_0;
    8474     4501854 :   S = gel(cache->vnew,N);
    8475     4501854 :   lcache = lg(S);
    8476     4501854 :   if (n < lcache) C = gel(S, n);
    8477     4501854 :   if (C) cache->newHIT++;
    8478     2673683 :   else C = mfnewtrace_i(N,k,n,cache);
    8479     4501854 :   cache->newTOTAL++;
    8480     4501854 :   if (n < lcache) gel(S,n) = C;
    8481     4501854 :   return C;
    8482             : }
    8483             : 
    8484             : static long
    8485        1400 : mfdim_Nkchi(long N, long k, GEN CHI, long space)
    8486             : {
    8487        1400 :   if (k < 0 || badchar(N,k,CHI)) return 0;
    8488        1099 :   if (k == 0)
    8489          35 :     return mfcharistrivial(CHI) && !space_is_cusp(space)? 1: 0;
    8490        1064 :   switch(space)
    8491             :   {
    8492         245 :     case mf_NEW: return mfnewdim(N,k,CHI);
    8493         203 :     case mf_CUSP:return mfcuspdim(N,k,CHI);
    8494         168 :     case mf_OLD: return mfolddim(N,k,CHI);
    8495         217 :     case mf_FULL:return mffulldim(N,k,CHI);
    8496         231 :     case mf_EISEN: return mfeisensteindim(N,k,CHI);
    8497           0 :     default: pari_err_FLAG("mfdim");
    8498             :   }
    8499             :   return 0;/*LCOV_EXCL_LINE*/
    8500             : }
    8501             : static long
    8502        2114 : mf1dimsum(long N, long space)
    8503             : {
    8504        2114 :   switch(space)
    8505             :   {
    8506        1050 :     case mf_NEW:  return mf1newdimsum(N);
    8507        1057 :     case mf_CUSP: return mf1cuspdimsum(N);
    8508           7 :     case mf_OLD:  return mf1olddimsum(N);
    8509             :   }
    8510           0 :   pari_err_FLAG("mfdim");
    8511             :   return 0; /*LCOV_EXCL_LINE*/
    8512             : }
    8513             : /* mfdim for k = nk/dk */
    8514             : static long
    8515       44744 : mfdim_Nndkchi(long N, long nk, long dk, GEN CHI, long space)
    8516       43463 : { return (dk == 2)? mf2dim_Nkchi(N, nk >> 1, CHI, space)
    8517       88186 :                   : mfdim_Nkchi(N, nk, CHI, space); }
    8518             : /* FIXME: use direct dim Gamma1(N) formula, don't compute individual spaces */
    8519             : static long
    8520         252 : mfkdimsum(long N, long k, long dk, long space)
    8521             : {
    8522         252 :   GEN w = mfchars(N, k, dk, NULL);
    8523         252 :   long i, j, D = 0, l = lg(w);
    8524        1239 :   for (i = j = 1; i < l; i++)
    8525             :   {
    8526         987 :     GEN CHI = gel(w,i);
    8527         987 :     long d = mfdim_Nndkchi(N,k,dk,CHI,space);
    8528         987 :     if (d) D += d * myeulerphiu(mfcharorder(CHI));
    8529             :   }
    8530         252 :   return D;
    8531             : }
    8532             : static GEN
    8533         105 : mf1dims(long N, GEN vCHI, long space)
    8534             : {
    8535         105 :   GEN D = NULL;
    8536         105 :   switch(space)
    8537             :   {
    8538          56 :     case mf_NEW: D = mf1newdimall(N, vCHI); break;
    8539          21 :     case mf_CUSP:D = mf1cuspdimall(N, vCHI); break;
    8540          28 :     case mf_OLD: D = mf1olddimall(N, vCHI); break;
    8541           0 :     default: pari_err_FLAG("mfdim");
    8542             :   }
    8543         105 :   return D;
    8544             : }
    8545             : static GEN
    8546        2961 : mfkdims(long N, long k, long dk, GEN vCHI, long space)
    8547             : {
    8548        2961 :   GEN D, w = mfchars(N, k, dk, vCHI);
    8549        2961 :   long i, j, l = lg(w);
    8550        2961 :   D = cgetg(l, t_VEC);
    8551       46592 :   for (i = j = 1; i < l; i++)
    8552             :   {
    8553       43631 :     GEN CHI = gel(w,i);
    8554       43631 :     long d = mfdim_Nndkchi(N,k,dk,CHI,space);
    8555       43631 :     if (vCHI)
    8556         574 :       gel(D, j++) = mkvec2s(d, 0);
    8557       43057 :     else if (d)
    8558        2520 :       gel(D, j++) = fmt_dim(CHI, d, 0);
    8559             :   }
    8560        2961 :   setlg(D,j); return D;
    8561             : }
    8562             : GEN
    8563        5719 : mfdim(GEN NK, long space)
    8564             : {
    8565        5719 :   pari_sp av = avma;
    8566             :   long N, k, dk, joker;
    8567             :   GEN CHI, mf;
    8568        5719 :   if ((mf = checkMF_i(NK))) return utoi(MF_get_dim(mf));
    8569        5586 :   checkNK2(NK, &N, &k, &dk, &CHI, 2);
    8570        5586 :   if (!CHI) joker = 1;
    8571             :   else
    8572        2611 :     switch(typ(CHI))
    8573             :     {
    8574        2373 :       case t_INT: joker = 2; break;
    8575         112 :       case t_COL: joker = 3; break;
    8576         126 :       default: joker = 0; break;
    8577             :     }
    8578        5586 :   if (joker)
    8579             :   {
    8580             :     long d;
    8581             :     GEN D;
    8582        5460 :     if (k < 0) switch(joker)
    8583             :     {
    8584           0 :       case 1: return cgetg(1,t_VEC);
    8585           7 :       case 2: return gen_0;
    8586           0 :       case 3: return mfdim0all(CHI);
    8587             :     }
    8588        5453 :     if (k == 0)
    8589             :     {
    8590          28 :       if (space_is_cusp(space)) switch(joker)
    8591             :       {
    8592           7 :         case 1: return cgetg(1,t_VEC);
    8593           0 :         case 2: return gen_0;
    8594           7 :         case 3: return mfdim0all(CHI);
    8595             :       }
    8596          14 :       switch(joker)
    8597             :       {
    8598             :         long i, l;
    8599           7 :         case 1: retmkvec(fmt_dim(mfchartrivial(),0,0));
    8600           0 :         case 2: return gen_1;
    8601           7 :         case 3: l = lg(CHI); D = cgetg(l,t_VEC);
    8602          35 :                 for (i = 1; i < l; i++)
    8603             :                 {
    8604          28 :                   long t = mfcharistrivial(gel(CHI,i));
    8605          28 :                   gel(D,i) = mkvec2(t? gen_1: gen_0, gen_0);
    8606             :                 }
    8607           7 :                 return D;
    8608             :       }
    8609             :     }
    8610        5425 :     if (dk == 1 && k == 1 && space != mf_EISEN)
    8611         105 :     {
    8612        2219 :       long fix = 0, space0 = space;
    8613        2219 :       if (space == mf_FULL) space = mf_CUSP; /* remove Eisenstein part */
    8614        2219 :       if (joker == 2)
    8615             :       {
    8616        2114 :         d = mf1dimsum(N, space);
    8617        2114 :         if (space0 == mf_FULL) d += mfkdimsum(N,k,dk,mf_EISEN);/*add it back*/
    8618        2114 :         return gc_utoi(av, d);
    8619             :       }
    8620             :       /* must initialize explicitly: trivial spaces for E_k/S_k differ */
    8621         105 :       if (space0 == mf_FULL)
    8622             :       {
    8623           7 :         if (!CHI) fix = 1; /* must remove 0 spaces */
    8624           7 :         CHI = mfchars(N, k, dk, CHI);
    8625             :       }
    8626         105 :       D = mf1dims(N, CHI, space);
    8627         105 :       if (space0 == mf_FULL)
    8628             :       {
    8629           7 :         GEN D2 = mfkdims(N, k, dk, CHI, mf_EISEN);
    8630           7 :         D = merge_dims(D, D2, fix? CHI: NULL);
    8631             :       }
    8632             :     }
    8633             :     else
    8634             :     {
    8635        3206 :       if (joker==2) { d = mfkdimsum(N,k,dk,space); return gc_utoi(av,d); }
    8636        2954 :       D = mfkdims(N, k, dk, CHI, space);
    8637             :     }
    8638        3059 :     if (!CHI) return gc_upto(av, vecsort(D, mkvecsmall(1)));
    8639         105 :     return gc_GEN(av, D);
    8640             :   }
    8641         126 :   return utoi( mfdim_Nndkchi(N, k, dk, CHI, space) );
    8642             : }
    8643             : 
    8644             : GEN
    8645         371 : mfbasis(GEN NK, long space)
    8646             : {
    8647         371 :   pari_sp av = avma;
    8648             :   long N, k, dk;
    8649             :   GEN mf, CHI;
    8650         371 :   if ((mf = checkMF_i(NK))) return gconcat(gel(mf,2), gel(mf,3));
    8651          14 :   checkNK2(NK, &N, &k, &dk, &CHI, 0);
    8652          14 :   if (dk == 2) return gc_GEN(av, mf2basis(N, k>>1, CHI, NULL, space));
    8653          14 :   mf = mfinit_Nkchi(N, k, CHI, space, 1);
    8654          14 :   return gc_GEN(av, MF_get_basis(mf));
    8655             : }
    8656             : 
    8657             : static GEN
    8658          49 : deg1ser_shallow(GEN a1, GEN a0, long v, long e)
    8659          49 : { return RgX_to_ser(deg1pol_shallow(a1, a0, v), e+2); }
    8660             : /* r / x + O(1) */
    8661             : static GEN
    8662          49 : simple_pole(GEN r)
    8663             : {
    8664          49 :   GEN S = deg1ser_shallow(gen_0, r, 0, 1);
    8665          49 :   setvalser(S, -1); return S;
    8666             : }
    8667             : 
    8668             : /* F form, E embedding; mfa = mfatkininit or root number (eigenform case) */
    8669             : static GEN
    8670         175 : mflfuncreate(GEN mfa, GEN F, GEN E, GEN N, GEN gk)
    8671             : {
    8672         175 :   GEN LF = cgetg(8,t_VEC), polar = cgetg(1,t_COL), eps;
    8673         175 :   long k = itou(gk);
    8674         175 :   gel(LF,1) = lfuntag(t_LFUN_MFCLOS, mkvec3(F,E,gen_1));
    8675         175 :   if (typ(mfa) != t_VEC)
    8676         112 :     eps = mfa; /* cuspidal eigenform: root number; no poles */
    8677             :   else
    8678             :   { /* mfatkininit */
    8679          63 :     GEN a0, b0, vF, vG, G = NULL;
    8680          63 :     GEN M = gel(mfa,2), C = gel(mfa,3), mf = gel(mfa,4);
    8681          63 :     M = gdiv(mfmatembed(E, M), C);
    8682          63 :     vF = mfvecembed(E, mftobasis_i(mf, F));
    8683          63 :     vG = RgM_RgC_mul(M, vF);
    8684          63 :     if (gequal(vF,vG)) eps = gen_1;
    8685          49 :     else if (gequal(vF,gneg(vG))) eps = gen_m1;
    8686             :     else
    8687             :     { /* not self-dual */
    8688          42 :       eps = NULL;
    8689          42 :       G = mfatkin(mfa, F);
    8690          42 :       gel(LF,2) = lfuntag(t_LFUN_MFCLOS, mkvec3(G,E,ginv(C)));
    8691          42 :       gel(LF,6) = powIs(k);
    8692             :     }
    8693             :     /* polar part */
    8694          63 :     a0 = mfembed(E, mfcoef(F,0));
    8695          63 :     b0 = eps? gmul(eps,a0): gdiv(mfembed(E, mfcoef(G,0)), C);
    8696          63 :     if (!gequal0(b0))
    8697             :     {
    8698          28 :       b0 = mulcxpowIs(gmul2n(b0,1), k);
    8699          28 :       polar = vec_append(polar, mkvec2(gk, simple_pole(b0)));
    8700             :     }
    8701          63 :     if (!gequal0(a0))
    8702             :     {
    8703          21 :       a0 = gneg(gmul2n(a0,1));
    8704          21 :       polar = vec_append(polar, mkvec2(gen_0, simple_pole(a0)));
    8705             :     }
    8706             :   }
    8707         175 :   if (eps) /* self-dual */
    8708             :   {
    8709         133 :     gel(LF,2) = mfcharorder(mf_get_CHI(F)) <= 2? gen_0: gen_1;
    8710         133 :     gel(LF,6) = mulcxpowIs(eps,k);
    8711             :   }
    8712         175 :   gel(LF,3) = mkvec2(gen_0, gen_1);
    8713         175 :   gel(LF,4) = gk;
    8714         175 :   gel(LF,5) = N;
    8715         175 :   if (lg(polar) == 1) setlg(LF,7); else gel(LF,7) = polar;
    8716         175 :   return LF;
    8717             : }
    8718             : static GEN
    8719         147 : mflfuncreateall(long sd, GEN mfa, GEN F, GEN vE, GEN gN, GEN gk)
    8720             : {
    8721         147 :   long i, l = lg(vE);
    8722         147 :   GEN L = cgetg(l, t_VEC);
    8723         322 :   for (i = 1; i < l; i++)
    8724         175 :     gel(L,i) = mflfuncreate(sd? gel(mfa,i): mfa, F, gel(vE,i), gN, gk);
    8725         147 :   return L;
    8726             : }
    8727             : GEN
    8728          98 : lfunmf(GEN mf, GEN F, long bitprec)
    8729             : {
    8730          98 :   pari_sp av = avma;
    8731          98 :   long i, l, prec = nbits2prec(bitprec);
    8732             :   GEN L, gk, gN;
    8733          98 :   mf = checkMF(mf);
    8734          98 :   gk = MF_get_gk(mf);
    8735          98 :   gN = MF_get_gN(mf);
    8736          98 :   if (typ(gk)!=t_INT) pari_err_IMPL("half-integral weight");
    8737          98 :   if (F)
    8738             :   {
    8739             :     GEN v;
    8740          91 :     long s = MF_get_space(mf);
    8741          91 :     if (!checkmf_i(F)) pari_err_TYPE("lfunmf", F);
    8742          91 :     if (!mfisinspace_i(mf, F)) err_space(F);
    8743          91 :     L = NULL;
    8744          91 :     if ((s == mf_NEW || s == mf_CUSP || s == mf_FULL)
    8745          77 :         && gequal(mfcoefs_i(F,1,1), mkvec2(gen_0,gen_1)))
    8746             :     { /* check if eigenform */
    8747          49 :       GEN vP, vF, b = mftobasis_i(mf, F);
    8748          49 :       long lF, d = degpol(mf_get_field(F));
    8749          49 :       v = mfsplit(mf, d, 0);
    8750          49 :       vF = gel(v,1);
    8751          49 :       vP = gel(v,2); lF = lg(vF);
    8752          49 :       for (i = 1; i < lF; i++)
    8753          42 :         if (degpol(gel(vP,i)) == d && gequal(gel(vF,i), b))
    8754             :         {
    8755          42 :           GEN vE = mfgetembed(F, prec);
    8756          42 :           GEN Z = mffrickeeigen_i(mf, mkvec(b), mkvec(vE), prec);
    8757          42 :           L = mflfuncreateall(1, gel(Z,1), F, vE, gN, gk);
    8758          42 :           break;
    8759             :         }
    8760             :     }
    8761          91 :     if (!L)
    8762             :     { /* not an eigenform: costly general case */
    8763          49 :       GEN mfa = mfatkininit_i(mf, itou(gN), 1, prec);
    8764          49 :       L = mflfuncreateall(0,mfa, F, mfgetembed(F,prec), gN, gk);
    8765             :     }
    8766          91 :     if (lg(L) == 2) L = gel(L,1);
    8767             :   }
    8768             :   else
    8769             :   {
    8770           7 :     GEN M = mfeigenbasis(mf), vE = mfeigenembed(mf, prec);
    8771           7 :     GEN v = mffrickeeigen(mf, vE, prec);
    8772           7 :     l = lg(vE); L = cgetg(l, t_VEC);
    8773          63 :     for (i = 1; i < l; i++)
    8774          56 :       gel(L,i) = mflfuncreateall(1,gel(v,i), gel(M,i), gel(vE,i), gN, gk);
    8775             :   }
    8776          98 :   return gc_GEN(av, L);
    8777             : }
    8778             : 
    8779             : GEN
    8780          28 : mffromell(GEN E)
    8781             : {
    8782          28 :   pari_sp av = avma;
    8783             :   GEN mf, F, z, v, S;
    8784             :   long N, i, l;
    8785             : 
    8786          28 :   checkell(E);
    8787          28 :   if (ell_get_type(E) != t_ELL_Q) pari_err_TYPE("mfffromell [E not over Q]", E);
    8788          28 :   N = itos(ellQ_get_N(E));
    8789          28 :   mf = mfinit_i(mkvec2(utoi(N), gen_2), mf_NEW);
    8790          28 :   v = split_i(mf, 1, 0);
    8791          28 :   S = gel(v,1); l = lg(S); /* rational newforms */
    8792          28 :   F = tag(t_MF_ELL, mkNK(N,2,mfchartrivial()), E);
    8793          28 :   z = mftobasis_i(mf, F);
    8794          28 :   for(i = 1; i < l; i++)
    8795          28 :     if (gequal(z, gel(S,i))) break;
    8796          28 :   if (i == l) pari_err_BUG("mffromell [E is not modular]");
    8797          28 :   return gc_GEN(av, mkvec3(mf, F, z));
    8798             : }
    8799             : 
    8800             : /* returns -1 if not, degree otherwise */
    8801             : long
    8802         140 : polishomogeneous(GEN P)
    8803             : {
    8804             :   long i, D, l;
    8805         140 :   if (typ(P) != t_POL) return 0;
    8806          77 :   D = -1; l = lg(P);
    8807         322 :   for (i = 2; i < l; i++)
    8808             :   {
    8809         245 :     GEN c = gel(P,i);
    8810             :     long d;
    8811         245 :     if (gequal0(c)) continue;
    8812         112 :     d = polishomogeneous(c);
    8813         112 :     if (d < 0) return -1;
    8814         112 :     if (D < 0) D = d + i-2; else if (D != d + i-2) return -1;
    8815             :   }
    8816          77 :   return D;
    8817             : }
    8818             : 
    8819             : /* M a pp((Gram q)^(-1)) ZM; P a homogeneous t_POL, is P spherical ? */
    8820             : static int
    8821          28 : RgX_isspherical(GEN M, GEN P)
    8822             : {
    8823          28 :   pari_sp av = avma;
    8824          28 :   GEN S, v = variables_vecsmall(P);
    8825          28 :   long i, j, l = lg(v);
    8826          28 :   if (l > lg(M)) pari_err(e_MISC, "too many variables in mffromqf");
    8827          21 :   S = gen_0;
    8828          63 :   for (j = 1; j < l; j++)
    8829             :   {
    8830          42 :     GEN Mj = gel(M, j), Pj = deriv(P, v[j]);
    8831         105 :     for (i = 1; i <= j; i++)
    8832             :     {
    8833          63 :       GEN c = gel(Mj, i);
    8834          63 :       if (!signe(c)) continue;
    8835          42 :       if (i != j) c = shifti(c, 1);
    8836          42 :       S = gadd(S, gmul(c, deriv(Pj, v[i])));
    8837             :     }
    8838             :   }
    8839          21 :   return gc_bool(av, gequal0(S));
    8840             : }
    8841             : 
    8842             : static GEN
    8843          49 : c_QFsimple_i(long n, GEN Q, GEN P)
    8844             : {
    8845          49 :   GEN V, v = qfrep0(Q, utoi(n), 1);
    8846          49 :   long i, l = lg(v);
    8847          49 :   V = cgetg(l+1, t_VEC);
    8848          49 :   if (!P || equali1(P))
    8849             :   {
    8850          42 :     gel(V,1) = gen_1;
    8851         420 :     for (i = 2; i <= l; i++) gel(V,i) = utoi(v[i-1] << 1);
    8852             :   }
    8853             :   else
    8854             :   {
    8855           7 :     gel(V,1) = gcopy(P);
    8856           7 :     for (i = 2; i <= l; i++) gel(V,i) = gmulgu(P, v[i-1] << 1);
    8857             :   }
    8858          49 :   return V;
    8859             : }
    8860             : 
    8861             : /* v a t_VECSMALL of variable numbers, lg(r) >= lg(v), r is a vector of
    8862             :  * scalars [not involving any variable in v] */
    8863             : static GEN
    8864          14 : gsubstvec_i(GEN e, GEN v, GEN r)
    8865             : {
    8866          14 :   long i, l = lg(v);
    8867          42 :   for(i = 1; i < l; i++) e = gsubst(e, v[i], gel(r,i));
    8868          14 :   return e;
    8869             : }
    8870             : static GEN
    8871          56 : c_QF_i(long n, GEN Q, GEN P)
    8872             : {
    8873          56 :   pari_sp av = avma;
    8874             :   GEN V, v, va;
    8875             :   long i, l;
    8876          56 :   if (!P || typ(P) != t_POL) return gc_upto(av, c_QFsimple_i(n, Q, P));
    8877           7 :   v = gel(minim(Q, utoi(2*n), NULL), 3);
    8878           7 :   va = variables_vecsmall(P);
    8879           7 :   V = zerovec(n + 1); l = lg(v);
    8880          21 :   for (i = 1; i < l; i++)
    8881             :   {
    8882          14 :     pari_sp av = avma;
    8883          14 :     GEN X = gel(v,i);
    8884          14 :     long c = (itos(qfeval(Q, X)) >> 1) + 1;
    8885          14 :     gel(V, c) = gc_upto(av, gadd(gel(V, c), gsubstvec_i(P, va, X)));
    8886             :   }
    8887           7 :   return gmul2n(V, 1);
    8888             : }
    8889             : 
    8890             : GEN
    8891          77 : mffromqf(GEN Q, GEN P)
    8892             : {
    8893          77 :   pari_sp av = avma;
    8894             :   GEN G, Qi, F, D, N, mf, v, gk, chi;
    8895             :   long m, d, space;
    8896          77 :   if (typ(Q) != t_MAT) pari_err_TYPE("mffromqf", Q);
    8897          77 :   if (!RgM_is_ZM(Q) || !qfiseven(Q))
    8898           0 :     pari_err_TYPE("mffromqf [not integral or even]", Q);
    8899          77 :   m = lg(Q)-1;
    8900          77 :   Qi = ZM_inv(Q, &N);
    8901          77 :   if (!qfiseven(Qi)) N = shifti(N, 1);
    8902          77 :   d = 0;
    8903          77 :   if (!P || gequal1(P)) P = NULL;
    8904             :   else
    8905             :   {
    8906          35 :     P = simplify_shallow(P);
    8907          35 :     if (typ(P) == t_POL)
    8908             :     {
    8909          28 :       d = polishomogeneous(P);
    8910          28 :       if (d < 0) pari_err_TYPE("mffromqf [not homogeneous t_POL]", P);
    8911          28 :       if (!RgX_isspherical(Qi, P))
    8912           7 :         pari_err_TYPE("mffromqf [not a spherical t_POL]", P);
    8913             :     }
    8914             :   }
    8915          63 :   gk = uutoQ(m + 2*d, 2);
    8916          63 :   D = ZM_det(Q);
    8917          63 :   if (!odd(m)) { if ((m & 3) == 2) D = negi(D); } else D = shifti(D, 1);
    8918          63 :   space = d > 0 ? mf_CUSP : mf_FULL;
    8919          63 :   G = znstar0(N,1);
    8920          63 :   chi = mkvec2(G, znchar_quad(G,D));
    8921          63 :   mf = mfinit(mkvec3(N, gk, chi), space);
    8922          63 :   if (odd(d))
    8923             :   {
    8924           7 :     F = mftrivial();
    8925           7 :     v = zerocol(MF_get_dim(mf));
    8926             :   }
    8927             :   else
    8928             :   {
    8929          56 :     F = c_QF_i(mfsturm(mf), Q, P);
    8930          56 :     v = mftobasis_i(mf, F);
    8931          56 :     F = mflinear(mf, v);
    8932             :   }
    8933          63 :   return gc_GEN(av, mkvec3(mf, F, v));
    8934             : }
    8935             : 
    8936             : /***********************************************************************/
    8937             : /*                          Eisenstein Series                          */
    8938             : /***********************************************************************/
    8939             : /* \sigma_{k-1}(\chi,n) */
    8940             : static GEN
    8941       24192 : sigchi(long k, GEN CHI, long n)
    8942             : {
    8943       24192 :   pari_sp av = avma;
    8944       24192 :   GEN S = gen_1, D = mydivisorsu(u_ppo(n,mfcharmodulus(CHI)));
    8945       24192 :   long i, l = lg(D), ord = mfcharorder(CHI), vt = varn(mfcharpol(CHI));
    8946       83671 :   for (i = 2; i < l; i++) /* skip D[1] = 1 */
    8947             :   {
    8948       59479 :     long d = D[i], a = mfcharevalord(CHI, d, ord);
    8949       59479 :     S = gadd(S, Qab_Czeta(a, ord, powuu(d, k-1), vt));
    8950             :   }
    8951       24192 :   return gc_upto(av,S);
    8952             : }
    8953             : 
    8954             : /* write n = n0*n1*n2, (n0,N1*N2) = 1, n1 | N1^oo, n2 | N2^oo;
    8955             :  * return NULL if (n,N1,N2) > 1, else return factoru(n0) */
    8956             : static GEN
    8957      686350 : sigchi2_dec(long n, long N1, long N2, long *pn1, long *pn2)
    8958             : {
    8959      686350 :   GEN P0, E0, P, E, fa = myfactoru(n);
    8960             :   long i, j, l;
    8961      686350 :   *pn1 = 1;
    8962      686350 :   *pn2 = 1;
    8963      686350 :   if (N1 == 1 && N2 == 1) return fa;
    8964      669242 :   P = gel(fa,1); l = lg(P);
    8965      669242 :   E = gel(fa,2);
    8966      669242 :   P0 = cgetg(l, t_VECSMALL);
    8967      669242 :   E0 = cgetg(l, t_VECSMALL);
    8968     1553958 :   for (i = j = 1; i < l; i++)
    8969             :   {
    8970      989975 :     long p = P[i], e = E[i];
    8971      989975 :     if (N1 % p == 0)
    8972             :     {
    8973      142919 :       if (N2 % p == 0) return NULL;
    8974       37660 :       *pn1 *= upowuu(p,e);
    8975             :     }
    8976      847056 :     else if (N2 % p == 0)
    8977      129717 :       *pn2 *= upowuu(p,e);
    8978      717339 :     else { P0[j] = p; E0[j] = e; j++; }
    8979             :   }
    8980      563983 :   setlg(P0, j);
    8981      563983 :   setlg(E0, j); return mkvec2(P0,E0);
    8982             : }
    8983             : 
    8984             : /* sigma_{k-1}(\chi_1,\chi_2,n), ord multiple of lcm(ord(CHI1),ord(CHI2)) */
    8985             : static GEN
    8986      608559 : sigchi2(long k, GEN CHI1, GEN CHI2, long n, long ord)
    8987             : {
    8988      608559 :   pari_sp av = avma;
    8989             :   GEN S, D;
    8990      608559 :   long i, l, n1, n2, vt, N1 = mfcharmodulus(CHI1), N2 = mfcharmodulus(CHI2);
    8991      608559 :   D = sigchi2_dec(n, N1, N2, &n1, &n2); if (!D) return gc_const(av, gen_0);
    8992      507983 :   D = divisorsu_fact(D); l = lg(D);
    8993      507983 :   vt = varn(mfcharpol(CHI1));
    8994     2192253 :   for (i = 1, S = gen_0; i < l; i++)
    8995             :   { /* S += d^(k-1)*chi1(d)*chi2(n/d) */
    8996     1684270 :     long a, d = n2*D[i], nd = n1*D[l-i]; /* (d,N1)=1; (n/d,N2) = 1 */
    8997     1684270 :     a = mfcharevalord(CHI1, d, ord) + mfcharevalord(CHI2, nd, ord);
    8998     1684270 :     if (a >= ord) a -= ord;
    8999     1684270 :     S = gadd(S, Qab_Czeta(a, ord, powuu(d, k-1), vt));
    9000             :   }
    9001      507983 :   return gc_upto(av, S);
    9002             : }
    9003             : 
    9004             : /**************************************************************************/
    9005             : /**           Dirichlet characters with precomputed values               **/
    9006             : /**************************************************************************/
    9007             : /* CHI mfchar */
    9008             : static GEN
    9009       33985 : mfcharcxinit(GEN CHI, long prec)
    9010             : {
    9011       33985 :   GEN G = gel(CHI,1), chi = gel(CHI,2), z, V;
    9012       33985 :   GEN v = ncharvecexpo(G, znconrey_normalized(G,chi));
    9013       33985 :   long n, l = lg(v), o = mfcharorder(CHI);
    9014       33985 :   V = cgetg(l, t_VEC);
    9015       33985 :   z = grootsof1(o, prec); /* Mod(t, Phi_o(t)) -> e(1/o) */
    9016      480851 :   for (n = 1; n < l; n++) gel(V,n) = v[n] < 0? gen_0: gel(z, v[n]+1);
    9017       33985 :   return mkvecn(6, G, chi, gmfcharorder(CHI), v, V, mfcharpol(CHI));
    9018             : }
    9019             : /* v a "CHIvec" */
    9020             : static long
    9021    28601909 : CHIvec_N(GEN v) { return itou(znstar_get_N(gel(v,1))); }
    9022             : static GEN
    9023       25914 : CHIvec_CHI(GEN v)
    9024       25914 : { return mkvec4(gel(v,1), gel(v,2), gel(v,3), gel(v,6)); }
    9025             : /* character order */
    9026             : static long
    9027       66311 : CHIvec_ord(GEN v) { return itou(gel(v,3)); }
    9028             : /* character exponents, i.e. t such that chi(n) = e(t) */
    9029             : static GEN
    9030      626913 : CHIvec_expo(GEN v) { return gel(v,4); }
    9031             : /* character values chi(n) */
    9032             : static GEN
    9033    27670174 : CHIvec_val(GEN v) { return gel(v,5); }
    9034             : /* CHI(n) */
    9035             : static GEN
    9036    27645779 : mychareval(GEN v, long n)
    9037             : {
    9038    27645779 :   long N = CHIvec_N(v), ind = n%N;
    9039    27645779 :   if (ind <= 0) ind += N;
    9040    27645779 :   return gel(CHIvec_val(v), ind);
    9041             : }
    9042             : /* return c such that CHI(n) = e(c / ordz) or -1 if (n,N) > 1 */
    9043             : static long
    9044      626913 : mycharexpo(GEN v, long n)
    9045             : {
    9046      626913 :   long N = CHIvec_N(v), ind = n%N;
    9047      626913 :   if (ind <= 0) ind += N;
    9048      626913 :   return CHIvec_expo(v)[ind];
    9049             : }
    9050             : /* faster than mfcharparity */
    9051             : static long
    9052       54754 : CHIvec_parity(GEN v) { return mycharexpo(v,-1) ? -1: 1; }
    9053             : /**************************************************************************/
    9054             : 
    9055             : static ulong
    9056       77791 : sigchi2_Fl(long k, GEN CHI1vec, GEN CHI2vec, long n, GEN vz, ulong p)
    9057             : {
    9058       77791 :   pari_sp av = avma;
    9059       77791 :   long ordz = lg(vz)-2, i, l, n1, n2;
    9060       77791 :   ulong S = 0;
    9061       77791 :   GEN D = sigchi2_dec(n, CHIvec_N(CHI1vec), CHIvec_N(CHI2vec), &n1, &n2);
    9062       77791 :   if (!D) return gc_ulong(av,S);
    9063       73108 :   D = divisorsu_fact(D);
    9064       73108 :   l = lg(D);
    9065      276444 :   for (i = 1; i < l; i++)
    9066             :   { /* S += d^(k-1)*chi1(d)*chi2(n/d) */
    9067      203336 :     long a, d = n2*D[i], nd = n1*D[l-i]; /* (d,N1)=1, (n/d,N2)=1 */
    9068      203336 :     a = mycharexpo(CHI2vec, nd) + mycharexpo(CHI1vec, d);
    9069      203336 :     if (a >= ordz) a -= ordz;
    9070      203336 :     S = Fl_add(S, Qab_Czeta_Fl(a, vz, Fl_powu(d,k-1,p), p), p);
    9071             :   }
    9072       73108 :   return gc_ulong(av,S);
    9073             : }
    9074             : 
    9075             : /**********************************************************************/
    9076             : /* Fourier expansions of Eisenstein series                            */
    9077             : /**********************************************************************/
    9078             : /* L(CHI_t,0) / 2, CHI_t(n) = CHI(n)(t/n) as a character modulo N*t,
    9079             :  * order(CHI) | ord != 0 */
    9080             : static GEN
    9081        2618 : charLFwt1(long N, GEN CHI, long ord, long t)
    9082             : {
    9083             :   GEN S;
    9084             :   long r, vt;
    9085             : 
    9086        2618 :   if (N == 1 && t == 1) return mkfrac(gen_m1,stoi(4));
    9087        2618 :   S = gen_0; vt = varn(mfcharpol(CHI));
    9088      295435 :   for (r = 1; r < N; r++)
    9089             :   { /* S += r*chi(r) */
    9090             :     long a, c;
    9091      292817 :     if (ugcd(N,r) != 1) continue;
    9092      233310 :     a = mfcharevalord(CHI,r,ord);
    9093      233310 :     c = (t != 1 && kross(t, r) < 0)? -r: r;
    9094      233310 :     S = gadd(S, Qab_Czeta(a, ord, stoi(c), vt));
    9095             :   }
    9096        2618 :   return gdivgs(S, -2*N);
    9097             : }
    9098             : /* L(CHI,0) / 2, mod p */
    9099             : static ulong
    9100        2002 : charLFwt1_Fl(GEN CHIvec, GEN vz, ulong p)
    9101             : {
    9102        2002 :   long r, m = CHIvec_N(CHIvec);
    9103             :   ulong S;
    9104        2002 :   if (m == 1) return Rg_to_Fl(mkfrac(gen_m1,stoi(4)), p);
    9105        2002 :   S = 0;
    9106       95977 :   for (r = 1; r < m; r++)
    9107             :   { /* S += r*chi(r) */
    9108       93975 :     long a = mycharexpo(CHIvec,r);
    9109       93975 :     if (a < 0) continue;
    9110       91616 :     S = Fl_add(S, Qab_Czeta_Fl(a, vz, r, p), p);
    9111             :   }
    9112        2002 :   return Fl_div(Fl_neg(S,p), 2*m, p);
    9113             : }
    9114             : /* L(CHI_t,1-k) / 2, CHI_t(n) = CHI(n) * (t/n), order(CHI) | ord != 0;
    9115             :  * assume conductor of CHI_t divides N */
    9116             : static GEN
    9117        4557 : charLFwtk(long N, long k, GEN CHI, long ord, long t)
    9118             : {
    9119             :   GEN S, P, dS;
    9120             :   long r, vt;
    9121             : 
    9122        4557 :   if (k == 1) return charLFwt1(N, CHI, ord, t);
    9123        1939 :   if (N == 1 && t == 1) return gdivgs(bernfrac(k),-2*k);
    9124        1176 :   vt = varn(mfcharpol(CHI));
    9125        1176 :   P = bern_init(N, k, &dS);
    9126        1176 :   dS = mul_denom(dS, stoi(-2*N*k));
    9127       17633 :   for (r = 1, S = gen_0; r < N; r++)
    9128             :   { /* S += P(r)*chi(r) */
    9129             :     long a;
    9130             :     GEN C;
    9131       16457 :     if (ugcd(r,N) != 1) continue;
    9132       13860 :     a = mfcharevalord(CHI,r,ord);
    9133       13860 :     C = ZX_Z_eval(P, utoi(r));
    9134       13860 :     if (t != 1 && kross(t, r) < 0) C = gneg(C);
    9135       13860 :     S = gadd(S, Qab_Czeta(a, ord, C, vt));
    9136             :   }
    9137        1176 :   return gdiv(S, dS);
    9138             : }
    9139             : /* L(CHI,1-k) / 2, mod p */
    9140             : static ulong
    9141        3227 : charLFwtk_Fl(long k, GEN CHIvec, GEN vz, ulong p)
    9142             : {
    9143             :   GEN P, dS;
    9144             :   long r, m;
    9145             :   ulong S, d;
    9146        3227 :   if (k == 1) return charLFwt1_Fl(CHIvec, vz, p);
    9147        1225 :   m = CHIvec_N(CHIvec);
    9148        1225 :   if (m == 1) return Rg_to_Fl(gdivgs(bernfrac(k),-2*k), p);
    9149         819 :   P = ZX_to_Flx(bern_init(m, k, &dS), p);
    9150       20167 :   for (r = 1, S = 0; r < m; r++)
    9151             :   { /* S += P(r)*chi(r) */
    9152       19348 :     long a = mycharexpo(CHIvec,r);
    9153       19348 :     if (a < 0) continue;
    9154       18088 :     S = Fl_add(S, Qab_Czeta_Fl(a, vz, Flx_eval(P,r,p), p), p);
    9155             :   }
    9156         819 :   d = (2 * k * m) % p; if (dS) d = Fl_mul(d, umodiu(dS, p), p);
    9157         819 :   return Fl_div(Fl_neg(S,p), d, p);
    9158             : }
    9159             : 
    9160             : static GEN
    9161        8365 : mfeisenstein2_0(long k, GEN CHI1, GEN CHI2, long ord)
    9162             : {
    9163        8365 :   long N1 = mfcharmodulus(CHI1), N2 = mfcharmodulus(CHI2);
    9164        8365 :   if (k == 1 && N1 == 1) return charLFwtk(N2, 1, CHI2, ord, 1);
    9165        5754 :   if (N2 == 1) return charLFwtk(N1, k, CHI1, ord, 1);
    9166        4025 :   return gen_0;
    9167             : }
    9168             : static ulong
    9169        5054 : mfeisenstein2_0_Fl(long k, GEN CHI1vec, GEN CHI2vec, GEN vz, ulong p)
    9170             : {
    9171        5054 :   if (k == 1 && CHIvec_N(CHI1vec) == 1)
    9172        2002 :     return charLFwtk_Fl(k, CHI2vec, vz, p);
    9173        3052 :   else if (CHIvec_N(CHI2vec) == 1)
    9174        1225 :     return charLFwtk_Fl(k, CHI1vec, vz, p);
    9175        1827 :   else return 0;
    9176             : }
    9177             : static GEN
    9178         140 : NK_eisen2(long k, GEN CHI1, GEN CHI2, long ord)
    9179             : {
    9180         140 :   long o, N = mfcharmodulus(CHI1)*mfcharmodulus(CHI2);
    9181         140 :   GEN CHI = mfcharmul(CHI1, CHI2);
    9182         140 :   o = mfcharorder(CHI);
    9183         140 :   if ((ord & 3) == 2) ord >>= 1;
    9184         140 :   if ((o & 3) == 2) o >>= 1;
    9185         140 :   if (ord != o) pari_err_IMPL("mfeisenstein for these characters");
    9186         133 :   return mkNK(N, k, CHI);
    9187             : }
    9188             : static GEN
    9189         343 : mfeisenstein_prim(long k, GEN CHI1, GEN CHI2)
    9190             : {
    9191             :   long ord, vt;
    9192             :   GEN E0, NK, vchi, T;
    9193         343 :   if (!CHI2)
    9194             :   { /* E_k(chi1) */
    9195         203 :     vt = varn(mfcharpol(CHI1));
    9196         203 :     ord = mfcharorder(CHI1);
    9197         203 :     NK = mkNK(mfcharmodulus(CHI1), k, CHI1);
    9198         203 :     E0 = charLFwtk(mfcharmodulus(CHI1), k, CHI1, ord, 1);
    9199         203 :     vchi = mkvec3(E0, mkvec(mfcharpol(CHI1)), CHI1);
    9200         203 :     return tag(t_MF_EISEN, NK, vchi);
    9201             :   }
    9202             :   /* E_k(chi1,chi2) */
    9203         140 :   vt = varn(mfcharpol(CHI1));
    9204         140 :   ord = ulcm(mfcharorder(CHI1), mfcharorder(CHI2));
    9205         140 :   NK = NK_eisen2(k, CHI1, CHI2, ord);
    9206         133 :   E0 = mfeisenstein2_0(k, CHI1, CHI2, ord);
    9207         133 :   T = mkvec(polcyclo(ord, vt));
    9208         133 :   vchi = mkvec4(E0, T, CHI1, CHI2);
    9209         133 :   return tag2(t_MF_EISEN, NK, vchi, mkvecsmall2(ord,0));
    9210             : }
    9211             : static GEN
    9212         378 : mfeisenstein_i(long k, GEN CHI1, GEN CHI2)
    9213             : {
    9214         378 :   long s = 1, i, f = 1, N1 = 1, Nf, lD;
    9215             :   GEN P, E, D, L1, L2;
    9216         378 :   if (CHI2) { CHI2 = get_mfchar(CHI2); if (mfcharparity(CHI2) < 0) s = -s; }
    9217         378 :   if (CHI1)
    9218             :   {
    9219         154 :     CHI1 = get_mfchar(CHI1);
    9220         140 :     N1 = mfcharmodulus(CHI1);
    9221         140 :     CHI1 = mfchartoprimitive(CHI1, &f);
    9222         140 :     if (mfcharparity(CHI1) < 0) s = -s;
    9223             :   } else
    9224         224 :     CHI1 = mfchartrivial();
    9225         364 :   if (s != m1pk(k)) return mftrivial();
    9226         343 :   E = mfeisenstein_prim(k,CHI1,CHI2);
    9227         336 :   if (N1 == f) return E;
    9228          14 :   Nf = N1 / f;
    9229          14 :   P = gel(factoru(u_ppo(Nf, f)), 1);
    9230          14 :   D = divisorsu_moebius(P); lD = lg(D);
    9231          14 :   L1 = cgetg(lD, t_VEC); L2 = cgetg(lD, t_VEC);
    9232          42 :   for (i = 1; i < lD; i++)
    9233             :   { /* m = mu(g)*g, g | Nf, coprime to f, squarefree */
    9234          28 :     long m = D[i], g = labs(m);
    9235          28 :     GEN c = gdiv(mfchareval(CHI1,g),powuu(g,k));
    9236          28 :     gel(L1,i) = m < 0 ? gneg(c): c;
    9237          28 :     gel(L2,i) = mfbd_i(E, Nf / g);
    9238             :   }
    9239          14 :   return mflinear(L2, L1);
    9240             : }
    9241             : 
    9242             : GEN
    9243         378 : mfeisenstein(long k, GEN CHI1, GEN CHI2)
    9244             : {
    9245         378 :   pari_sp av = avma;
    9246         378 :   if (k < 1) pari_err_DOMAIN("mfeisenstein", "k", "<", gen_1, stoi(k));
    9247         378 :   return gc_GEN(av, mfeisenstein_i(k, CHI1, CHI2));
    9248             : }
    9249             : 
    9250             : static GEN
    9251        2646 : mfeisenstein2all(long N0, GEN NK, long k, GEN CHI1, GEN CHI2, GEN T, long o)
    9252             : {
    9253        2646 :   GEN E, E0 = mfeisenstein2_0(k, CHI1,CHI2, o), vchi = mkvec4(E0, T, CHI1,CHI2);
    9254        2646 :   long j, d = (lg(T)==4)? itou(gmael(T,3,1)): 1;
    9255        2646 :   E = cgetg(d+1, t_VEC);
    9256        5411 :   for (j=1; j<=d; j++) gel(E,j) = tag2(t_MF_EISEN, NK,vchi,mkvecsmall2(o,j-1));
    9257        2646 :   return mfbdall(E, N0 / mf_get_N(gel(E,1)));
    9258             : }
    9259             : 
    9260             : /* list of characters on G = (Z/NZ)^*, v[i] = NULL if (i,N) > 1, else
    9261             :  * the conductor of Conrey label i, [conductor, primitive char].
    9262             :  * Trivial chi (label 1) comes first */
    9263             : static GEN
    9264        1169 : zncharsG(GEN G)
    9265             : {
    9266        1169 :   long i, l, N = itou(znstar_get_N(G));
    9267             :   GEN vCHI, V;
    9268        1169 :   if (N == 1) return mkvec2(gen_1,cgetg(1,t_COL));
    9269        1169 :   vCHI = const_vec(N,NULL);
    9270        1169 :   V = cyc2elts(znstar_get_conreycyc(G));
    9271        1169 :   l = lg(V);
    9272      207739 :   for (i = 1; i < l; i++)
    9273             :   {
    9274      206570 :     GEN chi0, chi = zc_to_ZC(gel(V,i)), n, F;
    9275      206570 :     F = znconreyconductor(G, chi, &chi0);
    9276      206570 :     if (typ(F) != t_INT) F = gel(F,1);
    9277      206570 :     n = znconreyexp(G, chi);
    9278      206570 :     gel(vCHI, itos(n)) = mkvec2(chi0, F);
    9279             :   }
    9280        1169 :   return vCHI;
    9281             : }
    9282             : 
    9283             : /* CHI primitive, f(CHI) | N. Return pairs (CHI1,CHI2) both primitive
    9284             :  * such that f(CHI1)*f(CHI2) | N and CHI1 * CHI2 = CHI;
    9285             :  * if k = 1, CHI1 is even; if k = 2, omit (1,1) if CHI = 1 */
    9286             : static GEN
    9287        1428 : mfeisensteinbasis_i(long N0, long k, GEN CHI)
    9288             : {
    9289        1428 :   GEN G = gel(CHI,1), chi = gel(CHI,2), vT = const_vec(myeulerphiu(N0), NULL);
    9290        1428 :   GEN CHI0, GN, chiN, Lchi, LG, V, RES, NK, T, C = mfcharpol(CHI);
    9291        1428 :   long i, j, l, n, n1, N, ord = mfcharorder(CHI);
    9292        1428 :   long F = mfcharmodulus(CHI), vt = varn(mfcharpol(CHI));
    9293             : 
    9294        1428 :   CHI0 = (F == 1)? CHI: mfchartrivial();
    9295        1428 :   j = 1; RES = cgetg(N0+1, t_VEC);
    9296        1428 :   T = gel(vT,ord) = Qab_trace_init(ord, ord, C, C);
    9297        1428 :   if (F != 1 || k != 2)
    9298             :   { /* N1 = 1 */
    9299        1274 :     NK = mkNK(F, k, CHI);
    9300        1274 :     gel(RES, j++) = mfeisenstein2all(N0, NK, k, CHI0, CHI, T, ord);
    9301        1274 :     if (F != 1 && k != 1)
    9302         329 :       gel(RES, j++) = mfeisenstein2all(N0, NK, k, CHI, CHI0, T, ord);
    9303             :   }
    9304        1428 :   if (N0 == 1) { setlg(RES,j); return RES; }
    9305        1330 :   GN = G; chiN = chi;
    9306        1330 :   if (F == N0) N = N0;
    9307             :   else
    9308             :   {
    9309         728 :     GEN faN = myfactoru(N0), P = gel(faN,1), E = gel(faN,2);
    9310         728 :     long lP = lg(P);
    9311        1876 :     for (i = N = 1; i < lP; i++)
    9312             :     {
    9313        1148 :       long p = P[i];
    9314        1148 :       N *= upowuu(p, maxuu(E[i]/2, z_lval(F,p)));
    9315             :     }
    9316         728 :     if ((N & 3) == 2) N >>= 1;
    9317         728 :     if (N == 1) { setlg(RES,j); return RES; }
    9318         567 :     if (F != N)
    9319             :     {
    9320         133 :       GN = znstar0(utoipos(N),1);
    9321         133 :       chiN = zncharinduce(G, chi, GN);
    9322             :     }
    9323             :   }
    9324        1169 :   LG = const_vec(N, NULL); /* LG[d] = znstar(d,1) or NULL */
    9325        1169 :   gel(LG,1) = gel(CHI0,1);
    9326        1169 :   gel(LG,F) = G;
    9327        1169 :   gel(LG,N) = GN;
    9328        1169 :   Lchi = coprimes_zv(N);
    9329        1169 :   n = itou(znconreyexp(GN,chiN));
    9330        1169 :   V = zncharsG(GN); l = lg(V);
    9331      263305 :   for (n1 = 2; n1 < l; n1++) /* skip 1 (trivial char) */
    9332             :   {
    9333      262136 :     GEN v = gel(V,n1), w, chi1, chi2, G1, G2, CHI1, CHI2;
    9334             :     long N12, N1, N2, no, o12, t, m;
    9335      262136 :     if (!Lchi[n1] || n1 == n) continue; /* skip trivial chi2 */
    9336      204197 :     chi1 = gel(v,1); N1 = itou(gel(v,2)); /* conductor of chi1 */
    9337      204197 :     w = gel(V, Fl_div(n,n1,N));
    9338      204197 :     chi2 = gel(w,1); N2 = itou(gel(w,2)); /* conductor of chi2 */
    9339      204197 :     N12 = N1 * N2;
    9340      204197 :     if (N0 % N12) continue;
    9341             : 
    9342        1771 :     G1 = gel(LG,N1); if (!G1) gel(LG,N1) = G1 = znstar0(utoipos(N1), 1);
    9343        1771 :     if (k == 1 && zncharisodd(G1,chi1)) continue;
    9344        1043 :     G2 = gel(LG,N2); if (!G2) gel(LG,N2) = G2 = znstar0(utoipos(N2), 1);
    9345        1043 :     CHI1 = mfcharGL(G1, chi1);
    9346        1043 :     CHI2 = mfcharGL(G2, chi2);
    9347        1043 :     o12 = ulcm(mfcharorder(CHI1), mfcharorder(CHI2));
    9348             :     /* remove Galois orbit: same trace */
    9349        1043 :     no = Fl_powu(n1, ord, N);
    9350        1414 :     for (t = 1+ord, m = n1; t <= o12; t += ord)
    9351             :     { /* m <-> CHI1^t, if t in Gal(Q(chi1,chi2)/Q), omit (CHI1^t,CHI2^t) */
    9352         371 :       m = Fl_mul(m, no, N); if (!m) break;
    9353         371 :       if (ugcd(t, o12) == 1) Lchi[m] = 0;
    9354             :     }
    9355        1043 :     T = gel(vT,o12);
    9356        1043 :     if (!T) T = gel(vT,o12) = Qab_trace_init(o12, ord, polcyclo(o12,vt), C);
    9357        1043 :     NK = mkNK(N12, k, CHI);
    9358        1043 :     gel(RES, j++) = mfeisenstein2all(N0, NK, k, CHI1, CHI2, T, o12);
    9359             :   }
    9360        1169 :   setlg(RES,j); return RES;
    9361             : }
    9362             : 
    9363             : static GEN
    9364         721 : mfbd_E2(GEN E2, long d, GEN CHI)
    9365             : {
    9366         721 :   GEN E2d = mfbd_i(E2, d);
    9367         721 :   GEN F = mkvec2(E2, E2d), L = mkvec2(gen_1, utoineg(d));
    9368             :   /* cannot use mflinear_i: E2 and E2d do not have the same level */
    9369         721 :   return tag3(t_MF_LINEAR, mkNK(d,2,CHI), F, L, gen_1);
    9370             : }
    9371             : /* C-basis of E_k(Gamma_0(N),chi). If k = 1, the first basis element must not
    9372             :  * vanish at oo [used in mf1basis]. Here E_1(CHI), whose q^0 coefficient
    9373             :  * does not vanish (since L(CHI,0) does not) *if* CHI is not trivial; which
    9374             :  * must be the case in weight 1.
    9375             :  *
    9376             :  * (k>=3): In weight k >= 3, basis is B(d) E(CHI1,(CHI/CHI1)_prim), where
    9377             :  * CHI1 is primitive modulo N1, and if N2 is the conductor of CHI/CHI1
    9378             :  * then d*N1*N2 | N.
    9379             :  * (k=2): In weight k=2, same if CHI is nontrivial. If CHI is trivial, must
    9380             :  * not take CHI1 trivial, and must add E_2(tau)-dE_2(d tau)), where
    9381             :  * d|N, d > 1.
    9382             :  * (k=1): In weight k=1, same as k >= 3 except that we restrict to CHI1 even */
    9383             : static GEN
    9384        1456 : mfeisensteinbasis(long N, long k, GEN CHI)
    9385             : {
    9386             :   long i, F;
    9387             :   GEN L;
    9388        1456 :   if (badchar(N, k, CHI)) return cgetg(1, t_VEC);
    9389        1456 :   if (k == 0) return mfcharistrivial(CHI)? mkvec(mf1()): cgetg(1, t_VEC);
    9390        1428 :   CHI = mfchartoprimitive(CHI, &F);
    9391        1428 :   L = mfeisensteinbasis_i(N, k, CHI);
    9392        1428 :   if (F == 1 && k == 2)
    9393             :   {
    9394         154 :     GEN v, E2 = mfeisenstein(2, NULL, NULL), D = mydivisorsu(N);
    9395         154 :     long nD = lg(D)-1;
    9396         154 :     v = cgetg(nD, t_VEC); L = vec_append(L,v);
    9397         868 :     for (i = 1; i < nD; i++) gel(v,i) = mfbd_E2(E2, D[i+1], CHI);
    9398             :   }
    9399        1428 :   return lg(L) == 1? L: shallowconcat1(L);
    9400             : }
    9401             : 
    9402             : static GEN
    9403          77 : not_in_space(GEN F, long flag)
    9404             : {
    9405          77 :   if (!flag) err_space(F);
    9406          70 :   return cgetg(1, t_COL);
    9407             : }
    9408             : /* when flag set, no error */
    9409             : GEN
    9410        1029 : mftobasis(GEN mf, GEN F, long flag)
    9411             : {
    9412        1029 :   pari_sp av2, av = avma;
    9413             :   GEN G, v, y, gk;
    9414        1029 :   long N, B, ismf = checkmf_i(F);
    9415             : 
    9416        1029 :   mf = checkMF(mf);
    9417        1029 :   if (ismf)
    9418             :   {
    9419         938 :     if (mfistrivial(F)) return zerocol(MF_get_dim(mf));
    9420         931 :     if (!mf_same_k(mf, F) || !mf_same_CHI(mf, F)) return not_in_space(F, flag);
    9421             :   }
    9422         980 :   N = MF_get_N(mf);
    9423         980 :   gk = MF_get_gk(mf);
    9424         980 :   if (ismf)
    9425             :   {
    9426         889 :     long NF = mf_get_N(F);
    9427         889 :     B = maxuu(mfsturmNgk(NF,gk), mfsturmNgk(N,gk)) + 1;
    9428         889 :     v = mfcoefs_i(F,B,1);
    9429             :   }
    9430             :   else
    9431             :   {
    9432          91 :     B = mfsturmNgk(N, gk) + 1;
    9433          91 :     switch(typ(F))
    9434             :     { /* F(0),...,F(lg(v)-2) */
    9435          63 :       case t_SER: v = sertocol(F); settyp(v,t_VEC); break;
    9436          14 :       case t_VEC: v = F; break;
    9437           7 :       case t_COL: v = shallowtrans(F); break;
    9438           7 :       default: pari_err_TYPE("mftobasis",F);
    9439             :                v = NULL;/*LCOV_EXCL_LINE*/
    9440             :     }
    9441          84 :     if (flag) B = minss(B, lg(v)-2);
    9442             :   }
    9443         973 :   y = mftobasis_i(mf, v);
    9444         973 :   if (typ(y) == t_VEC)
    9445             :   {
    9446          21 :     if (flag) return gc_GEN(av, y);
    9447           0 :     pari_err(e_MISC, "not enough coefficients in mftobasis");
    9448             :   }
    9449         952 :   av2 = avma;
    9450         952 :   if (MF_get_space(mf) == mf_FULL || mfsturm(mf)+1 == B) return y;
    9451         476 :   G = mflinear(mf, y);
    9452         476 :   if (!gequal(v, mfcoefs_i(G, lg(v)-2,1))) y = NULL;
    9453         476 :   if (!y) { set_avma(av); return not_in_space(F, flag); }
    9454         441 :   set_avma(av2); return gc_upto(av, y);
    9455             : }
    9456             : 
    9457             : /* assume N > 0; first cusp is always 0 */
    9458             : static GEN
    9459          49 : mfcusps_i(long N)
    9460             : {
    9461             :   long i, c, l;
    9462             :   GEN D, v;
    9463             : 
    9464          49 :   if (N == 1) return mkvec(gen_0);
    9465          49 :   D = mydivisorsu(N); l = lg(D); /* left on stack */
    9466          49 :   c = mfnumcuspsu_fact(myfactoru(N));
    9467          49 :   v = cgetg(c + 1, t_VEC);
    9468         350 :   for (i = c = 1; i < l; i++)
    9469             :   {
    9470         301 :     long C = D[i], NC = D[l-i], lima = ugcd(C, NC), A0, A;
    9471         889 :     for (A0 = 0; A0 < lima; A0++)
    9472         588 :       if (ugcd(A0, lima) == 1)
    9473             :       {
    9474         539 :         A = A0; while (ugcd(A,C) > 1) A += lima;
    9475         392 :         gel(v, c++) = uutoQ(A, C);
    9476             :       }
    9477             :   }
    9478          49 :   return v;
    9479             : }
    9480             : /* List of cusps of Gamma_0(N) */
    9481             : GEN
    9482          28 : mfcusps(GEN gN)
    9483             : {
    9484             :   long N;
    9485             :   GEN mf;
    9486          28 :   if (typ(gN) == t_INT) N = itos(gN);
    9487          14 :   else if ((mf = checkMF_i(gN))) N = MF_get_N(mf);
    9488           0 :   else { pari_err_TYPE("mfcusps", gN); N = 0; }
    9489          28 :   if (N <= 0) pari_err_DOMAIN("mfcusps", "N", "<=", gen_0, stoi(N));
    9490          28 :   return mfcusps_i(N);
    9491             : }
    9492             : 
    9493             : long
    9494         315 : mfcuspisregular(GEN NK, GEN cusp)
    9495             : {
    9496             :   long v, N, dk, nk, t, o;
    9497             :   GEN mf, CHI, go, A, C, g, c, d;
    9498         315 :   if ((mf = checkMF_i(NK)))
    9499             :   {
    9500          49 :     GEN gk = MF_get_gk(mf);
    9501          49 :     N = MF_get_N(mf);
    9502          49 :     CHI = MF_get_CHI(mf);
    9503          49 :     Qtoss(gk, &nk, &dk);
    9504             :   }
    9505             :   else
    9506         266 :     checkNK2(NK, &N, &nk, &dk, &CHI, 0);
    9507         315 :   if (typ(cusp) == t_INFINITY) return 1;
    9508         315 :   if (typ(cusp) == t_FRAC) { A = gel(cusp,1); C = gel(cusp,2); }
    9509          28 :   else { A = cusp; C = gen_1; }
    9510         315 :   g = diviuexact(mului(N,C), ugcd(N, Fl_sqr(umodiu(C,N), N)));
    9511         315 :   c = mulii(negi(C),g);
    9512         315 :   d = addiu(mulii(A,g), 1);
    9513         315 :   if (!CHI) return 1;
    9514         315 :   go = gmfcharorder(CHI);
    9515         315 :   v = vali(go); if (v < 2) go = shifti(go, 2-v);
    9516         315 :   t = itou( znchareval(gel(CHI,1), gel(CHI,2), d, go) );
    9517         315 :   if (dk == 1) return t == 0;
    9518         154 :   o = itou(go);
    9519         154 :   if (kronecker(c,d) < 0) t = Fl_add(t, o/2, o);
    9520         154 :   if (Mod4(d) == 1) return t == 0;
    9521          14 :   t = Fl_sub(t, Fl_mul(o/4, nk, o), o);
    9522          14 :   return t == 0;
    9523             : }
    9524             : 
    9525             : /* Some useful closures */
    9526             : 
    9527             : /* sum_{d|n} d^k */
    9528             : static GEN
    9529       48020 : mysumdivku(ulong n, ulong k)
    9530             : {
    9531       48020 :   GEN fa = myfactoru(n);
    9532       48020 :   return k == 1? usumdiv_fact(fa): usumdivk_fact(fa,k);
    9533             : }
    9534             : static GEN
    9535         882 : c_Ek(long n, long d, GEN F)
    9536             : {
    9537         882 :   GEN E = cgetg(n + 2, t_VEC), C = gel(F,2);
    9538         882 :   long i, k = mf_get_k(F);
    9539         882 :   gel (E, 1) = gen_1;
    9540       26264 :   for (i = 1; i <= n; i++)
    9541             :   {
    9542       25382 :     pari_sp av = avma;
    9543       25382 :     gel(E, i+1) = gc_upto(av, gmul(C, mysumdivku(i*d, k-1)));
    9544             :   }
    9545         882 :   return E;
    9546             : }
    9547             : 
    9548             : GEN
    9549         406 : mfEk(long k)
    9550             : {
    9551         406 :   pari_sp av = avma;
    9552             :   GEN E0, NK;
    9553         406 :   if (k < 0 || odd(k)) pari_err_TYPE("mfEk [incorrect k]", stoi(k));
    9554         406 :   if (!k) return mf1();
    9555         399 :   E0 = gdivsg(-2*k, bernfrac(k));
    9556         399 :   NK = mkNK(1,k,mfchartrivial());
    9557         399 :   return gc_GEN(av, tag(t_MF_Ek, NK, E0));
    9558             : }
    9559             : 
    9560             : GEN
    9561          56 : mfDelta(void)
    9562             : {
    9563          56 :   pari_sp av = avma;
    9564          56 :   return gc_GEN(av, tag0(t_MF_DELTA, mkNK(1,12,mfchartrivial())));
    9565             : }
    9566             : 
    9567             : GEN
    9568         805 : mfTheta(GEN psi)
    9569             : {
    9570         805 :   pari_sp av = avma;
    9571             :   GEN N, gk, psi2;
    9572             :   long par;
    9573         805 :   if (!psi) { psi = mfchartrivial(); N = utoipos(4); par = 1; }
    9574             :   else
    9575             :   {
    9576             :     long FC;
    9577          21 :     psi = get_mfchar(psi);
    9578          21 :     FC = mfcharconductor(psi);
    9579          21 :     if (mfcharmodulus(psi) != FC)
    9580           0 :       pari_err_TYPE("mfTheta [nonprimitive character]", psi);
    9581          21 :     par = mfcharparity(psi);
    9582          21 :     N = shifti(sqru(FC),2);
    9583             :   }
    9584         805 :   if (par > 0) { gk = ghalf; psi2 = psi; }
    9585           7 :   else { gk = gsubsg(2, ghalf); psi2 = mfcharmul(psi, get_mfchar(stoi(-4))); }
    9586         805 :   return gc_GEN(av, tag(t_MF_THETA, mkgNK(N, gk, psi2, pol_x(1)), psi));
    9587             : }
    9588             : 
    9589             : /* Output 0 if not desired eta product: if flag=0 (default) require
    9590             :  * holomorphic at cusps. If flag set, accept meromorphic, but sill in some
    9591             :  * modular function space */
    9592             : GEN
    9593         210 : mffrometaquo(GEN eta, long flag)
    9594             : {
    9595         210 :   pari_sp av = avma;
    9596             :   GEN NK, N, k, BR, P;
    9597         210 :   long v, cusp = 0;
    9598         210 :   if (!etaquotype(&eta, &N,&k,&P, &v, NULL, flag? NULL: &cusp) || cusp < 0)
    9599          14 :     return gc_const(av, gen_0);
    9600         196 :   if (lg(gel(eta,1)) == 1) { set_avma(av); return mf1(); }
    9601         189 :   BR = mkvec2(ZV_to_zv(gel(eta,1)), ZV_to_zv(gel(eta,2)));
    9602         189 :   if (v < 0) v = 0;
    9603         189 :   NK = mkgNK(N, k, get_mfchar(P), pol_x(1));
    9604         189 :   return gc_GEN(av, tag2(t_MF_ETAQUO, NK, BR, utoi(v)));
    9605             : }
    9606             : 
    9607             : /* Q^(-r) */
    9608             : static GEN
    9609         375 : RgXn_negpow(GEN Q, long r, long L)
    9610             : {
    9611         375 :   if (r < 0) r = -r; else Q = RgXn_inv_i(Q, L);
    9612         375 :   if (r != 1) Q = RgXn_powu_i(Q, r, L);
    9613         375 :   return Q;
    9614             : }
    9615             : /* flag same as in mffrometaquo: if set, accept meromorphic. */
    9616             : static GEN
    9617          49 : mfisetaquo_i(GEN F, long flag)
    9618             : {
    9619             :   GEN gk, P, E, M, S, G, CHI, v, w;
    9620             :   long b, l, L, N, vS, m, j;
    9621          49 :   const long bextra = 10;
    9622             : 
    9623          49 :   if (!checkmf_i(F)) pari_err_TYPE("mfisetaquo",F);
    9624          49 :   CHI = mf_get_CHI(F); if (mfcharorder(CHI) > 2) return NULL;
    9625          49 :   N = mf_get_N(F);
    9626          49 :   gk = mf_get_gk(F);
    9627          49 :   b = mfsturmNgk(N, gk);
    9628          49 :   L = maxss(N, b) + bextra;
    9629          49 :   S = mfcoefs_i(F, L, 1);
    9630          49 :   if (!RgV_is_ZV(S)) return NULL;
    9631         889 :   for (vS = 1; vS <= L+1; vS++)
    9632         889 :     if (signe(gel(S,vS))) break;
    9633          49 :   vS--;
    9634          49 :   if (vS >= bextra - 1) { L += vS; S = mfcoefs_i(F, L, 1); }
    9635          49 :   if (vS) { S = vecslice(S, vS+1, L+1); L -= vS; }
    9636          49 :   S = RgV_to_RgX(S, 0); l = lg(S)-2;
    9637          49 :   P = cgetg(l, t_COL);
    9638          49 :   E = cgetg(l, t_COL); w = v = gen_0; /* w = weight, v = valuation */
    9639        1908 :   for (m = j = 1; m+2 < lg(S); m++)
    9640             :   {
    9641        1866 :     GEN c = gel(S,m+2);
    9642             :     long r;
    9643        1866 :     if (is_bigint(c)) return NULL;
    9644        1859 :     r = -itos(c);
    9645        1859 :     if (r)
    9646             :     {
    9647         375 :       S = ZXn_mul(S, RgXn_negpow(eta_ZXn(m, L), r, L), L);
    9648         375 :       gel(P,j) = utoipos(m);
    9649         375 :       gel(E,j) = stoi(r);
    9650         375 :       v = addmuliu(v, gel(E,j), m);
    9651         375 :       w = addis(w, r);
    9652         375 :       j++;
    9653             :     }
    9654             :   }
    9655          42 :   if (!equalii(w, gmul2n(gk, 1)) || (!flag && !equalii(v, muluu(24,vS))))
    9656           7 :     return NULL;
    9657          35 :   setlg(P, j);
    9658          35 :   setlg(E, j); M = mkmat2(P, E); G = mffrometaquo(M, flag);
    9659          35 :   return (typ(G) != t_INT
    9660          35 :           && (mfsturmmf(G) <= b + bextra || mfisequal(F, G, b)))? M: NULL;
    9661             : }
    9662             : GEN
    9663          49 : mfisetaquo(GEN F, long flag)
    9664             : {
    9665          49 :   pari_sp av = avma;
    9666          49 :   GEN M = mfisetaquo_i(F, flag);
    9667          49 :   return M? gc_GEN(av, M): gc_const(av, gen_0);
    9668             : }
    9669             : 
    9670             : #if 0
    9671             : /* number of primitive characters modulo N */
    9672             : static ulong
    9673             : numprimchars(ulong N)
    9674             : {
    9675             :   GEN fa, P, E;
    9676             :   long i, l;
    9677             :   ulong n;
    9678             :   if ((N & 3) == 2) return 0;
    9679             :   fa = myfactoru(N);
    9680             :   P = gel(fa,1); l = lg(P);
    9681             :   E = gel(fa,2);
    9682             :   for (i = n = 1; i < l; i++)
    9683             :   {
    9684             :     ulong p = P[i], e = E[i];
    9685             :     if (e == 2) n *= p-2; else n *= (p-1)*(p-1)*upowuu(p,e-2);
    9686             :   }
    9687             :   return n;
    9688             : }
    9689             : #endif
    9690             : 
    9691             : /* Space generated by products of two Eisenstein series */
    9692             : 
    9693             : static int
    9694       74431 : cmp_small_priority(void *E, GEN a, GEN b)
    9695             : {
    9696       74431 :   GEN prio = (GEN)E;
    9697       74431 :   return cmpss(prio[(long)a], prio[(long)b]);
    9698             : }
    9699             : static long
    9700        1302 : znstar_get_expo(GEN G) { return itou(cyc_get_expo(znstar_get_cyc(G))); }
    9701             : 
    9702             : /* Return [vchi, bymod, vG]:
    9703             :  * vG[f] = znstar(f,1) for f a conductor of (at least) a char mod N; else NULL
    9704             :  * bymod[f] = vecsmall of conrey indexes of chars modulo f | N; else NULL
    9705             :  * vchi[n] = a list of CHIvec [G0,chi0,o,ncharvecexpo(G0,nchi0),...]:
    9706             :  *   chi0 = primitive char attached to Conrey Mod(n,N)
    9707             :  * (resp. NULL if (n,N) > 1) */
    9708             : static GEN
    9709         651 : charsmodN(long N)
    9710             : {
    9711         651 :   GEN D, G, prio, phio, dummy = cgetg(1,t_VEC);
    9712         651 :   GEN vP, vG = const_vec(N,NULL), vCHI  = const_vec(N,NULL);
    9713         651 :   GEN bymod = const_vec(N,NULL);
    9714         651 :   long pn, i, l, vt = fetch_user_var("t");
    9715         651 :   D = mydivisorsu(N); l = lg(D);
    9716        3941 :   for (i = 1; i < l; i++)
    9717        3290 :     gel(bymod, D[i]) = vecsmalltrunc_init(myeulerphiu(D[i])+1);
    9718         651 :   gel(vG,N) = G = znstar0(utoipos(N),1);
    9719         651 :   pn = znstar_get_expo(G);  /* exponent(Z/NZ)^* */
    9720         651 :   vP = const_vec(pn,NULL);
    9721       27069 :   for (i = 1; i <= N; i++)
    9722             :   {
    9723             :     GEN P, gF, G0, chi0, nchi0, chi, v, go;
    9724             :     long j, F, o;
    9725       26418 :     if (ugcd(i,N) != 1) continue;
    9726       14147 :     chi = znconreylog(G, utoipos(i));
    9727       14147 :     gF = znconreyconductor(G, chi, &chi0);
    9728       14147 :     F = (typ(gF) == t_INT)? itou(gF): itou(gel(gF,1));
    9729       14147 :     G0 = gel(vG, F); if (!G0) G0 = gel(vG,F) = znstar0(gF, 1);
    9730       14147 :     nchi0 = znconreylog_normalize(G0,chi0);
    9731       14147 :     go = gel(nchi0,1); o = itou(go); /* order(chi0) */
    9732       14147 :     v = ncharvecexpo(G0, nchi0);
    9733       14147 :     if (!equaliu(go, pn)) v = zv_z_mul(v, pn / o);
    9734       14147 :     P = gel(vP, o); if (!P) P = gel(vP,o) = polcyclo(o,vt);
    9735             :     /* mfcharcxinit with dummy complex powers */
    9736       14147 :     gel(vCHI,i) = mkvecn(6, G0, chi0, go, v, dummy, P);
    9737       14147 :     D = mydivisorsu(N / F); l = lg(D);
    9738       40565 :     for (j = 1; j < l; j++) vecsmalltrunc_append(gel(bymod, F*D[j]), i);
    9739             :   }
    9740         651 :   phio = zero_zv(pn); l = lg(vCHI); prio = cgetg(l, t_VEC);
    9741       27069 :   for (i = 1; i < l; i++)
    9742             :   {
    9743       26418 :     GEN CHI = gel(vCHI,i);
    9744             :     long o;
    9745       26418 :     if (!CHI) continue;
    9746       14147 :     o = CHIvec_ord(CHI);
    9747       14147 :     if (!phio[o]) phio[o] = myeulerphiu(o);
    9748       14147 :     prio[i] = phio[o];
    9749             :   }
    9750         651 :   l = lg(bymod);
    9751             :   /* sort characters by increasing value of phi(order) */
    9752       27069 :   for (i = 1; i < l; i++)
    9753             :   {
    9754       26418 :     GEN z = gel(bymod,i);
    9755       26418 :     if (z) gen_sort_inplace(z, (void*)prio, &cmp_small_priority, NULL);
    9756             :   }
    9757         651 :   return mkvec3(vCHI, bymod, vG);
    9758             : }
    9759             : 
    9760             : static GEN
    9761        5586 : mfeisenstein2pure(long k, GEN CHI1, GEN CHI2, long ord, GEN P, long lim)
    9762             : {
    9763        5586 :   GEN c, V = cgetg(lim+2, t_COL);
    9764             :   long n;
    9765        5586 :   c = mfeisenstein2_0(k, CHI1, CHI2, ord);
    9766        5586 :   if (P) c = grem(c, P);
    9767        5586 :   gel(V,1) = c;
    9768      113400 :   for (n=1; n <= lim; n++)
    9769             :   {
    9770      107814 :     c = sigchi2(k, CHI1, CHI2, n, ord);
    9771      107814 :     if (P) c = grem(c, P);
    9772      107814 :     gel(V,n+1) = c;
    9773             :   }
    9774        5586 :   return V;
    9775             : }
    9776             : static GEN
    9777        5054 : mfeisenstein2pure_Fl(long k, GEN CHI1vec, GEN CHI2vec, GEN vz, ulong p, long lim)
    9778             : {
    9779        5054 :   GEN V = cgetg(lim+2, t_VECSMALL);
    9780             :   long n;
    9781        5054 :   V[1] = mfeisenstein2_0_Fl(k, CHI1vec, CHI2vec, vz, p);
    9782       82845 :   for (n=1; n <= lim; n++) V[n+1] = sigchi2_Fl(k, CHI1vec, CHI2vec, n, vz, p);
    9783        5054 :   return V;
    9784             : }
    9785             : 
    9786             : static GEN
    9787         252 : getcolswt2(GEN M, GEN D, ulong p)
    9788             : {
    9789         252 :   GEN R, v = gel(M,1);
    9790         252 :   long i, l = lg(M) - 1;
    9791         252 :   R = cgetg(l, t_MAT); /* skip D[1] = 1 */
    9792        1008 :   for (i = 1; i < l; i++)
    9793             :   {
    9794         756 :     GEN w = Flv_Fl_mul(gel(M,i+1), D[i+1], p);
    9795         756 :     gel(R,i) = Flv_sub(v, w, p);
    9796             :   }
    9797         252 :   return R;
    9798             : }
    9799             : static GEN
    9800        5852 : expandbd(GEN V, long d)
    9801             : {
    9802             :   long L, n, nd;
    9803             :   GEN W;
    9804        5852 :   if (d == 1) return V;
    9805        2121 :   L = lg(V)-1; W = zerocol(L); /* nd = n/d */
    9806       18263 :   for (n = nd = 0; n < L; n += d, nd++) gel(W, n+1) = gel(V, nd+1);
    9807        2121 :   return W;
    9808             : }
    9809             : static GEN
    9810        7714 : expandbd_Fl(GEN V, long d)
    9811             : {
    9812             :   long L, n, nd;
    9813             :   GEN W;
    9814        7714 :   if (d == 1) return V;
    9815        2660 :   L = lg(V)-1; W = zero_Flv(L); /* nd = n/d */
    9816       16429 :   for (n = nd = 0; n < L; n += d, nd++) W[n+1] = V[nd+1];
    9817        2660 :   return W;
    9818             : }
    9819             : static void
    9820        5054 : getcols_i(GEN *pM, GEN *pvj, GEN gk, GEN CHI1vec, GEN CHI2vec, long NN1, GEN vz,
    9821             :           ulong p, long lim)
    9822             : {
    9823        5054 :   GEN CHI1 = CHIvec_CHI(CHI1vec), CHI2 = CHIvec_CHI(CHI2vec);
    9824        5054 :   long N2 = CHIvec_N(CHI2vec);
    9825        5054 :   GEN vj, M, D = mydivisorsu(NN1/N2);
    9826        5054 :   long i, l = lg(D), k = gk[2];
    9827        5054 :   GEN V = mfeisenstein2pure_Fl(k, CHI1vec, CHI2vec, vz, p, lim);
    9828        5054 :   M = cgetg(l, t_MAT);
    9829       12768 :   for (i = 1; i < l; i++) gel(M,i) = expandbd_Fl(V, D[i]);
    9830        5054 :   if (k == 2 && N2 == 1 && CHIvec_N(CHI1vec) == 1)
    9831             :   {
    9832         252 :     M = getcolswt2(M, D, p); l--;
    9833         252 :     D = vecslice(D, 2, l);
    9834             :   }
    9835        5054 :   *pM = M;
    9836        5054 :   *pvj = vj = cgetg(l, t_VEC);
    9837       12516 :   for (i = 1; i < l; i++) gel(vj,i) = mkvec4(gk, CHI1, CHI2, utoipos(D[i]));
    9838        5054 : }
    9839             : 
    9840             : /* find all CHI1, CHI2 mod N such that CHI1*CHI2 = CHI, f(CHI1)*f(CHI2) | N.
    9841             :  * set M = mfcoefs(B_e E(CHI1,CHI2), lim), vj = [e,i1,i2] */
    9842             : static void
    9843        2037 : getcols(GEN *pM, GEN *pv, long k, long nCHI, GEN allN, GEN vz, ulong p,
    9844             :         long lim)
    9845             : {
    9846        2037 :   GEN vCHI = gel(allN,1), gk = utoi(k);
    9847        2037 :   GEN M = cgetg(1,t_MAT), v = cgetg(1,t_VEC);
    9848        2037 :   long i1, N = lg(vCHI)-1;
    9849       93527 :   for (i1 = 1; i1 <= N; i1++)
    9850             :   {
    9851       91490 :     GEN CHI1vec = gel(vCHI, i1), CHI2vec, M1, v1;
    9852             :     long NN1, i2;
    9853      160972 :     if (!CHI1vec) continue;
    9854       73150 :     if (k == 1 && CHIvec_parity(CHI1vec) == -1) continue;
    9855       48391 :     NN1 = N/CHIvec_N(CHI1vec); /* N/f(chi1) */;
    9856       48391 :     i2 = Fl_div(nCHI,i1, N);
    9857       48391 :     if (!i2) i2 = 1;
    9858       48391 :     CHI2vec = gel(vCHI,i2);
    9859       48391 :     if (NN1 % CHIvec_N(CHI2vec)) continue; /* f(chi1)f(chi2) | N ? */
    9860        3668 :     getcols_i(&M1, &v1, gk, CHI1vec, CHI2vec, NN1, vz, p, lim);
    9861        3668 :     M = shallowconcat(M, M1);
    9862        3668 :     v = shallowconcat(v, v1);
    9863             :   }
    9864        2037 :   *pM = M;
    9865        2037 :   *pv = v;
    9866        2037 : }
    9867             : 
    9868             : static void
    9869        1239 : update_Mj(GEN *M, GEN *vecj, GEN *pz, ulong p)
    9870             : {
    9871             :   GEN perm;
    9872        1239 :   *pz = Flm_indexrank(*M, p); perm = gel(*pz,2);
    9873        1239 :   *M = vecpermute(*M, perm);
    9874        1239 :   *vecj = vecpermute(*vecj, perm);
    9875        1239 : }
    9876             : static int
    9877         441 : getcolsgen(long dim, GEN *pM, GEN *pvj, GEN *pz, long k, long ell, long nCHI,
    9878             :            GEN allN, GEN vz, ulong p, long lim)
    9879             : {
    9880         441 :   GEN vCHI = gel(allN,1), bymod = gel(allN,2), gell = utoi(ell);
    9881         441 :   long i1, N = lg(vCHI)-1;
    9882         441 :   long L = lim+1;
    9883         441 :   if (lg(*pvj)-1 >= dim) update_Mj(pM, pvj, pz, p);
    9884         441 :   if (lg(*pvj)-1 == dim) return 1;
    9885        1806 :   for (i1 = 1; i1 <= N; i1++)
    9886             :   {
    9887        1778 :     GEN CHI1vec = gel(vCHI, i1), T;
    9888             :     long par1, j, l, N1, NN1;
    9889             : 
    9890        1778 :     if (!CHI1vec) continue;
    9891        1750 :     par1 = CHIvec_parity(CHI1vec);
    9892        1750 :     if (ell == 1 && par1 == -1) continue;
    9893        1169 :     if (odd(ell)) par1 = -par1;
    9894        1169 :     N1 = CHIvec_N(CHI1vec);
    9895        1169 :     NN1 = N/N1;
    9896        1169 :     T = gel(bymod, NN1); l = lg(T);
    9897        4277 :     for (j = 1; j < l; j++)
    9898             :     {
    9899        3486 :       long i2 = T[j], l1, l2, j1, s, nC;
    9900        3486 :       GEN M, M1, M2, vj, vj1, vj2, CHI2vec = gel(vCHI, i2);
    9901        3486 :       if (CHIvec_parity(CHI2vec) != par1) continue;
    9902        1386 :       nC = Fl_div(nCHI, Fl_mul(i1,i2,N), N);
    9903        1386 :       getcols(&M2, &vj2, k-ell, nC, allN, vz, p, lim);
    9904        1386 :       l2 = lg(M2); if (l2 == 1) continue;
    9905        1386 :       getcols_i(&M1, &vj1, gell, CHI1vec, CHI2vec, NN1, vz, p, lim);
    9906        1386 :       l1 = lg(M1);
    9907        1386 :       M1 = Flm_to_FlxV(M1, 0);
    9908        1386 :       M2 = Flm_to_FlxV(M2, 0);
    9909        1386 :       M  = cgetg((l1-1)*(l2-1) + 1, t_MAT);
    9910        1386 :       vj = cgetg((l1-1)*(l2-1) + 1, t_VEC);
    9911        3318 :       for (j1 = s = 1; j1 < l1; j1++)
    9912             :       {
    9913        1932 :         GEN E = gel(M1,j1), v = gel(vj1,j1);
    9914             :         long j2;
    9915        7805 :         for (j2 = 1; j2 < l2; j2++, s++)
    9916             :         {
    9917        5873 :           GEN c = Flx_to_Flv(Flxn_mul(E, gel(M2,j2), L, p), L);
    9918        5873 :           gel(M,s) = c;
    9919        5873 :           gel(vj,s) = mkvec2(v, gel(vj2,j2));
    9920             :         }
    9921             :       }
    9922        1386 :       *pM = shallowconcat(*pM, M);
    9923        1386 :       *pvj = shallowconcat(*pvj, vj);
    9924        1386 :       if (lg(*pvj)-1 >= dim) update_Mj(pM, pvj, pz, p);
    9925        1386 :       if (lg(*pvj)-1 == dim) return 1;
    9926             :     }
    9927             :   }
    9928          28 :   if (ell == 1)
    9929             :   {
    9930          21 :     update_Mj(pM, pvj, pz, p);
    9931          21 :     return (lg(*pvj)-1 == dim);
    9932             :   }
    9933           7 :   return 0;
    9934             : }
    9935             : 
    9936             : static GEN
    9937        1645 : mkF2bd(long d, long lim)
    9938             : {
    9939        1645 :   GEN V = zerovec(lim + 1);
    9940             :   long n;
    9941        1645 :   gel(V, 1) = sstoQ(-1, 24);
    9942       24248 :   for (n = 1; n <= lim/d; n++) gel(V, n*d + 1) = mysumdivku(n, 1);
    9943        1645 :   return V;
    9944             : }
    9945             : 
    9946             : static GEN
    9947        6202 : mkeisen(GEN E, long ord, GEN P, long lim)
    9948             : {
    9949        6202 :   long k = itou(gel(E,1)), e = itou(gel(E,4));
    9950        6202 :   GEN CHI1 = gel(E,2), CHI2 = gel(E,3);
    9951        6202 :   if (k == 2 && mfcharistrivial(CHI1) && mfcharistrivial(CHI2))
    9952         616 :     return gsub(mkF2bd(1,lim), gmulgu(mkF2bd(e,lim), e));
    9953             :   else
    9954             :   {
    9955        5586 :     GEN V = mfeisenstein2pure(k, CHI1, CHI2, ord, P, lim);
    9956        5586 :     return expandbd(V, e);
    9957             :   }
    9958             : }
    9959             : static GEN
    9960         609 : mkM(GEN vj, long pn, GEN P, long lim)
    9961             : {
    9962         609 :   long j, l = lg(vj), L = lim+1;
    9963         609 :   GEN M = cgetg(l, t_MAT);
    9964        5061 :   for (j = 1; j < l; j++)
    9965             :   {
    9966             :     GEN E1, E2;
    9967        4452 :     parse_vecj(gel(vj,j), &E1,&E2);
    9968        4452 :     E1 = RgV_to_RgX(mkeisen(E1, pn, P, lim), 0);
    9969        4452 :     if (E2)
    9970             :     {
    9971        1750 :       E2 = RgV_to_RgX(mkeisen(E2, pn, P, lim), 0);
    9972        1750 :       E1 = RgXn_mul(E1, E2, L);
    9973             :     }
    9974        4452 :     E1 = RgX_to_RgC(E1, L);
    9975        4452 :     if (P && E2) E1 = RgXQV_red(E1, P);
    9976        4452 :     gel(M,j) = E1;
    9977             :   }
    9978         609 :   return M;
    9979             : }
    9980             : 
    9981             : /* assume N > 2 */
    9982             : static GEN
    9983          35 : mffindeisen1(long N)
    9984             : {
    9985          35 :   GEN G = znstar0(utoipos(N), 1), L = chargalois(G, NULL), chi0 = NULL;
    9986          35 :   long j, m = N, l = lg(L);
    9987         259 :   for (j = 1; j < l; j++)
    9988             :   {
    9989         245 :     GEN chi = gel(L,j);
    9990         245 :     long r = myeulerphiu(itou(zncharorder(G,chi)));
    9991         245 :     if (r >= m) continue;
    9992         182 :     chi = znconreyfromchar(G, chi);
    9993         182 :     if (zncharisodd(G,chi)) { m = r; chi0 = chi; if (r == 1) break; }
    9994             :   }
    9995          35 :   if (!chi0) pari_err_BUG("mffindeisen1 [no Eisenstein series found]");
    9996          35 :   chi0 = znchartoprimitive(G,chi0);
    9997          35 :   return mfcharGL(gel(chi0,1), gel(chi0,2));
    9998             : }
    9999             : 
   10000             : static GEN
   10001         651 : mfeisensteinspaceinit_i(long N, long k, GEN CHI)
   10002             : {
   10003         651 :   GEN M, Minv, vj, vG, GN, allN, P, vz, z = NULL;
   10004         651 :   long nCHI, lim, ell, ord, dim = mffulldim(N, k, CHI);
   10005             :   ulong r, p;
   10006             : 
   10007         651 :   if (!dim) retmkvec3(cgetg(1,t_VECSMALL),
   10008             :                       mkvec2(cgetg(1,t_MAT),gen_1),cgetg(1,t_VEC));
   10009         651 :   lim = mfsturmNk(N, k) + 1;
   10010         651 :   allN = charsmodN(N);
   10011         651 :   vG = gel(allN,3);
   10012         651 :   GN = gel(vG,N);
   10013         651 :   ord = znstar_get_expo(GN);
   10014         651 :   P = ord <= 2? NULL: polcyclo(ord, varn(mfcharpol(CHI)));
   10015         651 :   CHI = induce(GN, CHI); /* lift CHI mod N before mfcharno*/
   10016         651 :   nCHI = mfcharno(CHI);
   10017         651 :   r = QabM_init(ord, &p);
   10018         651 :   vz = Fl_powers(r, ord, p);
   10019         651 :   getcols(&M, &vj, k, nCHI, allN, vz, p, lim);
   10020         679 :   for (ell = k>>1; ell >= 1; ell--)
   10021         441 :     if (getcolsgen(dim, &M, &vj, &z, k, ell, nCHI, allN, vz, p, lim)) break;
   10022         651 :   if (!z) update_Mj(&M, &vj, &z, p);
   10023         651 :   if (lg(vj) - 1 < dim) return NULL;
   10024         609 :   M = mkM(vj, ord, P, lim);
   10025         609 :   Minv = QabM_Minv(rowpermute(M, gel(z,1)), P, ord);
   10026         609 :   return mkvec4(gel(z,1), Minv, vj, utoi(ord));
   10027             : }
   10028             : /* true mf */
   10029             : static GEN
   10030         609 : mfeisensteinspaceinit(GEN mf)
   10031             : {
   10032         609 :   pari_sp av = avma;
   10033         609 :   GEN z, CHI = MF_get_CHI(mf);
   10034         609 :   long N = MF_get_N(mf), k = MF_get_k(mf);
   10035         609 :   if (!CHI) CHI = mfchartrivial();
   10036         609 :   z = mfeisensteinspaceinit_i(N, k, CHI);
   10037         609 :   if (!z)
   10038             :   {
   10039          35 :     GEN E, CHIN = mffindeisen1(N), CHI0 = mfchartrivial();
   10040          35 :     z = mfeisensteinspaceinit_i(N, k+1, mfcharmul(CHI, CHIN));
   10041          35 :     if (z) E = mkvec4(gen_1, CHI0, CHIN, gen_1);
   10042             :     else
   10043             :     {
   10044           7 :       z = mfeisensteinspaceinit_i(N, k+2, CHI);
   10045           7 :       E = mkvec4(gen_2, CHI0, CHI0, utoipos(N));
   10046             :     }
   10047          35 :     z = mkvec2(z, E);
   10048             :   }
   10049         609 :   return gc_GEN(av, z);
   10050             : }
   10051             : 
   10052             : /* decomposition of modular form on eisenspace */
   10053             : static GEN
   10054        1218 : mfeisensteindec(GEN mf, GEN F)
   10055             : {
   10056        1218 :   pari_sp av = avma;
   10057             :   GEN M, Mindex, Mvecj, V, B, CHI;
   10058             :   long o, ord;
   10059             : 
   10060        1218 :   Mvecj = obj_checkbuild(mf, MF_EISENSPACE, &mfeisensteinspaceinit);
   10061        1218 :   if (lg(Mvecj) < 5)
   10062             :   {
   10063          56 :     GEN E, e = gel(Mvecj,2), gkE = gel(e,1);
   10064          56 :     long dE = itou(gel(e,4));
   10065          56 :     Mvecj = gel(Mvecj,1);
   10066          56 :     E = mfeisenstein(itou(gkE), NULL, gel(e,3));
   10067          56 :     if (dE != 1) E = mfbd_E2(E, dE, gel(e,2)); /* here k = 2 */
   10068          56 :     F = mfmul(F, E);
   10069             :   }
   10070        1218 :   M = gel(Mvecj, 2);
   10071        1218 :   if (lg(M) == 1) return cgetg(1, t_VEC);
   10072        1218 :   Mindex = gel(Mvecj, 1);
   10073        1218 :   ord = itou(gel(Mvecj,4));
   10074        1218 :   V = mfcoefs(F, Mindex[lg(Mindex)-1]-1, 1); settyp(V, t_COL);
   10075        1218 :   CHI = mf_get_CHI(F);
   10076        1218 :   o = mfcharorder(CHI);
   10077        1218 :   if (o > 2 && o != ord)
   10078             :   { /* convert Mod(.,polcyclo(o)) to Mod(., polcyclo(N)) for o | N,
   10079             :      * o and N both != 2 (mod 4) */
   10080          84 :     GEN z, P = gel(M,4); /* polcyclo(ord) */
   10081          84 :     long vt = varn(P);
   10082          84 :     z = gmodulo(pol_xn(ord/o, vt), P);
   10083          84 :     if (ord % o) pari_err_TYPE("mfeisensteindec", V);
   10084          84 :     V = gsubst(liftpol_shallow(V), vt, z);
   10085             :   }
   10086        1218 :   B = Minv_RgC_mul(M, vecpermute(V, Mindex));
   10087        1218 :   return gc_upto(av, B);
   10088             : }
   10089             : 
   10090             : /*********************************************************************/
   10091             : /*                        END EISENSPACE                             */
   10092             : /*********************************************************************/
   10093             : 
   10094             : static GEN
   10095          70 : sertocol2(GEN S, long l)
   10096             : {
   10097          70 :   GEN C = cgetg(l + 2, t_COL);
   10098             :   long i;
   10099         420 :   for (i = 0; i <= l; i++) gel(C, i+1) = polcoef_i(S, i, -1);
   10100          70 :   return C;
   10101             : }
   10102             : 
   10103             : /* Compute polynomial P0 such that F=E4^(k/4)P0(E6/E4^(3/2)). */
   10104             : static GEN
   10105          14 : mfcanfindp0(GEN F, long k)
   10106             : {
   10107          14 :   pari_sp ltop = avma;
   10108             :   GEN E4, E6, V, V1, Q, W, res, M, B;
   10109             :   long l, j;
   10110          14 :   l = k/6 + 2;
   10111          14 :   V = mfcoefsser(F,l);
   10112          14 :   E4 = mfcoefsser(mfEk(4),l);
   10113          14 :   E6 = mfcoefsser(mfEk(6),l);
   10114          14 :   V1 = gdiv(V, gpow(E4, uutoQ(k,4), 0));
   10115          14 :   Q = gdiv(E6, gpow(E4, uutoQ(3,2), 0));
   10116          14 :   W = gpowers(Q, l - 1);
   10117          14 :   M = cgetg(l + 1, t_MAT);
   10118          70 :   for (j = 1; j <= l; j++) gel(M,j) = sertocol2(gel(W,j), l);
   10119          14 :   B = sertocol2(V1, l);
   10120          14 :   res = inverseimage(M, B);
   10121          14 :   if (lg(res) == 1) err_space(F);
   10122          14 :   return gc_GEN(ltop, gtopolyrev(res, 0));
   10123             : }
   10124             : 
   10125             : /* Compute the first n+1 Taylor coeffs at tau=I of a modular form
   10126             :  * on SL_2(Z). */
   10127             : GEN
   10128          14 : mftaylor(GEN F, long n, long flreal, long prec)
   10129             : {
   10130          14 :   pari_sp ltop = avma;
   10131          14 :   GEN P0, Pm1 = gen_0, v;
   10132          14 :   GEN X2 = mkpoln(3, ghalf,gen_0,gneg(ghalf)); /* (x^2-1) / 2 */
   10133             :   long k, m;
   10134          14 :   if (!checkmf_i(F)) pari_err_TYPE("mftaylor",F);
   10135          14 :   k = mf_get_k(F);
   10136          14 :   if (mf_get_N(F) != 1 || k < 0) pari_err_IMPL("mftaylor for this form");
   10137          14 :   P0 = mfcanfindp0(F, k);
   10138          14 :   v = cgetg(n+2, t_VEC); gel(v, 1) = RgX_coeff(P0,0);
   10139         154 :   for (m = 0; m < n; m++)
   10140             :   {
   10141         140 :     GEN P1 = gdivgu(gmulsg(-(k + 2*m), RgX_shift(P0,1)), 12);
   10142         140 :     P1 = gadd(P1, gmul(X2, RgX_deriv(P0)));
   10143         140 :     if (m) P1 = gsub(P1, gdivgu(gmulsg(m*(m+k-1), Pm1), 144));
   10144         140 :     Pm1 = P0; P0 = P1;
   10145         140 :     gel(v, m+2) = RgX_coeff(P0, 0);
   10146             :   }
   10147          14 :   if (flreal)
   10148             :   {
   10149           7 :     GEN pi2 = Pi2n(1, prec), pim4 = gmulsg(-2, pi2), VPC;
   10150           7 :     GEN C = gmulsg(3, gdiv(gpowgs(ggamma(uutoQ(1,4), prec), 8), gpowgs(pi2, 6)));
   10151             :     /* E_4(i): */
   10152           7 :     GEN facn = gen_1;
   10153           7 :     VPC = gpowers(gmul(pim4, gsqrt(C, prec)), n);
   10154           7 :     C = gpow(C, uutoQ(k,4), prec);
   10155          84 :     for (m = 0; m <= n; m++)
   10156             :     {
   10157          77 :       gel(v, m+1) = gdiv(gmul(C, gmul(gel(v, m+1), gel(VPC, m+1))), facn);
   10158          77 :       facn = gmulgu(facn, m+1);
   10159             :     }
   10160             :   }
   10161          14 :   return gc_GEN(ltop, v);
   10162             : }
   10163             : 
   10164             : #if 0
   10165             : /* To be used in mfeigensearch() */
   10166             : GEN
   10167             : mfreadratfile()
   10168             : {
   10169             :   GEN eqn;
   10170             :   pariFILE *F = pari_fopengz("rateigen300.gp");
   10171             :   eqn = gp_readvec_stream(F->file);
   10172             :   pari_fclose(F);
   10173             :   return eqn;
   10174             : }
   10175             : #endif
   10176             :  /*****************************************************************/
   10177             : /*           EISENSTEIN CUSPS: COMPLEX DIRECTLY: one F_k         */
   10178             : /*****************************************************************/
   10179             : 
   10180             : /* CHIvec = charinit(CHI); data = [N1g/g1,N2g/g2,g1/g,g2/g,C/g1,C/g2,
   10181             :  * (N1g/g1)^{-1},(N2g/g2)^{-1}] */
   10182             : 
   10183             : /* nm = n/m;
   10184             :  * z1 = powers of \z_{C/g}^{(Ae/g)^{-1}},
   10185             :  * z2 = powers of \z_N^{A^{-1}(g1g2/C)}]
   10186             :  * N.B. : we compute value and conjugate at the end, so it is (Ae/g)^{-1}
   10187             :  * and not -(Ae/g)^{-1} */
   10188             : static GEN
   10189     9635178 : eiscnm(long nm, long m, GEN CHI1vec, GEN CHI2vec, GEN data, GEN z1)
   10190             : {
   10191     9635178 :   long Cg1 = data[5], s10 = (nm*data[7]) % Cg1, r10 = (nm - data[1]*s10) / Cg1;
   10192     9635178 :   long Cg2 = data[6], s20 = (m *data[8]) % Cg2, r20 = (m  - data[2]*s20) / Cg2;
   10193             :   long j1, r1, s1;
   10194     9635178 :   GEN T = gen_0;
   10195    22465660 :   for (j1 = 0, r1 = r10, s1 = s10; j1 < data[3]; j1++, r1 -= data[1], s1 += Cg1)
   10196             :   {
   10197    12830482 :     GEN c1 = mychareval(CHI1vec, r1);
   10198    12830482 :     if (!gequal0(c1))
   10199             :     {
   10200             :       long j2, r2, s2;
   10201     9925790 :       GEN S = gen_0;
   10202    24733030 :       for (j2 = 0, r2 = r20, s2 = s20; j2 < data[4]; j2++, r2 -= data[2], s2 += Cg2)
   10203             :       {
   10204    14807240 :         GEN c2 = mychareval(CHI2vec, r2);
   10205    14807240 :         if (!gequal0(c2)) S = gadd(S, gmul(c2, rootsof1pow(z1, s1*s2)));
   10206             :       }
   10207     9925790 :       T = gadd(T, gmul(c1, S));
   10208             :     }
   10209             :   }
   10210     9635178 :   return conj_i(T);
   10211             : }
   10212             : 
   10213             : static GEN
   10214      855267 : fg1g2n(long n, long k, GEN CHI1vec, GEN CHI2vec, GEN data, GEN z1, GEN z2)
   10215             : {
   10216      855267 :   pari_sp av = avma;
   10217      855267 :   GEN S = gen_0, D = mydivisorsu(n);
   10218      855267 :   long i, l = lg(D);
   10219     5672856 :   for (i = 1; i < l; i++)
   10220             :   {
   10221     4817589 :     long m = D[i], nm = D[l-i]; /* n/m */
   10222     4817589 :     GEN u = eiscnm( nm,  m, CHI1vec, CHI2vec, data, z1);
   10223     4817589 :     GEN v = eiscnm(-nm, -m, CHI1vec, CHI2vec, data, z1);
   10224     4817589 :     GEN w = odd(k) ? gsub(u, v) : gadd(u, v);
   10225     4817589 :     S = gadd(S, gmul(powuu(m, k-1), w));
   10226             :   }
   10227      855267 :   return gc_upto(av, gmul(S, rootsof1pow(z2, n)));
   10228             : }
   10229             : 
   10230             : static GEN
   10231       33985 : gausssumcx(GEN CHIvec, long prec)
   10232             : {
   10233             :   GEN z, S, V;
   10234       33985 :   long m, N = CHIvec_N(CHIvec);
   10235       33985 :   if (N == 1) return gen_1;
   10236       18277 :   V = CHIvec_val(CHIvec);
   10237       18277 :   z = rootsof1u_cx(N, prec);
   10238       18277 :   S = gmul(z, gel(V, N));
   10239      431158 :   for (m = N-1; m >= 1; m--) S = gmul(z, gadd(gel(V, m), S));
   10240       18277 :   return S;
   10241             : }
   10242             : 
   10243             : /* Computation of Q_k(\z_N^s) as a polynomial in \z_N^s. FIXME: explicit
   10244             :  * formula ? */
   10245             : static GEN
   10246        6118 : mfqk(long k, long N)
   10247             : {
   10248             :   GEN X, P, ZI, Q, Xm1, invden;
   10249             :   long i;
   10250        6118 :   ZI = gdivgu(RgX_shift_shallow(RgV_to_RgX(identity_ZV(N-1), 0), 1), N);
   10251        6118 :   if (k == 1) return ZI;
   10252        4956 :   P = gsubgs(pol_xn(N,0), 1);
   10253        4956 :   invden = RgXQ_powu(ZI, k, P);
   10254        4956 :   X = pol_x(0); Q = gneg(X); Xm1 = gsubgs(X, 1);
   10255       21784 :   for (i = 2; i < k; i++)
   10256       16828 :     Q = RgX_shift_shallow(ZX_add(gmul(Xm1, ZX_deriv(Q)), gmulsg(-i, Q)), 1);
   10257        4956 :   return RgXQ_mul(Q, invden, P);
   10258             : }
   10259             : 
   10260             : /* CHI mfchar; M is a multiple of the conductor of CHI, but is NOT
   10261             :  * necessarily its modulus */
   10262             : static GEN
   10263        7903 : mfskcx(long k, GEN CHI, long M, long prec)
   10264             : {
   10265             :   GEN S, CHIvec, P;
   10266             :   long F, m, i, l;
   10267        7903 :   CHI = mfchartoprimitive(CHI, &F);
   10268        7903 :   CHIvec = mfcharcxinit(CHI, prec);
   10269        7903 :   if (F == 1) S = gdivgu(bernfrac(k), k);
   10270             :   else
   10271             :   {
   10272        6118 :     GEN Q = mfqk(k, F), V = CHIvec_val(CHIvec);
   10273        6118 :     S = gmul(gel(V, F), RgX_coeff(Q, 0));
   10274      156611 :     for (m = 1; m < F; m++) S = gadd(S, gmul(gel(V, m), RgX_coeff(Q, m)));
   10275        6118 :     S = conj_i(S);
   10276             :   }
   10277             :   /* prime divisors of M not dividing f(chi) */
   10278        7903 :   P = gel(myfactoru(u_ppo(M/F,F)), 1); l = lg(P);
   10279        8057 :   for (i = 1; i < l; i++)
   10280             :   {
   10281         154 :     long p = P[i];
   10282         154 :     S = gmul(S, gsubsg(1, gdiv(mychareval(CHIvec, p), powuu(p, k))));
   10283             :   }
   10284        7903 :   return gmul(gmul(gausssumcx(CHIvec, prec), S), powuu(M/F, k));
   10285             : }
   10286             : 
   10287             : static GEN
   10288       13727 : f00_i(long k, GEN CHI1vec, GEN CHI2vec, GEN G2, GEN S, long prec)
   10289             : {
   10290             :   GEN c, a;
   10291       13727 :   long N1 = CHIvec_N(CHI1vec), N2 = CHIvec_N(CHI2vec);
   10292       13727 :   if (S[2] != N1) return gen_0;
   10293        7903 :   c = mychareval(CHI1vec, S[3]);
   10294        7903 :   if (isintzero(c)) return gen_0;
   10295        7903 :   a = mfskcx(k, mfchardiv(CHIvec_CHI(CHI2vec), CHIvec_CHI(CHI1vec)), N1*N2, prec);
   10296        7903 :   a = gmul(a, conj_i(gmul(c,G2)));
   10297        7903 :   return gdiv(a, mulsi(-N2, powuu(S[1], k-1)));
   10298             : }
   10299             : 
   10300             : static GEN
   10301       12446 : f00(long k, GEN CHI1vec,GEN CHI2vec, GEN G1,GEN G2, GEN data, long prec)
   10302             : {
   10303             :   GEN T1, T2;
   10304       12446 :   T2 = f00_i(k, CHI1vec, CHI2vec, G2, data, prec);
   10305       12446 :   if (k > 1) return T2;
   10306        1281 :   T1 = f00_i(k, CHI2vec, CHI1vec, G1, data, prec);
   10307        1281 :   return gadd(T1, T2);
   10308             : }
   10309             : 
   10310             : /* ga in SL_2(Z), find beta [a,b;c,d] in Gamma_0(N) and mu in Z such that
   10311             :  * beta * ga * T^u = [A',B';C',D'] with C' | N and N | B', C' > 0 */
   10312             : static void
   10313       13041 : mfgatogap(GEN ga, long N, long *pA, long *pC, long *pD, long *pd, long *pmu)
   10314             : {
   10315       13041 :   GEN A = gcoeff(ga,1,1), B = gcoeff(ga,1,2);
   10316       13041 :   GEN C = gcoeff(ga,2,1), D = gcoeff(ga,2,2), a, b, c, d;
   10317             :   long t, Ap, Cp, B1, D1, mu;
   10318       13041 :   Cp = itou(bezout(muliu(A,N), C, &c, &d)); /* divides N */
   10319       13041 :   t = 0;
   10320       13041 :   if (Cp > 1)
   10321             :   { /* (d, N/Cp) = 1, find t such that (d - t*(A*N/Cp), N) = 1 */
   10322        2604 :     long dN = umodiu(d,Cp), Q = (N/Cp * umodiu(A,Cp)) % Cp;
   10323        2989 :     while (ugcd(dN, Cp) > 1) { t++; dN = Fl_sub(dN, Q, Cp); }
   10324             :   }
   10325       13041 :   if (t)
   10326             :   {
   10327         385 :     c = addii(c, mului(t, diviuexact(C,Cp)));
   10328         385 :     d = subii(d, mului(t, muliu(A, N/Cp))); /* (d,N) = 1 */
   10329             :   }
   10330       13041 :   D1 = umodiu(mulii(d,D), N);
   10331       13041 :   (void)bezout(d, mulis(c,-N), &a, &b); /* = 1 */
   10332       13041 :   t = 0; Ap = umodiu(addii(mulii(a,A), mulii(b,C)), N); /* (Ap,Cp) = 1 */
   10333       22267 :   while (ugcd(Ap, N) > 1) { t++; Ap = Fl_add(Ap, Cp, N); }
   10334       13041 :   B1 = umodiu(a,N)*umodiu(B,N) + umodiu(b,N)*umodiu(D,N) + t*D1;
   10335       13041 :   B1 %= N;
   10336       13041 :   *pmu = mu = Fl_neg(Fl_div(B1, Ap, N), N);
   10337             :   /* A', D' and d only needed modulo N */
   10338       13041 :   *pd = umodiu(d, N);
   10339       13041 :   *pA = Ap;
   10340       13041 :   *pC = Cp; *pD = (D1 + Cp*mu) % N;
   10341       13041 : }
   10342             : 
   10343             : #if 0
   10344             : /* CHI is a mfchar, return alpha(CHI) */
   10345             : static long
   10346             : mfalchi(GEN CHI, long AN, long cg)
   10347             : {
   10348             :   GEN G = gel(CHI,1), chi = gel(CHI,2), go = gmfcharorder(CHI);
   10349             :   long o = itou(go), a = itos( znchareval(G, chi, stoi(1 + AN/cg), go) );
   10350             :   if (a < 0 || (cg * a) % o) pari_err_BUG("mfalchi");
   10351             :   return (cg * a) / o;
   10352             : }
   10353             : #endif
   10354             : /* return A such that CHI1(t) * CHI2(t) = e(A) or NULL if (t,N1*N2) > 1 */
   10355             : static GEN
   10356       26082 : mfcharmuleval(GEN CHI1vec, GEN CHI2vec, long t)
   10357             : {
   10358       26082 :   long a1 = mycharexpo(CHI1vec, t), o1 = CHIvec_ord(CHI1vec);
   10359       26082 :   long a2 = mycharexpo(CHI2vec, t), o2 = CHIvec_ord(CHI2vec);;
   10360       26082 :   if (a1 < 0 || a2 < 0) return NULL;
   10361       26082 :   return sstoQ(a1*o2 + a2*o1, o1*o2);
   10362             : }
   10363             : static GEN
   10364       13041 : mfcharmulcxeval(GEN CHI1vec, GEN CHI2vec, long t, long prec)
   10365             : {
   10366       13041 :   GEN A = mfcharmuleval(CHI1vec, CHI2vec, t);
   10367             :   long n, d;
   10368       13041 :   if (!A) return gen_0;
   10369       13041 :   Qtoss(A, &n,&d); return rootsof1q_cx(n, d, prec);
   10370             : }
   10371             : /* alpha(CHI1 * CHI2) */
   10372             : static long
   10373       13041 : mfalchi2(GEN CHI1vec, GEN CHI2vec, long AN, long cg)
   10374             : {
   10375       13041 :   GEN A = mfcharmuleval(CHI1vec, CHI2vec, 1 + AN/cg);
   10376             :   long a;
   10377       13041 :   if (!A) pari_err_BUG("mfalchi2");
   10378       13041 :   A = gmulsg(cg, A);
   10379       13041 :   if (typ(A) != t_INT) pari_err_BUG("mfalchi2");
   10380       13041 :   a = itos(A) % cg; if (a < 0) a += cg;
   10381       13041 :   return a;
   10382             : }
   10383             : 
   10384             : /* return g = (a,b), set u >= 0 s.t. g = a * u (mod b) */
   10385             : static long
   10386       52164 : mybezout(long a, long b, long *pu)
   10387             : {
   10388       52164 :   long junk, g = cbezout(a, b, pu, &junk);
   10389       52164 :   if (*pu < 0) *pu += b/g;
   10390       52164 :   return g;
   10391             : }
   10392             : 
   10393             : /* E = [k, CHI1,CHI2, e], CHI1 and CHI2 primitive mfchars such that,
   10394             :  * CHI1(-1)*CHI2(-1) = (-1)^k; expansion of (B_e (E_k(CHI1,CHI2))) | ga.
   10395             :  * w is the width for the space of the calling function. */
   10396             : static GEN
   10397       13041 : mfeisensteingacx(GEN E, long w, GEN ga, long lim, long prec)
   10398             : {
   10399       13041 :   GEN CHI1vec, CHI2vec, CHI1 = gel(E,2), CHI2 = gel(E,3), v, S, ALPHA;
   10400             :   GEN G1, G2, z1, z2, data;
   10401       13041 :   long k = itou(gel(E,1)), e = itou(gel(E,4));
   10402       13041 :   long N1 = mfcharmodulus(CHI1);
   10403       13041 :   long N2 = mfcharmodulus(CHI2), N = e * N1 * N2;
   10404             :   long NsurC, cg, wN, A, C, Ai, d, mu, alchi, na, da;
   10405             :   long eg, g, gH, U, u0, u1, u2, Aig, H, m, n, t, Cg, NC1, NC2;
   10406             : 
   10407       13041 :   mfgatogap(ga, N, &A, &C, &Ai, &d, &mu);
   10408       13041 :   CHI1vec = mfcharcxinit(CHI1, prec);
   10409       13041 :   CHI2vec = mfcharcxinit(CHI2, prec);
   10410       13041 :   NsurC = N/C; cg  = ugcd(C, NsurC); wN = NsurC / cg;
   10411       13041 :   if (w%wN) pari_err_BUG("mfeisensteingacx [wN does not divide w]");
   10412       13041 :   alchi = mfalchi2(CHI1vec, CHI2vec, A*N, cg);
   10413       13041 :   ALPHA = sstoQ(alchi, NsurC);
   10414             : 
   10415       13041 :   g = mybezout(A*e, C, &u0); Cg = C/g; eg = e/g;
   10416       13041 :   NC1 = mybezout(N1, Cg, &u1);
   10417       13041 :   NC2 = mybezout(N2, Cg, &u2);
   10418       13041 :   H = (NC1*NC2*g)/Cg;
   10419       13041 :   Aig = (Ai*H)%N; if (Aig < 0) Aig += N;
   10420       13041 :   z1 = rootsof1powinit(u0, Cg, prec);
   10421       13041 :   z2 = rootsof1powinit(Aig, N, prec);
   10422       13041 :   data = mkvecsmalln(8, N1/NC1, N2/NC2, NC1, NC2, Cg/NC1, Cg/NC2, u1, u2);
   10423       13041 :   v = zerovec(lim + 1);
   10424             :   /* need n*H = alchi (mod cg) */
   10425       13041 :   gH = mybezout(H, cg, &U);
   10426       13041 :   if (gH > 1)
   10427             :   {
   10428         511 :     if (alchi % gH) return mkvec2(gen_0, v);
   10429         511 :     alchi /= gH; cg /= gH; H /= gH;
   10430             :   }
   10431       13041 :   G1 = gausssumcx(CHI1vec, prec);
   10432       13041 :   G2 = gausssumcx(CHI2vec, prec);
   10433       13041 :   if (!alchi)
   10434       12446 :     gel(v,1) = f00(k, CHI1vec,CHI2vec,G1,G2, mkvecsmall3(NC2,Cg,A*eg), prec);
   10435       13041 :   n = Fl_mul(alchi,U,cg); if (!n) n = cg;
   10436       13041 :   m = (n*H - alchi) / cg; /* positive, exact division */
   10437      868308 :   for (; m <= lim; n+=cg, m+=H)
   10438      855267 :     gel(v, m+1) = fg1g2n(n, k, CHI1vec, CHI2vec, data, z1,z2);
   10439       13041 :   t = (2*e)/g; if (odd(k)) t = -t;
   10440       13041 :   v = gdiv(v, gmul(conj_i(gmul(G1,G2)), mulsi(t, powuu(eg*N2/NC2, k-1))));
   10441       13041 :   if (k == 2 && N1 == 1 && N2 == 1) v = gsub(mkF2bd(wN,lim), gmulsg(e,v));
   10442             : 
   10443       13041 :   Qtoss(ALPHA, &na,&da);
   10444       13041 :   S = conj_i( mfcharmulcxeval(CHI1vec,CHI2vec,d,prec) ); /* CHI(1/d) */
   10445       13041 :   if (wN > 1)
   10446             :   {
   10447       11354 :     GEN z = rootsof1powinit(-mu, wN, prec);
   10448       11354 :     long i, l = lg(v);
   10449      823739 :     for (i = 1; i < l; i++) gel(v,i) = gmul(gel(v,i), rootsof1pow(z,i-1));
   10450             :   }
   10451       13041 :   v = RgV_Rg_mul(v, gmul(S, rootsof1q_cx(-mu*na, da, prec)));
   10452       13041 :   return mkvec2(ALPHA, bdexpand(v, w/wN));
   10453             : }
   10454             : 
   10455             : /*****************************************************************/
   10456             : /*                       END EISENSTEIN CUSPS                    */
   10457             : /*****************************************************************/
   10458             : 
   10459             : static GEN
   10460        1596 : mfchisimpl(GEN CHI)
   10461             : {
   10462             :   GEN G, chi;
   10463        1596 :   if (typ(CHI) == t_INT) return CHI;
   10464        1596 :   G = gel(CHI, 1); chi = gel(CHI, 2);
   10465        1596 :   switch(mfcharorder(CHI))
   10466             :   {
   10467        1148 :     case 1: chi = gen_1; break;
   10468         427 :     case 2: chi = znchartokronecker(G,chi,1); break;
   10469          21 :     default:chi = mkintmod(znconreyexp(G,chi), znstar_get_N(G)); break;
   10470             :   }
   10471        1596 :   return chi;
   10472             : }
   10473             : 
   10474             : GEN
   10475         700 : mfparams(GEN F)
   10476             : {
   10477         700 :   pari_sp av = avma;
   10478             :   GEN z, mf, CHI;
   10479         700 :   if ((mf = checkMF_i(F)))
   10480             :   {
   10481          14 :     long N = MF_get_N(mf);
   10482          14 :     GEN gk = MF_get_gk(mf);
   10483          14 :     CHI = MF_get_CHI(mf);
   10484          14 :     z = mkvec5(utoi(N), gk, CHI, utoi(MF_get_space(mf)), mfcharpol(CHI));
   10485             :   }
   10486             :   else
   10487             :   {
   10488         686 :     if (!checkmf_i(F)) pari_err_TYPE("mfparams", F);
   10489         686 :     z = vec_append(mf_get_NK(F), mfcharpol(mf_get_CHI(F)));
   10490             :   }
   10491         700 :   gel(z,3) = mfchisimpl(gel(z,3));
   10492         700 :   return gc_GEN(av, z);
   10493             : }
   10494             : 
   10495             : GEN
   10496          14 : mfisCM(GEN F)
   10497             : {
   10498          14 :   pari_sp av = avma;
   10499             :   forprime_t S;
   10500             :   GEN D, v;
   10501             :   long N, k, lD, sb, p, i;
   10502          14 :   if (!checkmf_i(F)) pari_err_TYPE("mfisCM", F);
   10503          14 :   N = mf_get_N(F);
   10504          14 :   k = mf_get_k(F); if (N < 0 || k < 0) pari_err_IMPL("mfisCM for this F");
   10505          14 :   D = mfunram(N, -1);
   10506          14 :   lD = lg(D);
   10507          14 :   sb = maxss(mfsturmNk(N, k), 4*N);
   10508          14 :   v = mfcoefs_i(F, sb, 1);
   10509          14 :   u_forprime_init(&S, 2, sb);
   10510         504 :   while ((p = u_forprime_next(&S)))
   10511             :   {
   10512         490 :     GEN ap = gel(v, p+1);
   10513         490 :     if (!gequal0(ap))
   10514         406 :       for (i = 1; i < lD; i++)
   10515         245 :         if (kross(D[i], p) == -1) { D = vecsplice(D, i); lD--; }
   10516             :   }
   10517          14 :   if (lD == 1) return gc_const(av, gen_0);
   10518          14 :   if (lD == 2) return gc_stoi(av, D[1]);
   10519           7 :   if (k > 1) pari_err_BUG("mfisCM");
   10520           7 :   return gc_upto(av, zv_to_ZV(D));
   10521             : }
   10522             : 
   10523             : static long
   10524         287 : mfspace_i(GEN mf, GEN F)
   10525             : {
   10526             :   GEN v, vF, gk;
   10527             :   long n, nE, i, l, s, N;
   10528             : 
   10529         287 :   mf = checkMF(mf); s = MF_get_space(mf);
   10530         287 :   if (!F) return s;
   10531         287 :   if (!checkmf_i(F)) pari_err_TYPE("mfspace",F);
   10532         287 :   v = mftobasis(mf, F, 1);
   10533         287 :   n = lg(v)-1; if (!n) return -1;
   10534         224 :   nE = lg(MF_get_E(mf))-1;
   10535         224 :   switch(s)
   10536             :   {
   10537          56 :     case mf_NEW: case mf_OLD: case mf_EISEN: return s;
   10538         140 :     case mf_FULL:
   10539         140 :       if (mf_get_type(F) == t_MF_THETA) return mf_EISEN;
   10540         133 :       if (!gequal0(vecslice(v,1,nE)))
   10541          63 :         return gequal0(vecslice(v,nE+1,n))? mf_EISEN: mf_FULL;
   10542             :   }
   10543             :   /* mf is mf_CUSP or mf_FULL, F a cusp form */
   10544          98 :   gk = mf_get_gk(F);
   10545          98 :   if (typ(gk) == t_FRAC || equali1(gk)) return mf_CUSP;
   10546          84 :   vF = mftonew_i(mf, vecslice(v, nE+1, n), &N);
   10547          84 :   if (N != MF_get_N(mf)) return mf_OLD;
   10548          56 :   l = lg(vF);
   10549          91 :   for (i = 1; i < l; i++)
   10550          56 :     if (itos(gmael(vF,i,1)) != N) return mf_CUSP;
   10551          35 :   return mf_NEW;
   10552             : }
   10553             : long
   10554         287 : mfspace(GEN mf, GEN F)
   10555         287 : { pari_sp av = avma; return gc_long(av, mfspace_i(mf,F)); }
   10556             : static GEN
   10557          21 : lfunfindchi(GEN ldata, GEN van, long prec)
   10558             : {
   10559          21 :   GEN gN = ldata_get_conductor(ldata), gk = ldata_get_k(ldata);
   10560          21 :   GEN G = znstar0(gN,1), cyc = znstar_get_conreycyc(G), L, go, vz;
   10561          21 :   long N = itou(gN), odd = typ(gk) == t_INT && mpodd(gk);
   10562          21 :   long i, j, o, l, B0 = 2, B = lg(van)-1, bit = 10 - prec2nbits(prec);
   10563             : 
   10564             :   /* if van is integral, chi must be trivial */
   10565          21 :   if (typ(van) == t_VECSMALL) return mfcharGL(G, zerocol(lg(cyc)-1));
   10566          14 :   L = cyc2elts(cyc); l = lg(L);
   10567          42 :   for (i = j = 1; i < l; i++)
   10568             :   {
   10569          28 :     GEN chi = zc_to_ZC(gel(L,i));
   10570          28 :     if (zncharisodd(G,chi) == odd) gel(L,j++) = mfcharGL(G,chi);
   10571             :   }
   10572          14 :   setlg(L,j); l = j;
   10573          14 :   if (l <= 2) return gel(L,1);
   10574           0 :   o = znstar_get_expo(G); go = utoi(o);
   10575           0 :   vz = grootsof1(o, prec);
   10576             :   for (;;)
   10577           0 :   {
   10578             :     long n;
   10579           0 :     for (n = B0; n <= B; n++)
   10580             :     {
   10581             :       GEN an, r;
   10582             :       long j;
   10583           0 :       if (ugcd(n, N) != 1) continue;
   10584           0 :       an = gel(van,n); if (gexpo(an) < bit) continue;
   10585           0 :       r = gdiv(an, conj_i(an));
   10586           0 :       for (i = 1; i < l; i++)
   10587             :       {
   10588           0 :         GEN CHI = gel(L,i);
   10589           0 :         if (gexpo(gsub(r, gel(vz, znchareval_i(CHI,n,go)+1))) > bit)
   10590           0 :           gel(L,i) = NULL;
   10591             :       }
   10592           0 :       for (i = j = 1; i < l; i++)
   10593           0 :         if (gel(L,i)) gel(L,j++) = gel(L,i);
   10594           0 :       l = j; setlg(L,l);
   10595           0 :       if (l == 2) return gel(L,1);
   10596             :     }
   10597           0 :     B0 = B+1; B <<= 1;
   10598           0 :     van = ldata_vecan(ldata_get_an(ldata), B, prec);
   10599             :   }
   10600             : }
   10601             : 
   10602             : GEN
   10603          21 : mffromlfun(GEN L, long prec)
   10604             : {
   10605          21 :   pari_sp av = avma;
   10606          21 :   GEN ldata = lfunmisc_to_ldata_shallow(L), Vga = ldata_get_gammavec(ldata);
   10607          21 :   GEN van, a0, CHI, NK, gk = ldata_get_k(ldata);
   10608             :   long N, space;
   10609          21 :   if (!gequal(Vga, mkvec2(gen_0, gen_1))) pari_err_TYPE("mffromlfun", L);
   10610          21 :   N = itou(ldata_get_conductor(ldata));
   10611          21 :   van = ldata_vecan(ldata_get_an(ldata), mfsturmNgk(N,gk) + 2, prec);
   10612          21 :   CHI = lfunfindchi(ldata, van, prec);
   10613          21 :   if (typ(van) != t_VEC) van = vecsmall_to_vec_inplace(van);
   10614          21 :   space = (lg(ldata) == 7)? mf_CUSP: mf_FULL;
   10615          21 :   a0 = (space == mf_CUSP)? gen_0: gneg(lfun(L, gen_0, prec2nbits(prec)));
   10616          21 :   NK = mkvec3(utoi(N), gk, mfchisimpl(CHI));
   10617          21 :   return gc_GEN(av, mkvec3(NK, utoi(space), shallowconcat(a0, van)));
   10618             : }
   10619             : /*******************************************************************/
   10620             : /*                                                                 */
   10621             : /*                       HALF-INTEGRAL WEIGHT                      */
   10622             : /*                                                                 */
   10623             : /*******************************************************************/
   10624             : /* We use the prefix mf2; k represents the weight -1/2, so e.g.
   10625             :    k = 2 is weight 5/2. N is the level, so 4\mid N, and CHI is the
   10626             :    character, always even. */
   10627             : 
   10628             : static long
   10629        3360 : lamCO(long r, long s, long p)
   10630             : {
   10631        3360 :   if ((s << 1) <= r)
   10632             :   {
   10633        1232 :     long rp = r >> 1;
   10634        1232 :     if (odd(r)) return upowuu(p, rp) << 1;
   10635         336 :     else return (p + 1)*upowuu(p, rp - 1);
   10636             :   }
   10637        2128 :   else return upowuu(p, r - s) << 1;
   10638             : }
   10639             : 
   10640             : static int
   10641        1568 : condC(GEN faN, GEN valF)
   10642             : {
   10643        1568 :   GEN P = gel(faN, 1), E = gel(faN, 2);
   10644        1568 :   long l = lg(P), i;
   10645        3696 :   for (i = 1; i < l; i++)
   10646        3024 :     if ((P[i] & 3L) == 3)
   10647             :     {
   10648        1120 :       long r = E[i];
   10649        1120 :       if (odd(r) || r < (valF[i] << 1)) return 1;
   10650             :     }
   10651         672 :   return 0;
   10652             : }
   10653             : 
   10654             : /* returns 2*zetaCO; weight is k + 1/2 */
   10655             : static long
   10656        3696 : zeta2CO(GEN faN, GEN valF, long r2, long s2, long k)
   10657             : {
   10658        3696 :   if (r2 >= 4) return lamCO(r2, s2, 2) << 1;
   10659        2912 :   if (r2 == 3) return 6;
   10660        1568 :   if (condC(faN, valF)) return 4;
   10661         672 :   if (odd(k)) return s2 ? 3 : 5; else return s2 ? 5: 3;
   10662             : }
   10663             : 
   10664             : /* returns 4 times last term in formula */
   10665             : static long
   10666        3696 : dim22(long N, long F, long k)
   10667             : {
   10668        3696 :   pari_sp av = avma;
   10669        3696 :   GEN vF, faN = myfactoru(N), P = gel(faN, 1), E = gel(faN, 2);
   10670        3696 :   long i, D, l = lg(P);
   10671        3696 :   vF = cgetg(l, t_VECSMALL);
   10672        9968 :   for (i = 1; i < l; i++) vF[i] = u_lval(F, P[i]);
   10673        3696 :   D = zeta2CO(faN, vF, E[1], vF[1], k);
   10674        6272 :   for (i = 2; i < l; i++) D *= lamCO(E[i], vF[i], P[i]);
   10675        3696 :   return gc_long(av,D);
   10676             : }
   10677             : 
   10678             : /* PSI not necessarily primitive, of conductor F */
   10679             : static int
   10680       13846 : charistotallyeven(GEN PSI, long F)
   10681             : {
   10682       13846 :   pari_sp av = avma;
   10683       13846 :   GEN P = gel(myfactoru(F), 1);
   10684       13846 :   GEN G = gel(PSI,1), psi = gel(PSI,2);
   10685             :   long i;
   10686       14350 :   for (i = 1; i < lg(P); i++)
   10687             :   {
   10688         532 :     GEN psip = znchardecompose(G, psi, utoipos(P[i]));
   10689         532 :     if (zncharisodd(G, psip)) return gc_bool(av,0);
   10690             :   }
   10691       13818 :   return gc_bool(av,1);
   10692             : }
   10693             : 
   10694             : static GEN
   10695      299775 : get_PSI(GEN CHI, long t)
   10696             : {
   10697      299775 :   long r = t & 3L, t2 = (r == 2 || r == 3) ? t << 2 : t;
   10698      299775 :   return mfcharmul_i(CHI, induce(gel(CHI,1), utoipos(t2)));
   10699             : }
   10700             : /* space = mf_CUSP, mf_EISEN or mf_FULL, weight k + 1/2 */
   10701             : static long
   10702       41363 : mf2dimwt12(long N, GEN CHI, long space)
   10703             : {
   10704       41363 :   pari_sp av = avma;
   10705       41363 :   GEN D = mydivisorsu(N >> 2);
   10706       41363 :   long i, l = lg(D), dim3 = 0, dim4 = 0;
   10707             : 
   10708       41363 :   CHI = induceN(N, CHI);
   10709      341138 :   for (i = 1; i < l; i++)
   10710             :   {
   10711      299775 :     long rp, t = D[i], Mt = D[l-i];
   10712      299775 :     GEN PSI = get_PSI(CHI,t);
   10713      299775 :     rp = mfcharconductor(PSI);
   10714      299775 :     if (Mt % (rp*rp) == 0) { dim4++; if (charistotallyeven(PSI,rp)) dim3++; }
   10715             :   }
   10716       41363 :   set_avma(av);
   10717       41363 :   switch (space)
   10718             :   {
   10719       40439 :     case mf_CUSP: return dim4 - dim3;
   10720         462 :     case mf_EISEN:return dim3;
   10721         462 :     case mf_FULL: return dim4;
   10722             :   }
   10723             :   return 0; /*LCOV_EXCL_LINE*/
   10724             : }
   10725             : 
   10726             : static long
   10727         693 : mf2dimwt32(long N, GEN CHI, long F, long space)
   10728             : {
   10729             :   long D;
   10730         693 :   switch(space)
   10731             :   {
   10732         231 :     case mf_CUSP: D = mypsiu(N) - 6*dim22(N, F, 1);
   10733         231 :       if (D%24) pari_err_BUG("mfdim");
   10734         231 :       return D/24 + mf2dimwt12(N, CHI, 4);
   10735         231 :     case mf_FULL: D = mypsiu(N) + 6*dim22(N, F, 0);
   10736         231 :       if (D%24) pari_err_BUG("mfdim");
   10737         231 :       return D/24 + mf2dimwt12(N, CHI, 1);
   10738         231 :     case mf_EISEN: D = dim22(N, F, 0) + dim22(N, F, 1);
   10739         231 :       if (D & 3L) pari_err_BUG("mfdim");
   10740         231 :       return (D >> 2) - mf2dimwt12(N, CHI, 3);
   10741             :   }
   10742             :   return 0; /*LCOV_EXCL_LINE*/
   10743             : }
   10744             : 
   10745             : /* F = conductor(CHI), weight k = r+1/2 */
   10746             : static long
   10747       43736 : checkmf2(long N, long r, GEN CHI, long F, long space)
   10748             : {
   10749       43736 :   switch(space)
   10750             :   {
   10751       43715 :     case mf_FULL: case mf_CUSP: case mf_EISEN: break;
   10752          14 :     case mf_NEW: case mf_OLD:
   10753          14 :       pari_err_TYPE("half-integral weight [new/old spaces]", utoi(space));
   10754           7 :     default:
   10755           7 :       pari_err_TYPE("half-integral weight [incorrect space]",utoi(space));
   10756             :   }
   10757       43715 :   if (N & 3L)
   10758           0 :     pari_err_DOMAIN("half-integral weight", "N % 4", "!=", gen_0, stoi(N));
   10759       43715 :   return r >= 0 && mfcharparity(CHI) == 1 && N % F == 0;
   10760             : }
   10761             : 
   10762             : /* weight k = r + 1/2 */
   10763             : static long
   10764       43463 : mf2dim_Nkchi(long N, long r, GEN CHI, ulong space)
   10765             : {
   10766       43463 :   long D, D2, F = mfcharconductor(CHI);
   10767       43463 :   if (!checkmf2(N, r, CHI, F, space)) return 0;
   10768       43442 :   if (r == 0) return mf2dimwt12(N, CHI, space);
   10769        2772 :   if (r == 1) return mf2dimwt32(N, CHI, F, space);
   10770        2079 :   if (space == mf_EISEN)
   10771             :   {
   10772         693 :     D = dim22(N, F, r) + dim22(N, F, 1-r);
   10773         693 :     if (D & 3L) pari_err_BUG("mfdim");
   10774         693 :     return D >> 2;
   10775             :   }
   10776        1386 :   D2 = space == mf_FULL? dim22(N, F, 1-r): -dim22(N, F, r);
   10777        1386 :   D = (2*r-1)*mypsiu(N) + 6*D2;
   10778        1386 :   if (D%24) pari_err_BUG("mfdim");
   10779        1386 :   return D/24;
   10780             : }
   10781             : 
   10782             : /* weight k=r+1/2 */
   10783             : static GEN
   10784         273 : mf2init_Nkchi(long N, long r, GEN CHI, long space, long flraw)
   10785             : {
   10786         273 :   GEN CHI1, Minv, Minvmat, B, M, gk = gaddsg(r,ghalf);
   10787         273 :   GEN mf1 = mkvec4(utoi(N),gk,CHI,utoi(space));
   10788             :   long L;
   10789         273 :   if (!checkmf2(N, r, CHI, mfcharconductor(CHI), space)) return mfEMPTY(mf1);
   10790         273 :   if (space==mf_EISEN) pari_err_IMPL("half-integral weight Eisenstein space");
   10791         273 :   L = mfsturmNgk(N, gk) + 1;
   10792         273 :   B = mf2basis(N, r, CHI, &CHI1, space);
   10793         273 :   M = mflineardivtomat(N,B,L); /* defined modulo T = charpol(CHI) */
   10794         273 :   if (flraw) M = mkvec3(gen_0,gen_0,M);
   10795             :   else
   10796             :   {
   10797         273 :     long o1 = mfcharorder(CHI1), o = mfcharorder(CHI);
   10798         273 :     M = mfcleanCHI(M, CHI, 0);
   10799         273 :     Minv = gel(M,2);
   10800         273 :     Minvmat = RgM_Minv_mul(NULL, Minv); /* mod T */
   10801         273 :     if (o1 != o)
   10802             :     {
   10803         140 :       GEN tr = Qab_trace_init(o, o1, mfcharpol(CHI), mfcharpol(CHI1));
   10804         140 :       Minvmat = QabM_tracerel(tr, 0, Minvmat);
   10805             :     }
   10806             :     /* Minvmat mod T1 = charpol(CHI1) */
   10807         273 :     B = vecmflineardiv_linear(B, Minvmat);
   10808         273 :     gel(M,3) = RgM_Minv_mul(gel(M,3), Minv);
   10809         273 :     gel(M,2) = mkMinv(matid(lg(B)-1), NULL,NULL,NULL);
   10810             :   }
   10811         273 :   return mkmf(mf1, cgetg(1,t_VEC), B, gen_0, M);
   10812             : }
   10813             : 
   10814             : /**************************************************************************/
   10815             : /*                          Kohnen + space                                */
   10816             : /**************************************************************************/
   10817             : 
   10818             : static GEN
   10819          28 : mfkohnenbasis_i(GEN mf, GEN CHI, long eps, long sb)
   10820             : {
   10821          28 :   GEN M = mfcoefs_mf(mf, sb, 1), p, P;
   10822          28 :   long c, i, n = mfcharorder(CHI), l = sb + 2;
   10823          28 :   p = cgetg(l, t_VECSMALL);
   10824             :   /* keep the a_n, n = (2 or 2+eps) mod 4 */
   10825         273 :   for (i = 3, c = 1; i < l; i+=4) p[c++] = i;
   10826         266 :   for (i = 3+eps;    i < l; i+=4) p[c++] = i;
   10827          28 :   P = n <= 2? NULL: mfcharpol(CHI);
   10828          28 :   setlg(p, c);
   10829          28 :   return QabM_ker(rowpermute(M, p), P, n);
   10830             : }
   10831             : GEN
   10832          28 : mfkohnenbasis(GEN mf)
   10833             : {
   10834          28 :   pari_sp av = avma;
   10835             :   GEN gk, CHI, CHIP, K;
   10836             :   long N4, r, eps, sb;
   10837          28 :   mf = checkMF(mf);
   10838          28 :   if (MF_get_space(mf) != mf_CUSP)
   10839           0 :     pari_err_TYPE("mfkohnenbasis [not a cuspidal space", mf);
   10840          28 :   if (!MF_get_dim(mf)) return cgetg(1, t_MAT);
   10841          28 :   N4 = MF_get_N(mf) >> 2; gk = MF_get_gk(mf); CHI = MF_get_CHI(mf);
   10842          28 :   if (typ(gk) == t_INT) pari_err_TYPE("mfkohnenbasis", gk);
   10843          28 :   r = MF_get_r(mf);
   10844          28 :   CHIP = mfcharchiliftprim(CHI, N4);
   10845          28 :   eps = CHIP==CHI? 1: -1;
   10846          28 :   if (odd(r)) eps = -eps;
   10847          28 :   if (uissquarefree(N4))
   10848             :   {
   10849          21 :     long d = mfdim_Nkchi(N4, 2*r, mfcharpow(CHI, gen_2), mf_CUSP);
   10850          21 :     sb = mfsturmNgk(N4 << 2, gk) + 1;
   10851          21 :     K = mfkohnenbasis_i(mf, CHIP, eps, sb);
   10852          21 :     if (lg(K) - 1 == d) return gc_GEN(av, K);
   10853             :   }
   10854           7 :   sb = mfsturmNgk(N4 << 4, gk) + 1;
   10855           7 :   K = mfkohnenbasis_i(mf, CHIP, eps, sb);
   10856           7 :   return gc_GEN(av, K);
   10857             : }
   10858             : 
   10859             : static GEN
   10860          21 : get_Shimura(GEN mf, GEN CHI, GEN vB, long t)
   10861             : {
   10862          21 :   long N = MF_get_N(mf), r = MF_get_k(mf) >> 1;
   10863          21 :   long i, d = MF_get_dim(mf), sb = mfsturm_mf(mf);
   10864          21 :   GEN a = cgetg(d+1, t_MAT);
   10865          84 :   for (i = 1; i <= d; i++)
   10866             :   {
   10867          63 :     pari_sp av = avma;
   10868          63 :     GEN f = c_deflate(sb*sb, t, gel(vB,i));
   10869          63 :     f = mftobasis_i(mf, RgV_shimura(f, sb, t, N, r, CHI));
   10870          63 :     gel(a,i) = gc_upto(av, f);
   10871             :   }
   10872          21 :   return a;
   10873             : }
   10874             : static long
   10875          35 : QabM_rank(GEN M, GEN P, long n)
   10876             : {
   10877          35 :   GEN z = QabM_indexrank(M, P, n);
   10878          35 :   return lg(gel(z,2))-1;
   10879             : }
   10880             : /* discard T[*i] */
   10881             : static void
   10882           0 : discard_Ti(GEN T, long *i, long *lt)
   10883             : {
   10884           0 :   long j, l = *lt-1;
   10885           0 :   for (j = *i; j < l; j++) T[j] = T[j+1];
   10886           0 :   (*i)--; *lt = l;
   10887           0 : }
   10888             : /* return [mf3, bijection, mfkohnenbasis, codeshi] */
   10889             : static GEN
   10890          14 : mfkohnenbijection_i(GEN mf)
   10891             : {
   10892          14 :   GEN CHI = MF_get_CHI(mf), K = mfkohnenbasis(mf);
   10893             :   GEN mres, dMi, Mi, M, C, vB, mf3, SHI, T, P;
   10894          14 :   long N4 = MF_get_N(mf)>>2, r = MF_get_r(mf), dK = lg(K) - 1;
   10895             :   long i, c, n, oldr, lt, ltold, sb3, t, limt;
   10896          14 :   const long MAXlt = 100;
   10897             : 
   10898          14 :   mf3 = mfinit_Nkchi(N4, r<<1, mfcharpow(CHI,gen_2), mf_CUSP, 0);
   10899          14 :   if (MF_get_dim(mf3) != dK)
   10900           0 :     pari_err_BUG("mfkohnenbijection [different dimensions]");
   10901          14 :   if (!dK) return mkvec4(mf3, cgetg(1, t_MAT), K, cgetg(1, t_VEC));
   10902          14 :   CHI = mfcharchiliftprim(CHI, N4);
   10903          14 :   n = mfcharorder(CHI);
   10904          14 :   P = n<=2? NULL: mfcharpol(CHI);
   10905          14 :   SHI = cgetg(MAXlt, t_COL);
   10906          14 :   T = cgetg(MAXlt, t_VECSMALL);
   10907          14 :   sb3 = mfsturm_mf(mf3);
   10908          14 :   limt = 6; oldr = 0; vB = C = M = NULL;
   10909          98 :   for (t = lt = ltold = 1; lt < MAXlt; t++)
   10910             :   {
   10911             :     pari_sp av;
   10912          98 :     if (!uissquarefree(t)) continue;
   10913          84 :     T[lt++] = t; if (t <= limt) continue;
   10914          14 :     av = avma;
   10915          14 :     if (vB) gunclone(vB);
   10916             :     /* could improve the rest but 99% of running time is spent here */
   10917          14 :     vB = gclone( RgM_mul(mfcoefs_mf(mf, t*sb3*sb3, 1), K) );
   10918          14 :     set_avma(av);
   10919          21 :     for (i = ltold; i < lt; i++)
   10920             :     {
   10921             :       pari_sp av;
   10922             :       long r;
   10923          21 :       M = get_Shimura(mf3, CHI, vB, T[i]);
   10924          21 :       r = QabM_rank(M, P, n); if (!r) { discard_Ti(T, &i, &lt); continue; }
   10925          21 :       gel(SHI, i) = M; setlg(SHI, i+1);
   10926          21 :       if (r >= dK) { C = vecsmall_ei(dK, i); goto DONE; }
   10927          14 :       if (i == 1) { oldr = r; continue; }
   10928           7 :       av = avma; M = shallowmatconcat(SHI);
   10929           7 :       r = QabM_rank(M, P, n); /* >= rank(sum C[j] SHI[j]), probably sharp */
   10930           7 :       if (r >= dK)
   10931             :       {
   10932           7 :         M = RgV_sum(SHI);
   10933           7 :         if (QabM_rank(M, P, n) >= dK) { C = const_vecsmall(dK, 1); goto DONE; }
   10934           0 :         C = random_Flv(dK, 16);
   10935           0 :         M = RgV_zc_mul(SHI, C);
   10936           0 :         if (QabM_rank(M, P, n) >= dK) goto DONE;
   10937             :       }
   10938           0 :       else if (r == oldr) discard_Ti(T, &i, &lt);
   10939           0 :       oldr = r; set_avma(av);
   10940             :     }
   10941           0 :     limt *= 2; ltold = lt;
   10942             :   }
   10943           0 :   pari_err_BUG("mfkohnenbijection");
   10944          14 : DONE:
   10945          14 :   gunclone(vB); lt = lg(SHI);
   10946          14 :   Mi = QabM_pseudoinv(M,P,n, NULL,&dMi); Mi = RgM_Rg_div(Mi,dMi);
   10947          14 :   mres = cgetg(lt, t_VEC);
   10948          35 :   for (i = c = 1; i < lt; i++)
   10949          21 :     if (C[i]) gel(mres,c++) = mkvec2s(T[i], C[i]);
   10950          14 :   setlg(mres,c); return mkvec4(mf3, Mi, K, mres);
   10951             : }
   10952             : GEN
   10953          14 : mfkohnenbijection(GEN mf)
   10954             : {
   10955          14 :   pari_sp av = avma;
   10956             :   long N;
   10957          14 :   mf = checkMF(mf); N = MF_get_N(mf);
   10958          14 :   if (!uissquarefree(N >> 2))
   10959           0 :     pari_err_TYPE("mfkohnenbijection [N/4 not squarefree]", utoi(N));
   10960          14 :   if (MF_get_space(mf) != mf_CUSP || MF_get_r(mf) == 0 || !mfshimura_space_cusp(mf))
   10961           0 :     pari_err_TYPE("mfkohnenbijection [incorrect mf for Kohnen]", mf);
   10962          14 :   return gc_GEN(av, mfkohnenbijection_i(mf));
   10963             : }
   10964             : 
   10965             : static int
   10966           7 : checkbij_i(GEN b)
   10967             : {
   10968           7 :   return typ(b) == t_VEC && lg(b) == 5 && checkMF_i(gel(b,1))
   10969           7 :          && typ(gel(b,2)) == t_MAT
   10970           7 :          && typ(gel(b,3)) == t_MAT
   10971          14 :          && typ(gel(b,4)) == t_VEC;
   10972             : }
   10973             : 
   10974             : /* bij is the output of mfkohnenbijection */
   10975             : GEN
   10976           7 : mfkohneneigenbasis(GEN mf, GEN bij)
   10977             : {
   10978           7 :   pari_sp av = avma;
   10979             :   GEN mf3, mf30, B, KM, M, k;
   10980             :   long r, i, l, N4;
   10981           7 :   mf = checkMF(mf);
   10982           7 :   if (!checkbij_i(bij))
   10983           0 :     pari_err_TYPE("mfkohneneigenbasis [bijection]", bij);
   10984           7 :   if (MF_get_space(mf) != mf_CUSP)
   10985           0 :     pari_err_TYPE("mfkohneneigenbasis [not a cuspidal space]", mf);
   10986           7 :   if (!MF_get_dim(mf))
   10987           0 :     retmkvec3(cgetg(1, t_VEC), cgetg(1, t_VEC), cgetg(1, t_VEC));
   10988           7 :   N4 = MF_get_N(mf) >> 2; k = MF_get_gk(mf);
   10989           7 :   if (typ(k) == t_INT) pari_err_TYPE("mfkohneneigenbasis", k);
   10990           7 :   if (!uissquarefree(N4))
   10991           0 :     pari_err_TYPE("mfkohneneigenbasis [N not squarefree]", utoipos(N4));
   10992           7 :   r = MF_get_r(mf);
   10993           7 :   KM = RgM_mul(gel(bij,3), gel(bij,2));
   10994           7 :   mf3 = gel(bij,1);
   10995           7 :   mf30 = mfinit_Nkchi(N4, 2*r, MF_get_CHI(mf3), mf_NEW, 0);
   10996           7 :   B = mfcoefs_mf(mf30, mfsturm_mf(mf3), 1); l = lg(B);
   10997           7 :   M = cgetg(l, t_MAT);
   10998          21 :   for (i=1; i<l; i++) gel(M,i) = RgM_RgC_mul(KM, mftobasis_i(mf3, gel(B,i)));
   10999           7 :   return gc_GEN(av, mkvec3(mf30, M, RgM_mul(M, MF_get_newforms(mf30))));
   11000             : }
   11001             : /*************************** End Kohnen ************************************/
   11002             : /***************************************************************************/
   11003             : 
   11004             : static GEN desc(GEN F);
   11005             : static GEN
   11006         504 : desc_mfeisen(GEN F)
   11007             : {
   11008         504 :   GEN R, gk = mf_get_gk(F);
   11009         504 :   if (typ(gk) == t_FRAC)
   11010           7 :     R = gsprintf("H_{%Ps}", gk);
   11011             :   else
   11012             :   {
   11013         497 :     GEN vchi = gel(F, 2), CHI = mfchisimpl(gel(vchi, 3));
   11014         497 :     long k = itou(gk);
   11015         497 :     if (lg(vchi) < 5) R = gsprintf("F_%ld(%Ps)", k, CHI);
   11016             :     else
   11017             :     {
   11018         294 :       GEN CHI2 = mfchisimpl(gel(vchi, 4));
   11019         294 :       R = gsprintf("F_%ld(%Ps, %Ps)", k, CHI, CHI2);
   11020             :     }
   11021             :   }
   11022         504 :   return R;
   11023             : }
   11024             : static GEN
   11025          35 : desc_hecke(GEN F)
   11026             : {
   11027             :   long n, N;
   11028          35 :   GEN D = gel(F,2);
   11029          35 :   if (typ(D) == t_VECSMALL) { N = D[3]; n = D[1]; }
   11030          14 :   else { GEN nN = gel(D,2); n = nN[1]; N = nN[2]; } /* half integer */
   11031          35 :   return gsprintf("T_%ld(%ld)(%Ps)", N, n, desc(gel(F,3)));
   11032             : }
   11033             : static GEN
   11034          98 : desc_linear(GEN FLD, GEN dL)
   11035             : {
   11036          98 :   GEN F = gel(FLD,2), L = gel(FLD,3), R = strtoGENstr("LIN([");
   11037          98 :   long n = lg(F) - 1, i;
   11038         168 :   for (i = 1; i <= n; i++)
   11039             :   {
   11040         168 :     R = shallowconcat(R, desc(gel(F,i))); if (i == n) break;
   11041          70 :     R = shallowconcat(R, strtoGENstr(", "));
   11042             :   }
   11043          98 :   return shallowconcat(R, gsprintf("], %Ps)", gdiv(L, dL)));
   11044             : }
   11045             : static GEN
   11046          21 : desc_dihedral(GEN F)
   11047             : {
   11048          21 :   GEN bnr = gel(F,2), D = nf_get_disc(bnr_get_nf(bnr)), f = bnr_get_mod(bnr);
   11049          21 :   GEN cyc = bnr_get_cyc(bnr);
   11050          21 :   GEN w = gel(F,3), chin = zv_to_ZV(gel(w,2)), o = utoi(gel(w,1)[1]);
   11051          21 :   GEN chi = char_denormalize(cyc, o, chin);
   11052          21 :   if (lg(gel(f,2)) == 1) f = gel(f,1);
   11053          21 :   return gsprintf("DIH(%Ps, %Ps, %Ps, %Ps)",D,f,cyc,chi);
   11054             : }
   11055             : 
   11056             : static void
   11057        1043 : unpack0(GEN *U)
   11058        1043 : { if (U) *U = mkvec2(cgetg(1, t_VEC), cgetg(1, t_VEC)); }
   11059             : static void
   11060          42 : unpack2(GEN F, GEN *U)
   11061          42 : { if (U) *U = mkvec2(mkvec2(gel(F,2), gel(F,3)), cgetg(1, t_VEC)); }
   11062             : static void
   11063         308 : unpack23(GEN F, GEN *U)
   11064         308 : { if (U) *U = mkvec2(mkvec(gel(F,2)), mkvec(gel(F,3))); }
   11065             : static GEN
   11066        1540 : desc_i(GEN F, GEN *U)
   11067             : {
   11068        1540 :   switch(mf_get_type(F))
   11069             :   {
   11070           7 :     case t_MF_CONST: unpack0(U); return gsprintf("CONST(%Ps)", gel(F,2));
   11071         504 :     case t_MF_EISEN: unpack0(U); return desc_mfeisen(F);
   11072         154 :     case t_MF_Ek: unpack0(U); return gsprintf("E_%ld", mf_get_k(F));
   11073          63 :     case t_MF_DELTA: unpack0(U); return gsprintf("DELTA");
   11074          35 :     case t_MF_THETA: unpack0(U);
   11075          35 :       return gsprintf("THETA(%Ps)", mfchisimpl(gel(F,2)));
   11076          56 :     case t_MF_ETAQUO: unpack0(U);
   11077          56 :       return gsprintf("ETAQUO(%Ps, %Ps)", gel(F,2), gel(F,3));
   11078          56 :     case t_MF_ELL: unpack0(U);
   11079          56 :       return gsprintf("ELL(%Ps)", vecslice(gel(F,2), 1, 5));
   11080           7 :     case t_MF_TRACE: unpack0(U); return gsprintf("TR(%Ps)", mfparams(F));
   11081         140 :     case t_MF_NEWTRACE: unpack0(U); return gsprintf("TR^new(%Ps)", mfparams(F));
   11082          21 :     case t_MF_DIHEDRAL: unpack0(U); return desc_dihedral(F);
   11083          28 :     case t_MF_MUL: unpack2(F, U);
   11084          28 :       return gsprintf("MUL(%Ps, %Ps)", desc(gel(F,2)), desc(gel(F,3)));
   11085          14 :     case t_MF_DIV: unpack2(F, U);
   11086          14 :       return gsprintf("DIV(%Ps, %Ps)", desc(gel(F,2)), desc(gel(F,3)));
   11087          14 :     case t_MF_POW: unpack23(F, U);
   11088          14 :       return gsprintf("POW(%Ps, %ld)", desc(gel(F,2)), itos(gel(F,3)));
   11089          14 :     case t_MF_SHIFT: unpack23(F, U);
   11090          14 :       return gsprintf("SHIFT(%Ps, %ld)", desc(gel(F,2)), itos(gel(F,3)));
   11091          14 :     case t_MF_DERIV: unpack23(F, U);
   11092          14 :       return gsprintf("DER^%ld(%Ps)", itos(gel(F,3)), desc(gel(F,2)));
   11093          21 :     case t_MF_DERIVE2: unpack23(F, U);
   11094          21 :       return gsprintf("DERE2^%ld(%Ps)", itos(gel(F,3)), desc(gel(F,2)));
   11095          14 :     case t_MF_TWIST: unpack23(F, U);
   11096          14 :       return gsprintf("TWIST(%Ps, %Ps)", desc(gel(F,2)), gel(F,3));
   11097         231 :     case t_MF_BD: unpack23(F, U);
   11098         231 :       return gsprintf("B(%ld)(%Ps)", itou(gel(F,3)), desc(gel(F,2)));
   11099          14 :     case t_MF_BRACKET:
   11100          14 :       if (U) *U = mkvec2(mkvec2(gel(F,2), gel(F,3)), mkvec(gel(F,4)));
   11101          14 :       return gsprintf("MULRC_%ld(%Ps, %Ps)", itos(gel(F,4)), desc(gel(F,2)), desc(gel(F,3)));
   11102          98 :     case t_MF_LINEAR_BHN:
   11103             :     case t_MF_LINEAR:
   11104          98 :       if (U) *U = mkvec2(gel(F,2), mkvec(gdiv(gel(F,3), gel(F,4))));
   11105          98 :       return desc_linear(F,gel(F,4));
   11106          35 :     case t_MF_HECKE:
   11107          35 :       if (U) *U = mkvec2(mkvec(gel(F,3)), mkvec(stoi(gel(F,2)[1])));
   11108          35 :       return desc_hecke(F);
   11109           0 :     default: pari_err_TYPE("mfdescribe",F);
   11110             :     return NULL;/*LCOV_EXCL_LINE*/
   11111             :   }
   11112             : }
   11113             : static GEN
   11114         623 : desc(GEN F) { return desc_i(F, NULL); }
   11115             : GEN
   11116         966 : mfdescribe(GEN F, GEN *U)
   11117             : {
   11118         966 :   pari_sp av = avma;
   11119             :   GEN mf;
   11120         966 :   if ((mf = checkMF_i(F)))
   11121             :   {
   11122          49 :     const char *f = NULL;
   11123          49 :     switch (MF_get_space(mf))
   11124             :     {
   11125           7 :       case mf_NEW:  f = "S_%Ps^new(G_0(%ld, %Ps))"; break;
   11126          14 :       case mf_CUSP: f = "S_%Ps(G_0(%ld, %Ps))"; break;
   11127           7 :       case mf_OLD:  f = "S_%Ps^old(G_0(%ld, %Ps))"; break;
   11128           7 :       case mf_EISEN:f = "E_%Ps(G_0(%ld, %Ps))"; break;
   11129          14 :       case mf_FULL: f = "M_%Ps(G_0(%ld, %Ps))"; break;
   11130             :     }
   11131          49 :     if (U) *U = cgetg(1, t_VEC);
   11132          49 :     return gsprintf(f, MF_get_gk(mf), MF_get_N(mf), mfchisimpl(MF_get_CHI(mf)));
   11133             :   }
   11134         917 :   if (!checkmf_i(F)) pari_err_TYPE("mfdescribe", F);
   11135         917 :   F = desc_i(F, U); return gc_all(av, U ? 2: 1, &F, U);
   11136             : }
   11137             : 
   11138             : /***********************************************************************/
   11139             : /*               Eisenstein series H_r of weight r+1/2                 */
   11140             : /***********************************************************************/
   11141             : /* radical(u_ppo(g,q)) */
   11142             : static long
   11143          28 : u_pporad(long g, long q)
   11144             : {
   11145          28 :   GEN F = myfactoru(g), P = gel(F,1);
   11146             :   long i, l, n;
   11147          28 :   if (q == 1) return zv_prod(P);
   11148          28 :   l = lg(P);
   11149          35 :   for (i = n = 1; i < l; i++)
   11150             :   {
   11151           7 :     long p = P[i];
   11152           7 :     if (q % p) n *= p;
   11153             :   }
   11154          28 :   return n;
   11155             : }
   11156             : static void
   11157         266 : c_F2TH4(long n, GEN *pF2, GEN *pTH4)
   11158             : {
   11159         266 :   GEN v = mfcoefs_i(mfEk(2), n, 1), v2 = bdexpand(v,2), v4 = bdexpand(v,4);
   11160         266 :   GEN F2 = gdivgs(ZC_add(ZC_sub(v, ZC_z_mul(v2,3)), ZC_z_mul(v4,2)), -24);
   11161         266 :   GEN TH4 = gdivgs(ZC_sub(v, ZC_z_mul(v4,4)), -3);
   11162         266 :   settyp(F2,t_VEC); *pF2 = F2;
   11163         266 :   settyp(TH4,t_VEC);*pTH4= TH4;
   11164         266 : }
   11165             : /* r > 0, N >= 0 */
   11166             : static GEN
   11167          77 : mfEHcoef(long r, long N)
   11168             : {
   11169             :   long D0, f, i, l, s;
   11170             :   GEN S, Df;
   11171             : 
   11172          77 :   if (r == 1) return hclassno(utoi(N));
   11173          77 :   if (N == 0) return gdivgs(bernfrac(2*r), -2*r);
   11174          56 :   s = N & 3L;
   11175          56 :   if (odd(r))
   11176             :   {
   11177          42 :     if (s == 2 || s == 1) return gen_0;
   11178          14 :     D0 = mycoredisc2neg(N,&f);
   11179             :   }
   11180             :   else
   11181             :   {
   11182          14 :     if (s == 2 || s == 3) return gen_0;
   11183          14 :     D0 = mycoredisc2pos(N,&f);
   11184             :   }
   11185          28 :   Df = mydivisorsu(u_pporad(f, D0)); l = lg(Df);
   11186          28 :   S = gen_0;
   11187          63 :   for (i = 1; i < l; i++)
   11188             :   {
   11189          35 :     long d = Df[i], s = mymoebiusu(d)*kross(D0, d); /* != 0 */
   11190          35 :     GEN c = gmul(powuu(d, r-1), mysumdivku(f/d, 2*r-1));
   11191          35 :     S = s > 0? addii(S, c): subii(S, c);
   11192             :   }
   11193          28 :   return gmul(lfunquadneg_naive(D0, r), S);
   11194             : }
   11195             : static GEN
   11196         266 : mfEHmat(long lim, long r)
   11197             : {
   11198         266 :   long j, l, d = r/2;
   11199             :   GEN f2, th4, th3, v, vth4, vf2;
   11200         266 :   c_F2TH4(lim, &f2, &th4);
   11201         266 :   f2 =  RgV_to_ser(f2, 0, lim+3);
   11202         266 :   th4 = RgV_to_ser(th4, 0, lim+3);
   11203         266 :   th3 = RgV_to_ser(c_theta(lim, 1, mfchartrivial()), 0, lim+3);
   11204         266 :   if (odd(r)) th3 = gpowgs(th3, 3);
   11205         266 :   vth4 = gpowers(th4, d);
   11206         266 :   vf2 = gpowers0(f2, d, th3); /* th3 f2^j */
   11207         266 :   l = d+2; v = cgetg(l, t_VEC);
   11208         924 :   for (j = 1; j < l; j++)
   11209         658 :     gel(v, j) = ser2rfrac_i(gmul(gel(vth4, l-j), gel(vf2, j)));
   11210         266 :   return RgXV_to_RgM(v, lim);
   11211             : }
   11212             : static GEN
   11213           7 : Hfind(long r, GEN *pden)
   11214             : {
   11215           7 :   long lim = (r/2)+3, i;
   11216             :   GEN res, M, B;
   11217             : 
   11218           7 :   if (r <= 0) pari_err_DOMAIN("mfEH", "r", "<=", gen_0, stoi(r));
   11219           7 :   M = mfEHmat(lim, r);
   11220           7 :   B = cgetg(lim+1, t_COL);
   11221          56 :   for (i = 1; i <= lim; i++) gel(B, i) = mfEHcoef(r, i-1);
   11222           7 :   res = QM_gauss(M, B);
   11223           7 :   if (lg(res) == 1) pari_err_BUG("mfEH");
   11224           7 :   return Q_remove_denom(res,pden);
   11225             : }
   11226             : GEN
   11227         266 : mfEH(GEN gk)
   11228             : {
   11229         266 :   pari_sp av = avma;
   11230         266 :   GEN v, d, NK, gr = gsub(gk, ghalf);
   11231             :   long r;
   11232         266 :   if (typ(gr) != t_INT) pari_err_TYPE("mfEH", gk);
   11233         266 :   r = itos(gr);
   11234         266 :   switch (r)
   11235             :   {
   11236           7 :     case 1: v=cgetg(1,t_VEC); d=gen_1; break;
   11237         133 :     case 2: v=mkvec2s(1,-20); d=utoipos(120); break;
   11238          56 :     case 3: v=mkvec2s(-1,14); d=utoipos(252); break;
   11239          35 :     case 4: v=mkvec3s(1,-16,16); d=utoipos(240); break;
   11240           7 :     case 5: v=mkvec3s(-1,22,-88); d=utoipos(132); break;
   11241          14 :     case 6: v=mkvec4s(691,-18096,110136,-4160); d=utoipos(32760); break;
   11242           7 :     case 7: v=mkvec4s(-1,30,-240,224); d=utoipos(12); break;
   11243           7 :     default: v = Hfind(r, &d); break;
   11244             :   }
   11245         266 :   NK = mkgNK(utoipos(4), gaddgs(ghalf,r), mfchartrivial(), pol_x(1));
   11246         266 :   return gc_GEN(av, tag(t_MF_EISEN, NK, mkvec2(v,d)));
   11247             : }
   11248             : 
   11249             : /**********************************************************/
   11250             : /*             T(f^2) for half-integral weight            */
   11251             : /**********************************************************/
   11252             : 
   11253             : /* T_p^2 V, p2 = p^2, c1 = chi(p) (-1/p)^r p^(r-1), c2 = chi(p^2)*p^(2r-1) */
   11254             : static GEN
   11255          70 : tp2apply(GEN V, long p, long p2, GEN c1, GEN c2)
   11256             : {
   11257          70 :   long lw = (lg(V) - 2)/p2 + 1, m, n;
   11258          70 :   GEN a0 = gel(V,1), W = cgetg(lw + 1, t_VEC);
   11259             : 
   11260          70 :   gel(W,1) = gequal0(a0)? gen_0: gmul(a0, gaddsg(1, c2));
   11261       11109 :   for (n = 1; n < lw; n++)
   11262             :   {
   11263       11039 :     GEN c = gel(V, p2*n + 1);
   11264       11039 :     if (n%p) c = gadd(c, gmulsg(kross(n,p), gmul(gel(V,n+1), c1)));
   11265       11039 :     gel(W, n+1) = c; /* a(p^2*n) + c1 * (n/p) a(n) */
   11266             :   }
   11267        1253 :   for (m = 1, n = p2; n < lw; m++, n += p2)
   11268        1183 :     gel(W, n+1) = gadd(gel(W,n+1), gmul(gel(V,m+1), c2));
   11269          70 :   return W;
   11270             : }
   11271             : 
   11272             : /* T_{p^{2e}} V; can derecursify [Purkait, Hecke operators in half-integral
   11273             :  * weight, Prop 4.3], not worth it */
   11274             : static GEN
   11275          70 : tp2eapply(GEN V, long p, long p2, long e, GEN q, GEN c1, GEN c2)
   11276             : {
   11277          70 :   GEN V4 = NULL;
   11278          70 :   if (e > 1)
   11279             :   {
   11280          21 :     V4 = vecslice(V, 1, (lg(V) - 2)/(p2*p2) + 1);
   11281          21 :     V = tp2eapply(V, p, p2, e-1, q, c1, c2);
   11282             :   }
   11283          70 :   V = tp2apply(V, p, p2, c1, c2);
   11284          70 :   if (e > 1)
   11285          28 :     V = gsub(V, (e == 2)? gmul(q, V4)
   11286           7 :                         : gmul(c2, tp2eapply(V4, p, p2, e-2, q, c1, c2)));
   11287          70 :   return V;
   11288             : }
   11289             : /* weight k = r+1/2 */
   11290             : static GEN
   11291          98 : RgV_heckef2(long n, long d, GEN V, GEN F, GEN DATA)
   11292             : {
   11293          98 :   GEN CHI = mf_get_CHI(F), fa = gel(DATA,1), P = gel(fa,1), E = gel(fa,2);
   11294          98 :   long i, l = lg(P), r = mf_get_r(F), s4 = odd(r)? -4: 4, k2m2 = (r<<1)-1;
   11295          98 :   if (typ(V) == t_COL) V = shallowtrans(V);
   11296         140 :   for (i = 1; i < l; i++)
   11297             :   { /* p does not divide N */
   11298          42 :     long p = P[i], e = E[i], p2 = p*p;
   11299          42 :     GEN c1, c2, a, b, q = NULL, C = mfchareval(CHI,p), C2 = gsqr(C);
   11300          42 :     a = r? powuu(p,r-1): mkfrac(gen_1,utoipos(p)); /* p^(r-1) = p^(k-3/2) */
   11301          42 :     b = r? mulii(powuu(p,r), a): a; /* p^(2r-1) = p^(2k-2) */
   11302          42 :     c1 = gmul(C, gmulsg(kross(s4,p),a));
   11303          42 :     c2 = gmul(C2, b);
   11304          42 :     if (e > 1)
   11305             :     {
   11306          14 :       q = r? powuu(p,k2m2): a;
   11307          14 :       if (e == 2) q = gmul(q, uutoQ(p+1,p)); /* special case T_{p^4} */
   11308          14 :       q = gmul(C2, q); /* chi(p^2) [ p^(2k-2) or (p+1)p^(2k-3) ] */
   11309             :     }
   11310          42 :     V = tp2eapply(V, p, p2, e, q, c1, c2);
   11311             :   }
   11312          98 :   return c_deflate(n, d, V);
   11313             : }
   11314             : 
   11315             : static GEN
   11316        1428 : GL2toSL2(GEN g, GEN *abd)
   11317             : {
   11318             :   GEN A, B, C, D, u, v, a, b, d, q;
   11319        1428 :   g = Q_primpart(g);
   11320        1428 :   if (!check_M2Z(g)) pari_err_TYPE("GL2toSL2", g);
   11321        1428 :   A = gcoeff(g,1,1); B = gcoeff(g,1,2);
   11322        1428 :   C = gcoeff(g,2,1); D = gcoeff(g,2,2);
   11323        1428 :   a = bezout(A, C, &u, &v);
   11324        1428 :   if (!equali1(a)) { A = diviiexact(A,a); C = diviiexact(C,a); }
   11325        1428 :   d = subii(mulii(A,D), mulii(B,C));
   11326        1428 :   if (signe(d) <= 0) pari_err_TYPE("GL2toSL2",g);
   11327        1421 :   q = dvmdii(addii(mulii(u,B), mulii(v,D)), d, &b);
   11328        1421 :   *abd = (equali1(a) && equali1(d))? NULL: mkvec3(a, b, d);
   11329        1421 :   return mkmat22(A, subii(mulii(q,A), v), C, addii(mulii(q,C), u));
   11330             : }
   11331             : 
   11332             : static GEN
   11333        8582 : Rg_approx(GEN t, long bit)
   11334             : {
   11335        8582 :   GEN a = real_i(t), b = imag_i(t);
   11336        8582 :   long e1 = gexpo(a), e2 = gexpo(b);
   11337        8582 :   if (e2 < -bit) { t = e1 < -bit? gen_0: a; }
   11338        6510 :   else if (e1 < -bit) t = gmul(b, gen_I());
   11339        8582 :   return t;
   11340             : }
   11341             : static GEN
   11342         126 : RgV_approx(GEN x, long bit)
   11343         840 : { pari_APPLY_same(Rg_approx(gel(x,i), bit)); }
   11344             : /* m != 2 (mod 4), D t_INT; V has "denominator" D, recognize in Q(zeta_m) */
   11345             : static GEN
   11346         126 : bestapprnf2(GEN V, long m, GEN D, long prec)
   11347             : {
   11348         126 :   long i, j, f, vt = fetch_user_var("t"), bit = prec2nbits_mul(prec, 0.8);
   11349         126 :   GEN Tinit, Vl, H, Pf, P = polcyclo(m, vt);
   11350             : 
   11351         126 :   V = liftpol_shallow(V);
   11352         126 :   V = gmul(RgV_approx(V, bit), D);
   11353         126 :   V = bestapprnf(V, P, NULL, prec);
   11354         126 :   Vl = liftpol_shallow(V);
   11355         126 :   H = coprimes_zv(m);
   11356         672 :   for (i = 2; i < m; i++)
   11357             :   {
   11358         546 :     if (H[i] != 1) continue;
   11359         280 :     if (!gequal(Vl, vecGalois(Vl, i, P, m))) H[i] = 0;
   11360          14 :     else for (j = i; j < m; j *= i) H[i] = 3;
   11361             :   }
   11362         126 :   f = znstar_conductor_bits(Flv_to_F2v(H));
   11363         126 :   if (f == 1) return gdiv(V, D);
   11364          98 :   if (f == m) return gmodulo(gdiv(V, D), P);
   11365           7 :   Pf = polcyclo(f, vt);
   11366           7 :   Tinit = Qab_trace_init(m, f, P, Pf);
   11367           7 :   return gmodulo(gdiv(QabV_tracerel(Tinit, 0, Vl), D), Pf);
   11368             : }
   11369             : 
   11370             : /* f | ga expansion; [f, mf_eisendec(f)]~ allowed */
   11371             : GEN
   11372        1365 : mfslashexpansion(GEN mf, GEN f, GEN ga, long n, long flrat, GEN *params, long prec)
   11373             : {
   11374        1365 :   pari_sp av = avma;
   11375        1365 :   GEN a, b, d, res, al, V, M, ad, abd, gk, A, awd = NULL;
   11376             :   long i, w;
   11377             : 
   11378        1365 :   mf = checkMF(mf);
   11379        1365 :   gk = MF_get_gk(mf);
   11380        1365 :   M = GL2toSL2(ga, &abd);
   11381        1358 :   if (abd) { a = gel(abd,1); b = gel(abd,2); d = gel(abd,3); }
   11382         903 :   else { a = d = gen_1; b = gen_0; }
   11383        1358 :   ad = gdiv(a,d);
   11384        1358 :   res = mfgaexpansion(mf, f, M, n, prec);
   11385        1358 :   al = gel(res,1);
   11386        1358 :   w = itou(gel(res,2));
   11387        1358 :   V = gel(res,3);
   11388        1358 :   if (flrat)
   11389             :   {
   11390         126 :     GEN CHI = MF_get_CHI(mf);
   11391         126 :     long N = MF_get_N(mf), F = mfcharconductor(CHI);
   11392         126 :     long ord = mfcharorder(CHI), k, deg;
   11393         126 :     long B = umodiu(gcoeff(M,1,2), N);
   11394         126 :     long C = umodiu(gcoeff(M,2,1), N);
   11395         126 :     long D = umodiu(gcoeff(M,2,2), N);
   11396         126 :     long CD = (C * D) % N, BC = (B * C) % F;
   11397             :     GEN CV, t;
   11398             :     /* weight of f * Theta in 1/2-integral weight */
   11399         126 :     k = typ(gk) == t_INT? (long) itou(gk): MF_get_r(mf)+1;
   11400         126 :     CV = odd(k) ? powuu(N, k - 1) : powuu(N, k >> 1);
   11401         126 :     deg = ulcm(ulcm(ord, N/ugcd(N,CD)), F/ugcd(F,BC));
   11402         126 :     if ((deg & 3) == 2) deg >>= 1;
   11403         126 :     if (typ(gk) != t_INT && odd(deg) && mfthetaI(C,D)) deg <<= 2;
   11404         126 :     V = bestapprnf2(V, deg, CV, prec);
   11405         126 :     if (abd && !signe(b))
   11406             :     { /* can [a,0; 0,d] be simplified to id ? */
   11407           7 :       long nk, dk; Qtoss(gk, &nk, &dk);
   11408           7 :       if (ispower(ad, utoipos(2*dk), &t)) /* t^(2*dk) = a/d or t = NULL */
   11409             :       {
   11410           7 :         V = RgV_Rg_mul(V, powiu(t,nk));
   11411           7 :         awd = gdiv(a, muliu(d,w));
   11412             :       }
   11413             :     }
   11414             :   }
   11415        1232 :   else if (abd)
   11416             :   { /* ga = M * [a,b;0,d] * rational, F := f | M = q^al * \sum V[j] q^(j/w) */
   11417         448 :     GEN u, t = NULL, wd = muliu(d,w);
   11418             :     /* a > 0, 0 <= b < d; f | ga = (a/d)^(k/2) * F(tau + b/d) */
   11419         448 :     if (signe(b))
   11420             :     {
   11421             :       long ns, ds;
   11422             :       GEN z;
   11423           0 :       Qtoss(gdiv(b, wd), &ns, &ds); z = rootsof1powinit(ns, ds, prec);
   11424           0 :       for (i = 1; i <= n+1; i++) gel(V,i) = gmul(gel(V,i), rootsof1pow(z, i-1));
   11425           0 :       if (!gequal0(al)) t = gexp(gmul(PiI2(prec), gmul(al, gdiv(b,d))), prec);
   11426             :     }
   11427         448 :     awd = gdiv(a, wd);
   11428         448 :     u = gpow(ad, gmul2n(gk,-1), prec);
   11429         448 :     t = t? gmul(t, u): u;
   11430         448 :     V = RgV_Rg_mul(V, t);
   11431             :   }
   11432        1358 :   if (!awd) A = mkmat22(a, b, gen_0, d);
   11433             :   else
   11434             :   { /* rescale and update w from [a,0; 0,d] */
   11435             :     long ns;
   11436         455 :     Qtoss(awd, &ns, &w); /* update w */
   11437         455 :     V = bdexpand(V, ns);
   11438         455 :     if (!gequal0(al))
   11439             :     {
   11440           0 :       GEN adal = gmul(ad, al), sh = gfloor(adal);
   11441           0 :       al = gsub(adal, sh);
   11442           0 :       V = RgV_shift(V, sh);
   11443             :     }
   11444         455 :     A = matid(2);
   11445             :   }
   11446        1358 :   if (params) *params = mkvec3(al, utoipos(w), A);
   11447        1358 :   return gc_all(av,params?2:1,&V,params);
   11448             : }
   11449             : 
   11450             : /**************************************************************/
   11451             : /*         Alternative method for 1/2-integral weight         */
   11452             : /**************************************************************/
   11453             : static GEN
   11454         273 : mf2basis(long N, long r, GEN CHI, GEN *pCHI1, long space)
   11455             : {
   11456             :   GEN CHI1, CHI2, mf1, mf2, B1, B2, BT, M1, M2, M, M2i, T, Th, v, den;
   11457         273 :   long sb, N2, o1, o2, k1 = r + 1;
   11458             : 
   11459         273 :   if (odd(k1))
   11460             :   {
   11461         161 :     CHI1 = mfcharmul(CHI, get_mfchar(stoi(-4)));
   11462         161 :     CHI2 = mfcharmul(CHI, get_mfchar(stoi(-8)));
   11463             :   }
   11464             :   else
   11465             :   {
   11466         112 :     CHI1 = CHI;
   11467         112 :     CHI2 = mfcharmul(CHI, get_mfchar(utoi(8)));
   11468             :   }
   11469         273 :   mf1 = mfinit_Nkchi(N, k1, CHI1, space, 1);
   11470         273 :   if (pCHI1) *pCHI1 = CHI1;
   11471         273 :   B1 = MF_get_basis(mf1); if (lg(B1) == 1) return cgetg(1,t_VEC);
   11472         266 :   N2 = ulcm(8, N);
   11473         266 :   mf2 = mfinit_Nkchi(N2, k1, CHI2, space, 1);
   11474         266 :   B2 = MF_get_basis(mf2); if (lg(B2) == 1) return cgetg(1,t_VEC);
   11475         266 :   sb = mfsturmNgk(N2, gaddsg(k1, ghalf));
   11476         266 :   M1 = mfcoefs_mf(mf1, sb, 1);
   11477         266 :   M2 = mfcoefs_mf(mf2, sb, 1);
   11478         266 :   Th = mfTheta(NULL);
   11479         266 :   BT = mfcoefs_i(Th, sb, 1);
   11480         266 :   M1 = mfmatsermul(M1, RgV_to_RgX(expandbd(BT,2),0));
   11481         266 :   M2 = mfmatsermul(M2, RgV_to_RgX(BT,0));
   11482         266 :   o1= mfcharorder(CHI1);
   11483         266 :   T = (o1 <= 2)? NULL: mfcharpol(CHI1);
   11484         266 :   if (o1 > 2) M1 = liftpol_shallow(M1);
   11485         266 :   o2= mfcharorder(CHI2);
   11486         266 :   if (T)
   11487             :   {
   11488          14 :     if (o2 == o1) M2 = liftpol_shallow(M2);
   11489             :     else
   11490             :     {
   11491           0 :       GEN tr = Qab_trace_init(o2, o1, mfcharpol(CHI2), mfcharpol(CHI1));
   11492           0 :       M2 = QabM_tracerel(tr, 0, M2);
   11493             :     }
   11494             :   }
   11495             :   /* now everything is defined mod T = mfcharpol(CHI1) */
   11496         266 :   M2i = QabM_pseudoinv_i(M2, T, o1, &v, &den);
   11497         266 :   M = RgM_mul(M2i, rowpermute(M1, gel(v,1)));
   11498         266 :   M = RgM_mul(M2, M);
   11499         266 :   M1 = RgM_Rg_mul(M1, den);
   11500         266 :   M = RgM_sub(M1, M); if (T) M = RgXQM_red(M, T);
   11501         266 :   return vecmflineardiv0(B1, QabM_ker(M, T, o1), Th);
   11502             : }
   11503             : 
   11504             : /*******************************************************************/
   11505             : /*                         Integration                             */
   11506             : /*******************************************************************/
   11507             : static GEN
   11508         490 : vanembed(GEN F, GEN v, long prec)
   11509             : {
   11510         490 :   GEN CHI = mf_get_CHI(F);
   11511         490 :   long o = mfcharorder(CHI);
   11512         490 :   if (o > 2 || degpol(mf_get_field(F)) > 1) v = liftpol_shallow(v);
   11513         490 :   if (o > 2) v = gsubst(v, varn(mfcharpol(CHI)), rootsof1u_cx(o, prec));
   11514         490 :   return v;
   11515             : }
   11516             : 
   11517             : static long
   11518        1253 : mfperiod_prelim_double(double t0, long k, long bitprec)
   11519             : {
   11520        1253 :   double nlim, c = 2*M_PI*t0;
   11521        1253 :   nlim = ceil(bitprec * M_LN2 / c);
   11522        1253 :   c -= (k - 1)/(2*nlim); if (c < 1) c = 1.;
   11523        1253 :   nlim += ceil((0.7 + (k-1)/2*log(nlim))/c);
   11524        1253 :   return (long)nlim;
   11525             : }
   11526             : static long
   11527         301 : mfperiod_prelim(GEN t0, long k, long bitprec)
   11528         301 : { return mfperiod_prelim_double(gtodouble(t0), k, bitprec); }
   11529             : 
   11530             : /* (-X)^(k-2) * P(-1/X) = (-1)^{k-2} P|_{k-2} S */
   11531             : static GEN
   11532        1288 : RgX_act_S(GEN P, long k)
   11533             : {
   11534        1288 :   P = RgX_unscale(RgX_recipspec_shallow(P+2, lgpol(P), k-1), gen_m1);
   11535        1288 :   setvarn(P,0); return P;
   11536             : }
   11537             : static int
   11538        2842 : RgX_act_typ(GEN P, long k)
   11539             : {
   11540        2842 :   switch(typ(P))
   11541             :   {
   11542          35 :     case t_RFRAC: return t_RFRAC;
   11543        2807 :     case t_POL:
   11544        2807 :       if (varn(P) == 0)
   11545             :       {
   11546        2807 :         long d = degpol(P);
   11547        2807 :         if (d > k-2) return t_RFRAC;
   11548        2653 :         if (d) return t_POL;
   11549             :       }
   11550             :   }
   11551        1211 :   return 0;
   11552             : }
   11553             : static GEN
   11554        2576 : act_S(GEN P, long k)
   11555             : {
   11556             :   GEN X;
   11557        2576 :   switch(RgX_act_typ(P, k))
   11558             :   {
   11559         140 :     case t_RFRAC:
   11560         140 :       X = gneg(pol_x(0));
   11561         140 :       return gmul(gsubst(P, 0, ginv(X)), gpowgs(X, k - 2));
   11562        1288 :     case t_POL: return RgX_act_S(P, k);
   11563             :   }
   11564        1148 :   return P;
   11565             : }
   11566             : 
   11567             : static GEN
   11568         203 : AX_B(GEN M)
   11569         203 : { GEN A = gcoeff(M,1,1), B = gcoeff(M,1,2); return deg1pol_shallow(A,B,0); }
   11570             : static GEN
   11571         203 : CX_D(GEN M)
   11572         203 : { GEN C = gcoeff(M,2,1), D = gcoeff(M,2,2); return deg1pol_shallow(C,D,0); }
   11573             : 
   11574             : /* P|_{2-k}M = (CX+D)^{k-2}P((AX+B)/(CX+D)) */
   11575             : static GEN
   11576         154 : RgX_act_gen(GEN P, GEN M, long k)
   11577             : {
   11578         154 :   GEN S = gen_0, PCD, PAB;
   11579             :   long i;
   11580         154 :   PCD = gpowers(CX_D(M), k-2);
   11581         154 :   PAB = gpowers(AX_B(M), k-2);
   11582         833 :   for (i = 0; i <= k-2; i++)
   11583             :   {
   11584         679 :     GEN t = RgX_coeff(P, i);
   11585         679 :     if (!gequal0(t)) S = gadd(S, gmul(t, gmul(gel(PCD, k-i-1), gel(PAB, i+1))));
   11586             :   }
   11587         154 :   return S;
   11588             : }
   11589             : static GEN
   11590         266 : act_GL2(GEN P, GEN M, long k)
   11591             : {
   11592         266 :   switch(RgX_act_typ(P, k))
   11593             :   {
   11594          49 :     case t_RFRAC:
   11595             :     {
   11596          49 :       GEN AB = AX_B(M), CD = CX_D(M);
   11597          49 :       return gmul(gsubst(P, 0, gdiv(AB, CD)), gpowgs(CD, k - 2));
   11598             :     }
   11599         154 :     case t_POL: return RgX_act_gen(P, M, k);
   11600             :   }
   11601          63 :   return P;
   11602             : }
   11603             : static GEN
   11604           7 : vecact_GL2(GEN x, GEN M, long k)
   11605          21 : { pari_APPLY_same(act_GL2(gel(x,i), M, k)); }
   11606             : 
   11607             : static GEN
   11608        2863 : RgX_approx(GEN x, long bit)
   11609       10731 : { pari_APPLY_pol_normalized(Rg_approx(gel(x,i),bit)); }
   11610             : 
   11611             : static GEN normalizeapprox(GEN x, long bit);
   11612             : static GEN
   11613        2898 : normalizeapprox_i(GEN x, long bit)
   11614             : {
   11615        2898 :   GEN D = gen_1;
   11616        2954 :   if (is_vec_t(typ(x))) pari_APPLY_same(normalizeapprox(gel(x,i), bit));
   11617        2870 :   if (typ(x) == t_RFRAC && varn(gel(x,2)) == 0) { D = gel(x,2); x = gel(x,1); }
   11618        2870 :   if (typ(x) != t_POL || varn(x) != 0) return gdiv(x, D);
   11619        2863 :   return gdiv(RgX_approx(x, bit), D);
   11620             : }
   11621             : static GEN
   11622          56 : normalizeapprox(GEN x, long bit)
   11623             : {
   11624          56 :   pari_sp av = avma;
   11625          56 :   return gc_upto(av, normalizeapprox_i(x, bit));
   11626             : }
   11627             : 
   11628             : /* make sure T is a t_POL in variable 0 */
   11629             : static GEN
   11630        2863 : toRgX0(GEN T)
   11631        2863 : { return typ(T) == t_POL && varn(T) == 0? T: scalarpol_shallow(T,0); }
   11632             : 
   11633             : /* integrate by summing  nlim+1 terms of van [may be < lg(van)]
   11634             :  * van can be an expansion with vector coefficients
   11635             :  * \int_A^oo \sum_n van[n] * q^(n/w + al) * P(z-A) dz, q = e(z) */
   11636             : static GEN
   11637         945 : intAoo(GEN van, long nlim, GEN al, long w, GEN P, GEN A, long k, long prec)
   11638             : {
   11639             :   GEN alw, P1, piI2A, q, S, van0;
   11640         945 :   long n, vz = varn(gel(P,2));
   11641             : 
   11642         945 :   if (nlim < 1) nlim = 1;
   11643         945 :   alw = gmulsg(w, al);
   11644         945 :   P1 = RgX_Rg_translate(P, gneg(A));
   11645         945 :   piI2A = gmul(PiI2n(1, prec), A);
   11646         945 :   q = gexp(gdivgu(piI2A, w), prec);
   11647         945 :   S = gen_0;
   11648      121674 :   for (n = nlim; n >= 1; n--)
   11649             :   {
   11650      120729 :     GEN t = gsubst(P1, vz, gdivsg(w, gaddsg(n, alw)));
   11651      120729 :     S = gadd(gmul(gel(van, n+1), t), gmul(q, S));
   11652             :   }
   11653         945 :   S = gmul(q, S);
   11654         945 :   van0 = gel(van, 1);
   11655         945 :   if (!gequal0(al))
   11656             :   {
   11657          42 :     S = gadd(S, gmul(gsubst(P1, vz, ginv(al)), van0));
   11658          42 :     S = gmul(S, gexp(gmul(piI2A, al), prec));
   11659             :   }
   11660         903 :   else if (!gequal0(van0))
   11661         231 :     S = gsub(S, gdivgu(gmul(van0, gpowgs(gsub(pol_x(0), A), k-1)), k-1));
   11662         945 :   if (is_vec_t(typ(S)))
   11663             :   {
   11664         637 :     long j, l = lg(S);
   11665        3192 :     for (j = 1; j < l; j++) gel(S,j) = toRgX0(gel(S,j));
   11666             :   }
   11667             :   else
   11668         308 :     S = toRgX0(S);
   11669         945 :   return gneg(S);
   11670             : }
   11671             : 
   11672             : /* \sum_{j <= k} X^j * (Y / (2I\pi))^{k+1-j} k! / j! */
   11673             : static GEN
   11674         259 : get_P(long k, long v, long prec)
   11675             : {
   11676         259 :   GEN a, S = cgetg(k + 1, t_POL), u = invr(Pi2n(1, prec+EXTRAPREC64));
   11677         259 :   long j, K = k-2;
   11678         259 :   S[1] = evalsigne(1)|evalvarn(0); a = u;
   11679         259 :   gel(S,K+2) = monomial(mulcxpowIs(a,3), 1, v); /* j = K */
   11680        1176 :   for(j = K-1; j >= 0; j--)
   11681             :   {
   11682         917 :     a = mulrr(mulru(a,j+1), u);
   11683         917 :     gel(S,j+2) = monomial(mulcxpowIs(a,3*(K+1-j)), K+1-j, v);
   11684             :   }
   11685         259 :   return S;
   11686             : }
   11687             : 
   11688             : static GEN
   11689        2555 : getw1w2(long N, GEN ga)
   11690        2555 : { return mkvecsmall2(mfZC_width(N, gel(ga,1)),
   11691        2555 :                      mfZC_width(N, gel(ga,2))); }
   11692             : 
   11693             : static GEN
   11694         147 : intAoowithvanall(GEN mf, GEN vanall, GEN P, GEN cosets, long bitprec)
   11695             : {
   11696         147 :   GEN vvan = gel(vanall,1), vaw = gel(vanall,2), W1W2, resall;
   11697         147 :   long prec = nbits2prec(bitprec), N, k, lco, j;
   11698             : 
   11699         147 :   N = MF_get_N(mf); k = MF_get_k(mf);
   11700         147 :   lco = lg(cosets);
   11701         147 :   W1W2 = cgetg(lco, t_VEC); resall = cgetg(lco, t_VEC);
   11702        2702 :   for (j = 1; j < lco; j++) gel(W1W2,j) = getw1w2(N, gel(cosets, j));
   11703        2702 :   for (j = 1; j < lco; j++)
   11704             :   {
   11705        2555 :     GEN w1w2j = gel(W1W2,j), alj, M, VAN, RES, AR, Q;
   11706             :     long jq, c, w1, w2, w;
   11707        2555 :     if (!w1w2j) continue;
   11708         637 :     alj = gel(vaw,j);
   11709         637 :     w1 = w1w2j[1]; Q = cgetg(lco, t_VECSMALL);
   11710         637 :     w2 = w1w2j[2]; M = cgetg(lco, t_COL);
   11711        8267 :     for (c = 1, jq = j; jq < lco; jq++)
   11712             :     {
   11713        7630 :       GEN W = gel(W1W2, jq);
   11714        7630 :       if (jq == j || (W && gequal(W, w1w2j) && gequal(gel(vaw, jq), alj)))
   11715             :       {
   11716        2555 :         Q[c] = jq; gel(W1W2, jq) = NULL;
   11717        2555 :         gel(M, c) = gel(vvan, jq); c++;
   11718             :       }
   11719             :     }
   11720         637 :     setlg(M,c); VAN = shallowmatconcat(M);
   11721         637 :     AR = mkcomplex(gen_0, sqrtr_abs(divru(utor(w1, prec+EXTRAPREC64), w2)));
   11722         637 :     w = itos(gel(alj,2));
   11723         637 :     RES = intAoo(VAN, lg(VAN)-2, gel(alj,1),w, P, AR, k, prec);
   11724        3192 :     for (jq = 1; jq < c; jq++) gel(resall, Q[jq]) = gel(RES, jq);
   11725             :   }
   11726         147 :   return resall;
   11727             : }
   11728             : 
   11729             : GEN
   11730         539 : mftobasisES(GEN mf, GEN F)
   11731             : {
   11732         539 :   GEN v = mftobasis(mf, F, 0);
   11733         532 :   long nE = lg(MF_get_E(mf))-1;
   11734         532 :   return mkvec2(vecslice(v,1,nE), vecslice(v,nE+1,lg(v)-1));
   11735             : }
   11736             : 
   11737             : static long
   11738           0 : wt1mulcond(GEN F, long D, long space)
   11739             : {
   11740           0 :   GEN E = mfeisenstein_i(1, mfchartrivial(), get_mfchar(stoi(D))), mf;
   11741           0 :   F = mfmul(F, E);
   11742           0 :   mf = mfinit_Nkchi(mf_get_N(F), mf_get_k(F), mf_get_CHI(F), space, 0);
   11743           0 :   return mfconductor(mf, F);
   11744             : }
   11745             : static int
   11746           7 : wt1newlevel(long N)
   11747             : {
   11748           7 :   GEN P = gel(myfactoru(N),1);
   11749           7 :   long l = lg(P), i;
   11750          14 :   for (i = 1; i < l; i++)
   11751           7 :     if (!wt1empty(N/P[i])) return 0;
   11752           7 :   return 1;
   11753             : }
   11754             : long
   11755         175 : mfconductor(GEN mf, GEN F)
   11756             : {
   11757         175 :   pari_sp av = avma;
   11758             :   GEN gk;
   11759             :   long space, N, M;
   11760             : 
   11761         175 :   mf = checkMF(mf);
   11762         175 :   if (!checkmf_i(F)) pari_err_TYPE("mfconductor",F);
   11763         175 :   if (mfistrivial(F)) return 1;
   11764         175 :   space = MF_get_space(mf);
   11765         175 :   if (space == mf_NEW) return mf_get_N(F);
   11766         175 :   gk = MF_get_gk(mf);
   11767         175 :   if (isint1(gk))
   11768             :   {
   11769           7 :     N = mf_get_N(F);
   11770           7 :     if (!wt1newlevel(N))
   11771             :     {
   11772           0 :       long s = space_is_cusp(space)? mf_CUSP: mf_FULL;
   11773           0 :       N = ugcd(N, wt1mulcond(F,-3,s));
   11774           0 :       if (!wt1newlevel(N)) N = ugcd(N, wt1mulcond(F,-4,s));
   11775             :     }
   11776           7 :     return gc_long(av,N);
   11777             :   }
   11778         168 :   if (typ(gk) != t_INT)
   11779             :   {
   11780          42 :     F = mfmultheta(F);
   11781          42 :     mf = obj_checkbuild(mf, MF_MF2INIT, &mf2init); /* mf_FULL */
   11782             :   }
   11783         168 :   N = 1;
   11784         168 :   if (space_is_cusp(space))
   11785             :   {
   11786           7 :     F = mftobasis_i(mf, F);
   11787           7 :     if (typ(gk) != t_INT) F = vecslice(F, lg(MF_get_E(mf)), lg(F) - 1);
   11788             :   }
   11789             :   else
   11790             :   {
   11791         161 :     GEN EF = mftobasisES(mf, F), vE = gel(EF,1), B = MF_get_E(mf);
   11792         161 :     long i, l = lg(B);
   11793        1267 :     for (i = 1; i < l; i++)
   11794        1106 :       if (!gequal0(gel(vE,i))) N = ulcm(N, mf_get_N(gel(B, i)));
   11795         161 :     F = gel(EF,2);
   11796             :   }
   11797         168 :   (void)mftonew_i(mf, F, &M); /* M = conductor of cuspidal part */
   11798         168 :   return gc_long(av, ulcm(M, N));
   11799             : }
   11800             : 
   11801             : static GEN
   11802        1463 : fs_get_MF(GEN fs) { return gel(fs,1); }
   11803             : static GEN
   11804         847 : fs_get_vES(GEN fs) { return gel(fs,2); }
   11805             : static GEN
   11806        1596 : fs_get_pols(GEN fs) { return gel(fs,3); }
   11807             : static GEN
   11808        2191 : fs_get_cosets(GEN fs) { return gel(fs,4); }
   11809             : static long
   11810         630 : fs_get_bitprec(GEN fs) { return itou(gel(fs,5)); }
   11811             : static GEN
   11812        1246 : fs_get_vE(GEN fs) { return gel(fs,6); }
   11813             : static GEN
   11814          70 : fs_get_EF(GEN fs) { return gel(fs,7); }
   11815             : static GEN
   11816        1890 : fs_get_expan(GEN fs) { return gel(fs,8); }
   11817             : static GEN
   11818          28 : fs_set_expan(GEN fs, GEN vanall)
   11819          28 : { GEN f = shallowcopy(fs); gel(f,8) = vanall; return f; }
   11820             : static int
   11821          49 : mfs_checkmf(GEN fs, GEN mf)
   11822          49 : { GEN mfF = fs_get_MF(fs); return gequal(gel(mfF,1), gel(mf,1)); }
   11823             : static long
   11824         798 : checkfs_i(GEN v)
   11825         798 : { return typ(v) == t_VEC && lg(v) == 9 && checkMF_i(fs_get_MF(v))
   11826         567 :     && typ(fs_get_vES(v)) == t_VEC
   11827         567 :     && typ(fs_get_pols(v)) == t_VEC
   11828         567 :     && typ(fs_get_cosets(v)) == t_VEC
   11829         567 :     && typ(fs_get_vE(v)) == t_VEC
   11830         567 :     && lg(fs_get_pols(v)) == lg(fs_get_cosets(v))
   11831         567 :     && typ(fs_get_expan(v)) == t_VEC
   11832         567 :     && lg(fs_get_expan(v)) == 3
   11833         567 :     && lg(gel(fs_get_expan(v), 1)) == lg(fs_get_cosets(v))
   11834        1596 :     && typ(gel(v,5)) == t_INT; }
   11835             : GEN
   11836       19292 : checkMF_i(GEN mf)
   11837             : {
   11838       19292 :   long l = lg(mf);
   11839             :   GEN v;
   11840       19292 :   if (typ(mf) != t_VEC) return NULL;
   11841       19264 :   if (l == 9) return checkMF_i(fs_get_MF(mf));
   11842       19264 :   if (l != 7) return NULL;
   11843        7980 :   v = gel(mf,1);
   11844        7980 :   if (typ(v) != t_VEC || lg(v) != 5) return NULL;
   11845        7980 :   return (typ(gel(v,1)) == t_INT
   11846        7980 :          && typ(gmul2n(gel(v,2), 1)) == t_INT
   11847        7980 :          && typ(gel(v,3)) == t_VEC
   11848       15960 :          && typ(gel(v,4)) == t_INT)? mf: NULL; }
   11849             : GEN
   11850        4228 : checkMF(GEN T)
   11851             : {
   11852        4228 :   GEN mf = checkMF_i(T);
   11853        4228 :   if (!mf) pari_err_TYPE("checkMF [please use mfinit]", T);
   11854        4228 :   return mf;
   11855             : }
   11856             : 
   11857             : /* c,d >= 0; c * Nc = N, find coset whose image in P1(Z/NZ) ~ (c, d + k(N/c)) */
   11858             : static GEN
   11859       11963 : coset_complete(long c, long d, long Nc)
   11860             : {
   11861             :   long a, b;
   11862       13307 :   while (ugcd(c, d) > 1) d += Nc;
   11863       11963 :   (void)cbezout(c, d, &b, &a);
   11864       11963 :   return mkmat22s(a, -b, c, d);
   11865             : }
   11866             : /* right cosets of $\G_0(N)$: $\G=\bigsqcup_j \G_0(N)\ga_j$. */
   11867             : /* We choose them with c\mid N and d mod N/c, not the reverse */
   11868             : GEN
   11869         168 : mfcosets(GEN gN)
   11870             : {
   11871         168 :   pari_sp av = avma;
   11872             :   GEN V, D, mf;
   11873         168 :   long l, i, ct, N = 0;
   11874         168 :   if (typ(gN) == t_INT) N = itos(gN);
   11875          14 :   else if ((mf = checkMF_i(gN))) N = MF_get_N(mf);
   11876           7 :   else pari_err_TYPE("mfcosets", gN);
   11877         161 :   if (N <= 0) pari_err_DOMAIN("mfcosets", "N", "<=", gen_0, stoi(N));
   11878         161 :   V = cgetg(mypsiu(N) + 1, t_VEC);
   11879         161 :   D = mydivisorsu(N); l = lg(D);
   11880         588 :   for (i = ct = 1; i < l; i++)
   11881             :   {
   11882         427 :     long d, c = D[i], Nc = D[l-i], e = ugcd(Nc, c);
   11883        3332 :     for (d = 0; d < Nc; d++)
   11884        2905 :       if (ugcd(d,e) == 1) gel(V, ct++) = coset_complete(c, d, Nc);
   11885             :   }
   11886         161 :   return gc_GEN(av, V);
   11887             : }
   11888             : static int
   11889       35469 : cmp_coset(void *E, GEN A, GEN B)
   11890             : {
   11891       35469 :   ulong N = (ulong)E, Nc, c = itou(gcoeff(A,2,1));
   11892       35469 :   int r = cmpuu(c, itou(gcoeff(B,2,1)));
   11893       35469 :   if (r) return r;
   11894       30660 :   Nc = N / c;
   11895       30660 :   return cmpuu(umodiu(gcoeff(A,2,2), Nc), umodiu(gcoeff(B,2,2), Nc));
   11896             : }
   11897             : /* M in SL_2(Z) */
   11898             : static long
   11899        9198 : mftocoset_i(ulong N, GEN M, GEN cosets)
   11900             : {
   11901        9198 :   pari_sp av = avma;
   11902        9198 :   long A = itos(gcoeff(M,1,1)), c, u, v, Nc, i;
   11903        9198 :   long C = itos(gcoeff(M,2,1)), D = itos(gcoeff(M,2,2));
   11904             :   GEN ga;
   11905        9198 :   c = cbezout(N*A, C, &u, &v); Nc = N/c;
   11906        9198 :   ga = coset_complete(c, umodsu(v*D, Nc), Nc);
   11907        9198 :   i = gen_search(cosets, ga, (void*)N, &cmp_coset);
   11908        9198 :   if (i < 0) pari_err_BUG("mftocoset [no coset found]");
   11909        9198 :   return gc_long(av,i);
   11910             : }
   11911             : /* (U * V^(-1))[2,2] mod N, assuming V in SL2(Z) */
   11912             : static long
   11913        9177 : SL2_div_D(ulong N, GEN U, GEN V)
   11914             : {
   11915        9177 :   long c = umodiu(gcoeff(U,2,1), N), d = umodiu(gcoeff(U,2,2), N);
   11916        9177 :   long a2 = umodiu(gcoeff(V,1,1), N), b2 = umodiu(gcoeff(V,1,2), N);
   11917        9177 :   return (a2*d - b2*c) % (long)N;
   11918             : }
   11919             : static long
   11920        9177 : mftocoset_iD(ulong N, GEN M, GEN cosets, long *D)
   11921             : {
   11922        9177 :   long i = mftocoset_i(N, M, cosets);
   11923        9177 :   *D = SL2_div_D(N, M, gel(cosets,i)); return i;
   11924             : }
   11925             : GEN
   11926           7 : mftocoset(ulong N, GEN M, GEN cosets)
   11927             : {
   11928             :   long i;
   11929           7 :   if (!check_SL2Z(M)) pari_err_TYPE("mftocoset",M);
   11930           7 :   i = mftocoset_i(N, M, cosets);
   11931           7 :   retmkvec2(gdiv(M,gel(cosets,i)), utoipos(i));
   11932             : }
   11933             : 
   11934             : static long
   11935        2555 : getnlim2(long N, long w1, long w2, long nlim, long k, long bitprec)
   11936             : {
   11937        2555 :   if (w2 == N) return nlim;
   11938         483 :   return mfperiod_prelim_double(1./sqrt((double)w1*w2), k, bitprec + 32);
   11939             : }
   11940             : 
   11941             : /* g * S, g 2x2 */
   11942             : static GEN
   11943        1337 : ZM_mulS(GEN g)
   11944        1337 : { return mkmat2(gel(g,2), ZC_neg(gel(g,1))); }
   11945             : /* g * T, g 2x2 */
   11946             : static GEN
   11947        4634 : ZM_mulT(GEN g)
   11948        4634 : { return mkmat2(gel(g,1), ZC_add(gel(g,2), gel(g,1))); }
   11949             : /* g * T^(-1), g 2x2 */
   11950             : static GEN
   11951        2352 : ZM_mulTi(GEN g)
   11952        2352 : { return mkmat2(gel(g,1), ZC_sub(gel(g,2), gel(g,1))); }
   11953             : 
   11954             : /* Compute all slashexpansions for all cosets */
   11955             : static GEN
   11956         175 : mfgaexpansionall(GEN mf, GEN FE, GEN cosets, double height, long prec)
   11957             : {
   11958         175 :   GEN CHI = MF_get_CHI(mf), vres, vresaw;
   11959         175 :   long lco, j, k = MF_get_k(mf), N = MF_get_N(mf), bitprec = prec2nbits(prec);
   11960             : 
   11961         175 :   lco = lg(cosets);
   11962         175 :   vres = const_vec(lco-1, NULL);
   11963         175 :   vresaw = cgetg(lco, t_VEC);
   11964        2912 :   for (j = 1; j < lco; j++) if (!gel(vres,j))
   11965             :   {
   11966         455 :     GEN ga = gel(cosets, j), van, aw, al, z, gai;
   11967         455 :     long w1 = mfZC_width(N, gel(ga,1));
   11968         455 :     long w2 = mfZC_width(N, gel(ga,2));
   11969             :     long nlim, nlim2, daw, da, na, i;
   11970         455 :     double sqNinvdbl = height ? height/w1 : 1./sqrt((double)w1*N);
   11971         455 :     nlim = mfperiod_prelim_double(sqNinvdbl, k, bitprec + 32);
   11972         455 :     van = mfslashexpansion(mf, FE, ga, nlim, 0, &aw, prec + EXTRAPREC64);
   11973         455 :     van = vanembed(gel(FE, 1), van, prec + EXTRAPREC64);
   11974         455 :     al = gel(aw, 1);
   11975         455 :     nlim2 = height? nlim: getnlim2(N, w1, w2, nlim, k, bitprec);
   11976         455 :     gel(vres, j) = vecslice(van, 1, nlim2 + 1);
   11977         455 :     gel(vresaw, j) = aw;
   11978         455 :     Qtoss(al, &na, &da); daw = da*w1;
   11979         455 :     z = rootsof1powinit(1, daw, prec + EXTRAPREC64);
   11980         455 :     gai = ga;
   11981        2737 :     for (i = 1; i < w1; i++)
   11982             :     {
   11983             :       GEN V, coe;
   11984        2282 :       long Di, n, ind, w2, s = ((i*na) % da) * w1, t = i*da;
   11985        2282 :       gai = ZM_mulT(gai);
   11986        2282 :       ind = mftocoset_iD(N, gai, cosets, &Di);
   11987        2282 :       w2 = mfZC_width(N, gel(gel(cosets,ind), 2));
   11988        2282 :       nlim2 = height? nlim: getnlim2(N, w1, w2, nlim, k, bitprec);
   11989        2282 :       gel(vresaw, ind) = aw;
   11990        2282 :       V = cgetg(nlim2 + 2, t_VEC);
   11991      909034 :       for (n = 0; n <= nlim2; n++, s = Fl_add(s, t, daw))
   11992      906752 :         gel(V, n+1) = gmul(gel(van, n+1), rootsof1pow(z, s));
   11993        2282 :       coe = mfcharcxeval(CHI, Di, prec + EXTRAPREC64);
   11994        2282 :       if (!gequal1(coe)) V = RgV_Rg_mul(V, conj_i(coe));
   11995        2282 :       gel(vres, ind) = V;
   11996             :     }
   11997             :   }
   11998         175 :   return mkvec2(vres, vresaw);
   11999             : }
   12000             : 
   12001             : /* Compute all period pols of F|_k\ga_j, vF = mftobasis(F_S) */
   12002             : static GEN
   12003         168 : mfperiodpols_i(GEN mf, GEN FE, GEN cosets, GEN *pvan, long bit)
   12004             : {
   12005         168 :   long N, i, prec = nbits2prec(bit), k = MF_get_k(mf);
   12006         168 :   GEN vP, P, CHI, intall = gen_0;
   12007             : 
   12008         168 :   *pvan = gen_0;
   12009         168 :   if (k == 0 && gequal0(gel(FE,2)))
   12010           0 :     return cosets? const_vec(lg(cosets)-1, pol_0(0)): pol_0(0);
   12011         168 :   N = MF_get_N(mf);
   12012         168 :   CHI = MF_get_CHI(mf);
   12013         168 :   P = get_P(k, fetch_var(), prec);
   12014         168 :   if (!cosets)
   12015             :   { /* ga = id */
   12016          21 :     long nlim, PREC = prec + EXTRAPREC64;
   12017          21 :     GEN F = gel(FE,1), sqNinv = invr(sqrtr_abs(utor(N, PREC))); /* A/w */
   12018             :     GEN AR, v, van, T1, T2;
   12019             : 
   12020          21 :     nlim = mfperiod_prelim(sqNinv, k, bit + 32);
   12021             :     /* F|id: al = 0, w = 1 */
   12022          21 :     v = mfcoefs_i(F, nlim, 1);
   12023          21 :     van = vanembed(F, v, PREC);
   12024          21 :     AR = mkcomplex(gen_0, sqNinv);
   12025          21 :     T1 = intAoo(van, nlim, gen_0,1, P, AR, k, prec);
   12026          21 :     if (N == 1) T2 = T1;
   12027             :     else
   12028             :     { /* F|S: al = 0, w = N */
   12029           7 :       v = mfgaexpansion(mf, FE, mkS(), nlim, PREC);
   12030           7 :       van = vanembed(F, gel(v,3), PREC);
   12031           7 :       AR = mkcomplex(gen_0, mulur(N,sqNinv));
   12032           7 :       T2 = intAoo(van, nlim, gen_0,N, P, AR, k, prec);
   12033             :     }
   12034          21 :     T1 = gsub(T1, act_S(T2, k));
   12035          21 :     T1 = normalizeapprox_i(T1, bit-20);
   12036          21 :     vP = gprec_wtrunc(T1, prec);
   12037             :   }
   12038             :   else
   12039             :   {
   12040         147 :     long lco = lg(cosets);
   12041         147 :     GEN vanall = mfgaexpansionall(mf, FE, cosets, 0, prec);
   12042         147 :     *pvan = vanall;
   12043         147 :     intall = intAoowithvanall(mf, vanall, P, cosets, bit);
   12044         147 :     vP = const_vec(lco-1, NULL);
   12045        2702 :     for (i = 1; i < lco; i++)
   12046             :     {
   12047        2555 :       GEN P, P1, P2, c, ga = gel(cosets, i);
   12048             :       long iS, DS;
   12049        2646 :       if (gel(vP,i)) continue;
   12050        1323 :       P1 = gel(intall, i);
   12051        1323 :       iS = mftocoset_iD(N, ZM_mulS(ga), cosets, &DS);
   12052        1323 :       c = mfcharcxeval(CHI, DS, prec + EXTRAPREC64);
   12053        1323 :       P2 = gel(intall, iS);
   12054             : 
   12055        1323 :       P = act_S(isint1(c)? P2: gmul(c, P2), k);
   12056        1323 :       P = normalizeapprox_i(gsub(P1, P), bit-20);
   12057        1323 :       gel(vP,i) = gprec_wtrunc(P, prec);
   12058        1323 :       if (iS == i) continue;
   12059             : 
   12060        1232 :       P = act_S(isint1(c)? P1: gmul(conj_i(c), P1), k);
   12061        1232 :       if (!odd(k)) P = gneg(P);
   12062        1232 :       P = normalizeapprox_i(gadd(P, P2), bit-20);
   12063        1232 :       gel(vP,iS) = gprec_wtrunc(P, prec);
   12064             :     }
   12065             :   }
   12066         168 :   delete_var(); return vP;
   12067             : }
   12068             : 
   12069             : /* when cosets = NULL, return a "fake" symbol containing only fs(oo->0) */
   12070             : static GEN
   12071         168 : mfsymbol_i(GEN mf, GEN F, GEN cosets, long bit)
   12072             : {
   12073         168 :   GEN FE, van, vP, vE, Mvecj, vES = mftobasisES(mf,F);
   12074         168 :   long precnew, prec = nbits2prec(bit), k = MF_get_k(mf);
   12075         168 :   vE = mfgetembed(F, prec);
   12076         168 :   Mvecj = obj_checkbuild(mf, MF_EISENSPACE, &mfeisensteinspaceinit);
   12077         168 :   if (lg(Mvecj) >= 5) precnew = prec;
   12078             :   else
   12079             :   {
   12080          14 :     long N = MF_get_N(mf), n = mfperiod_prelim_double(1/(double)N, k, bit + 32);
   12081          14 :     precnew = prec + inveis_extraprec(N, mkS(), Mvecj, n);
   12082             :   }
   12083         168 :   FE = mkcol2(F, mf_eisendec(mf,F,precnew));
   12084         168 :   vP = mfperiodpols_i(mf, FE, cosets, &van, bit);
   12085         168 :   return mkvecn(8, mf, vES, vP, cosets, utoi(bit), vE, FE, van);
   12086             : }
   12087             : 
   12088             : static GEN
   12089          56 : fs2_get_cusps(GEN f) { return gel(f,3); }
   12090             : static GEN
   12091          56 : fs2_get_MF(GEN f) { return gel(f,1); }
   12092             : static GEN
   12093          56 : fs2_get_W(GEN f) { return gel(f,2); }
   12094             : static GEN
   12095          56 : fs2_get_F(GEN f) { return gel(f,4); }
   12096             : static long
   12097           0 : fs2_get_bitprec(GEN f) { return itou(gel(f,5)); }
   12098             : static GEN
   12099          56 : fs2_get_al0(GEN f) { return gel(f,6); }
   12100             : static GEN
   12101          21 : fs2_get_den(GEN f) { return gel(f,7); }
   12102             : static int
   12103         210 : checkfs2_i(GEN f)
   12104             : {
   12105             :   GEN W, C, F, al0;
   12106             :   long l;
   12107         210 :   if (typ(f) != t_VEC || lg(f) != 8 || typ(gel(f,5)) != t_INT) return 0;
   12108          35 :   C = fs2_get_cusps(f); l = lg(C);
   12109          35 :   W = fs2_get_W(f);
   12110          35 :   F = fs2_get_F(f);
   12111          35 :   al0 = fs2_get_al0(f);
   12112          35 :   return checkMF_i(fs2_get_MF(f))
   12113          35 :       && typ(W) == t_VEC && typ(F) == t_VEC && typ(al0) == t_VECSMALL
   12114          70 :       && lg(W) == l && lg(F) == l && lg(al0) == l;
   12115             : }
   12116             : static GEN fs2_init(GEN mf, GEN F, long bit);
   12117             : GEN
   12118         175 : mfsymbol(GEN mf, GEN F, long bit)
   12119             : {
   12120         175 :   pari_sp av = avma;
   12121         175 :   GEN cosets = NULL;
   12122         175 :   if (!F)
   12123             :   {
   12124          35 :     F = mf;
   12125          35 :     if (!checkmf_i(F)) pari_err_TYPE("mfsymbol", F);
   12126          35 :     mf = mfinit_i(F, mf_FULL);
   12127             :   }
   12128         140 :   else if (!checkmf_i(F)) pari_err_TYPE("mfsymbol", F);
   12129         175 :   if (checkfs2_i(mf)) return fs2_init(mf, F, bit);
   12130         175 :   if (checkfs_i(mf))
   12131             :   {
   12132           0 :     cosets = fs_get_cosets(mf);
   12133           0 :     mf = fs_get_MF(mf);
   12134             :   }
   12135         175 :   else if (checkMF_i(mf))
   12136             :   {
   12137         175 :     GEN gk = MF_get_gk(mf);
   12138         175 :     if (typ(gk) != t_INT || equali1(gk)) return fs2_init(mf, F, bit);
   12139         154 :     if (signe(gk) <= 0) pari_err_TYPE("mfsymbol [k <= 0]", mf);
   12140         147 :     cosets = mfcosets(MF_get_gN(mf));
   12141             :   }
   12142           0 :   else pari_err_TYPE("mfsymbol",mf);
   12143         147 :   return gc_GEN(av, mfsymbol_i(mf, F, cosets, bit));
   12144             : }
   12145             : 
   12146             : static GEN
   12147          14 : RgX_by_parity(GEN P, long odd)
   12148             : {
   12149          14 :   long i, l = lg(P);
   12150             :   GEN Q;
   12151          14 :   if (l < 4) return odd ? pol_x(0): P;
   12152          14 :   Q = cgetg(l, t_POL); Q[1] = P[1];
   12153          91 :   for (i = odd? 2: 3; i < l; i += 2) gel(Q,i) = gen_0;
   12154          91 :   for (i = odd? 3: 2; i < l; i += 2) gel(Q,i) = gel(P,i);
   12155          14 :   return normalizepol_lg(Q, l);
   12156             : }
   12157             : /* flag 0: period polynomial of F, >0 or <0 with corresponding parity */
   12158             : GEN
   12159          35 : mfperiodpol(GEN mf0, GEN F, long flag, long bit)
   12160             : {
   12161          35 :   pari_sp av = avma;
   12162          35 :   GEN pol, mf = checkMF_i(mf0);
   12163          35 :   if (!mf) pari_err_TYPE("mfperiodpol",mf0);
   12164          35 :   if (checkfs_i(F))
   12165             :   {
   12166          14 :     GEN mfpols = fs_get_pols(F);
   12167          14 :     if (!mfs_checkmf(F, mf)) pari_err_TYPE("mfperiodpol [different mf]",F);
   12168          14 :     pol = veclast(mfpols); /* trivial coset is last */
   12169             :   }
   12170             :   else
   12171             :   {
   12172          21 :     GEN gk = MF_get_gk(mf);
   12173          21 :     if (typ(gk) != t_INT) pari_err_TYPE("mfperiodpol [half-integral k]", mf);
   12174          21 :     if (equali1(gk)) pari_err_TYPE("mfperiodpol [k = 1]", mf);
   12175          21 :     F = mfsymbol_i(mf, F, NULL, bit);
   12176          21 :     pol = fs_get_pols(F);
   12177             :   }
   12178          35 :   if (flag) pol = RgX_by_parity(pol, flag < 0);
   12179          35 :   return gc_GEN(av, RgX_embedall(pol, fs_get_vE(F)));
   12180             : }
   12181             : 
   12182             : static int
   12183          35 : mfs_iscusp(GEN mfs) { return gequal0(gmael(mfs,2,1)); }
   12184             : /* given cusps s1 and s2 (rationals or oo)
   12185             :  * compute $\int_{s1}^{s2}(X-\tau)^{k-2}F|_k\ga_j(\tau)\,d\tau$ */
   12186             : /* If flag = 1, do not give an error message if divergent, but
   12187             :    give the rational function as result. */
   12188             : 
   12189             : static GEN
   12190         126 : col2cusp(GEN v)
   12191             : {
   12192             :   GEN A, C;
   12193         126 :   if (lg(v) != 3 || !RgV_is_ZV(v)) pari_err_TYPE("col2cusp",v);
   12194         126 :   A = gel(v,1);
   12195         126 :   C = gel(v,2);
   12196         126 :   if (gequal0(C))
   12197             :   {
   12198           0 :     if (gequal0(A)) pari_err_TYPE("mfsymboleval", mkvec2(A, C));
   12199           0 :     return mkoo();
   12200             :   }
   12201         126 :   return gdiv(A, C);
   12202             : }
   12203             : /* g.oo */
   12204             : static GEN
   12205         112 : mat2cusp(GEN g) { return col2cusp(gel(g,1)); }
   12206             : 
   12207             : static GEN
   12208           7 : pathmattovec(GEN path)
   12209           7 : { return mkvec2(col2cusp(gel(path,1)), col2cusp(gel(path,2))); }
   12210             : 
   12211             : static void
   12212         546 : get_mf_F(GEN fs, GEN *mf, GEN *F)
   12213             : {
   12214         546 :   if (lg(fs) == 3) { *mf = gel(fs,1); *F = gel(fs,2); }
   12215         546 :   else { *mf = fs_get_MF(fs); *F = NULL; }
   12216         546 : }
   12217             : static GEN
   12218         189 : mfgetvan(GEN fs, GEN ga, GEN *pal, long nlim, long prec)
   12219             : {
   12220             :   GEN van, mf, F, W;
   12221             :   long PREC;
   12222         189 :   get_mf_F(fs, &mf, &F);
   12223         189 :   if (!F)
   12224             :   {
   12225         189 :     GEN vanall = fs_get_expan(fs), cosets = fs_get_cosets(fs);
   12226         189 :     long D, jga = mftocoset_iD(MF_get_N(mf), ga, cosets, &D);
   12227         189 :     van = gmael(vanall, 1, jga);
   12228         189 :     W   = gmael(vanall, 2, jga);
   12229         189 :     if (lg(van) >= nlim + 2)
   12230             :     {
   12231         182 :       GEN z = mfcharcxeval(MF_get_CHI(mf), D, prec);
   12232         182 :       if (!gequal1(z)) van = RgV_Rg_mul(van, z);
   12233         182 :       *pal = gel(W,1); return van;
   12234             :     }
   12235           7 :     F = gel(fs_get_EF(fs), 1);
   12236             :   }
   12237           7 :   PREC = prec + EXTRAPREC64;
   12238           7 :   van = mfslashexpansion(mf, F, ga, nlim, 0, &W, PREC);
   12239           7 :   van = vanembed(F, van, PREC);
   12240           7 :   *pal = gel(W,1); return van;
   12241             : }
   12242             : /* Computation of int_A^oo (f | ga)(t)(X-t)^{k-2} dt, assuming convergence;
   12243             :  * fs is either a symbol or a triple [mf,F,bitprec]. A != oo and im(A) > 0 */
   12244             : static GEN
   12245          77 : intAoo0(GEN fs, GEN A, GEN ga, GEN P, long bit)
   12246             : {
   12247          77 :   long nlim, N, k, w, prec = nbits2prec(bit);
   12248             :   GEN van, mf, F, al;
   12249          77 :   get_mf_F(fs, &mf,&F); N = MF_get_N(mf); k = MF_get_k(mf);
   12250          77 :   w = mfZC_width(N, gel(ga,1));
   12251          77 :   nlim = mfperiod_prelim(gdivgu(imag_i(A), w), k, bit + 32);
   12252          77 :   van = mfgetvan(fs, ga, &al, nlim, prec);
   12253          77 :   return intAoo(van, nlim, al,w, P, A, k, prec);
   12254             : }
   12255             : 
   12256             : /* fs symbol, naive summation, A != oo, im(A) > 0 and B = oo or im(B) > 0 */
   12257             : static GEN
   12258         112 : mfsymboleval_direct(GEN fs, GEN path, GEN ga, GEN P)
   12259             : {
   12260         112 :   GEN A, B, van, S, al, mf = fs_get_MF(fs);
   12261         112 :   long w, nlimA, nlimB = 0, N = MF_get_N(mf), k = MF_get_k(mf);
   12262         112 :   long bit = fs_get_bitprec(fs), prec = nbits2prec(bit);
   12263             : 
   12264         112 :   A = gel(path, 1);
   12265         112 :   B = gel(path, 2); if (typ(B) == t_INFINITY) B = NULL;
   12266         112 :   w = mfZC_width(N, gel(ga,1));
   12267         112 :   nlimA = mfperiod_prelim(gdivgu(imag_i(A),w), k, bit + 32);
   12268         112 :   if (B) nlimB = mfperiod_prelim(gdivgu(imag_i(B),w), k, bit + 32);
   12269         112 :   van = mfgetvan(fs, ga, &al, maxss(nlimA,nlimB), prec);
   12270         112 :   S = intAoo(van, nlimA, al,w, P, A, k, prec);
   12271         112 :   if (B) S = gsub(S, intAoo(van, nlimB, al,w, P, B, k, prec));
   12272         112 :   return RgX_embedall(S, fs_get_vE(fs));
   12273             : }
   12274             : 
   12275             : /* Computation of int_A^oo (f | ga)(t)(X-t)^{k-2} dt, assuming convergence;
   12276             :  * fs is either a symbol or a pair [mf,F]. */
   12277             : static GEN
   12278          77 : mfsymbolevalpartial(GEN fs, GEN A, GEN ga, long bit)
   12279             : {
   12280             :   GEN Y, F, S, P, mf;
   12281          77 :   long N, k, w, prec = nbits2prec(bit);
   12282             : 
   12283          77 :   get_mf_F(fs, &mf, &F);
   12284          77 :   N = MF_get_N(mf); w = mfZC_width(N, gel(ga,1));
   12285          77 :   k = MF_get_k(mf);
   12286          77 :   Y = gdivgu(imag_i(A), w);
   12287          77 :   P = get_P(k, fetch_var(), prec);
   12288          77 :   if (lg(fs) != 3 && gtodouble(Y)*(2*N) < 1)
   12289          21 :   { /* true symbol + low imaginary part: use GL_2 action to improve */
   12290          21 :     GEN U, ga2, czd, A2 = cxredga0N(N, A, &U, &czd, 1);
   12291          21 :     GEN vE = fs_get_vE(fs);
   12292          21 :     ga2 = ZM_mul(ga, ZM_inv(U, NULL));
   12293          21 :     S = RgX_embedall(intAoo0(fs, A2, ga2, P, bit), vE);
   12294          21 :     S = gsub(S, mfsymboleval(fs, mkvec2(mat2cusp(U), mkoo()), ga2, bit));
   12295          21 :     S = typ(S) == t_VEC? vecact_GL2(S, U, k): act_GL2(S, U, k);
   12296             :   }
   12297             :   else
   12298             :   {
   12299          56 :     S = intAoo0(fs, A, ga, P, bit);
   12300          56 :     S = RgX_embedall(S, F? mfgetembed(F,prec): fs_get_vE(fs));
   12301             :   }
   12302          77 :   delete_var(); return normalizeapprox_i(S, bit-20);
   12303             : }
   12304             : 
   12305             : static GEN
   12306          42 : actal(GEN x, GEN vabd)
   12307             : {
   12308          42 :   if (typ(x) == t_INFINITY) return x;
   12309          35 :   return gdiv(gadd(gmul(gel(vabd,1), x), gel(vabd,2)), gel(vabd,3));
   12310             : }
   12311             : 
   12312             : static GEN
   12313          14 : unact(GEN z, GEN vabd, long k, long prec)
   12314             : {
   12315          14 :   GEN res = gsubst(z, 0, actal(pol_x(0), vabd));
   12316          14 :   GEN CO = gpow(gdiv(gel(vabd,3), gel(vabd,1)), sstoQ(k-2, 2), prec);
   12317          14 :   return gmul(CO, res);
   12318             : }
   12319             : 
   12320             : GEN
   12321         210 : mfsymboleval(GEN fs, GEN path, GEN ga, long bitprec)
   12322             : {
   12323         210 :   pari_sp av = avma;
   12324         210 :   GEN tau, V, LM, S, CHI, mfpols, cosets, al, be, mf, F, vabd = NULL;
   12325             :   long D, B, m, u, v, a, b, c, d, j, k, N, prec, tsc1, tsc2;
   12326             : 
   12327         210 :   if (checkfs_i(fs))
   12328             :   {
   12329         203 :     get_mf_F(fs, &mf, &F);
   12330         203 :     bitprec = minss(bitprec, fs_get_bitprec(fs));
   12331             :   }
   12332             :   else
   12333             :   {
   12334           7 :     if (checkfs2_i(fs)) pari_err_TYPE("mfsymboleval [need integral k > 1]",fs);
   12335           0 :     if (typ(fs) != t_VEC || lg(fs) != 3) pari_err_TYPE("mfsymboleval",fs);
   12336           0 :     get_mf_F(fs, &mf, &F);
   12337           0 :     mf = checkMF_i(mf);
   12338           0 :     if (!mf ||!checkmf_i(F)) pari_err_TYPE("mfsymboleval",fs);
   12339             :   }
   12340         203 :   if (lg(path) != 3) pari_err_TYPE("mfsymboleval",path);
   12341         203 :   if (typ(path) == t_MAT) path = pathmattovec(path);
   12342         203 :   if (typ(path) != t_VEC) pari_err_TYPE("mfsymboleval",path);
   12343         203 :   al = gel(path,1);
   12344         203 :   be = gel(path,2);
   12345         203 :   ga = ga? GL2toSL2(ga, &vabd): matid(2);
   12346         203 :   if (vabd)
   12347             :   {
   12348          14 :     al = actal(al, vabd);
   12349          14 :     be = actal(be, vabd); path = mkvec2(al, be);
   12350             :   }
   12351         203 :   tsc1 = cusp_AC(al, &a, &c);
   12352         203 :   tsc2 = cusp_AC(be, &b, &d);
   12353         203 :   prec = nbits2prec(bitprec);
   12354         203 :   k = MF_get_k(mf);
   12355         203 :   if (!tsc1)
   12356             :   {
   12357          42 :     GEN z2, z = mfsymbolevalpartial(fs, al, ga, bitprec);
   12358          42 :     if (tsc2)
   12359          28 :       z2 = d? mfsymboleval(fs, mkvec2(be, mkoo()), ga, bitprec): gen_0;
   12360             :     else
   12361          14 :       z2 = mfsymbolevalpartial(fs, be, ga, bitprec);
   12362          42 :     z = gsub(z, z2);
   12363          42 :     if (vabd) z = unact(z, vabd, k, prec);
   12364          42 :     return gc_upto(av, normalizeapprox_i(z, bitprec-20));
   12365             :   }
   12366         161 :   else if (!tsc2)
   12367             :   {
   12368          21 :     GEN z = mfsymbolevalpartial(fs, be, ga, bitprec);
   12369          21 :     if (c) z = gsub(mfsymboleval(fs, mkvec2(al, mkoo()), ga, bitprec), z);
   12370           7 :     else z = gneg(z);
   12371          21 :     if (vabd) z = unact(z, vabd, k, prec);
   12372          21 :     return gc_upto(av, normalizeapprox_i(z, bitprec-20));
   12373             :   }
   12374         140 :   if (F) pari_err_TYPE("mfsymboleval", fs);
   12375         140 :   D = a*d-b*c;
   12376         140 :   if (!D) { set_avma(av); return RgX_embedall(gen_0, fs_get_vE(fs)); }
   12377         126 :   mfpols = fs_get_pols(fs);
   12378         126 :   cosets = fs_get_cosets(fs);
   12379         126 :   CHI = MF_get_CHI(mf); N = MF_get_N(mf);
   12380         126 :   cbezout(a, c, &u, &v); B = u*b + v*d; tau = mkmat22s(a, -v, c, u);
   12381         126 :   V = gcf(sstoQ(B, D));
   12382         126 :   LM = shallowconcat(mkcol2(gen_1, gen_0), contfracpnqn(V, lg(V)));
   12383         126 :   S = gen_0; m = lg(LM) - 2;
   12384         364 :   for (j = 0; j < m; j++)
   12385             :   {
   12386             :     GEN M, P;
   12387             :     long D, iN;
   12388         238 :     M = mkmat2(gel(LM, j+2), gel(LM, j+1));
   12389         238 :     if (!odd(j)) gel(M,1) = ZC_neg(gel(M,1));
   12390         238 :     M = ZM_mul(tau, M);
   12391         238 :     iN = mftocoset_iD(N, ZM_mul(ga, M), cosets, &D);
   12392         238 :     P = gmul(gel(mfpols,iN), mfcharcxeval(CHI,D,prec));
   12393         238 :     S = gadd(S, act_GL2(P, ZM_inv(M, NULL), k));
   12394             :   }
   12395         126 :   if (typ(S) == t_RFRAC)
   12396             :   {
   12397             :     GEN R, S1, co;
   12398          21 :     gel(S,2) = primitive_part(gel(S,2), &co);
   12399          21 :     if (co) gel(S,1) = gdiv(gel(S,1), gtofp(co,prec));
   12400          21 :     S1 = poldivrem(gel(S,1), gel(S,2), &R);
   12401          21 :     if (gexpo(R) < -bitprec + 20) S = S1;
   12402             :   }
   12403         126 :   if (vabd) S = unact(S, vabd, k, prec);
   12404         126 :   S = RgX_embedall(S, fs_get_vE(fs));
   12405         126 :   return gc_upto(av, normalizeapprox_i(S, bitprec-20));
   12406             : }
   12407             : 
   12408             : /* v a scalar or t_POL; set *pw = a if expo(a) > E for some coefficient;
   12409             :  * take the 'a' with largest exponent */
   12410             : static void
   12411        5740 : improve(GEN v, GEN *pw, long *E)
   12412             : {
   12413        5740 :   if (typ(v) != t_POL)
   12414             :   {
   12415        4270 :     long e = gexpo(v);
   12416        4270 :     if (e > *E) { *E = e; *pw = v; }
   12417             :   }
   12418             :   else
   12419             :   {
   12420        1470 :     long j, l = lg(v);
   12421        5740 :     for (j = 2; j < l; j++) improve(gel(v,j), pw, E);
   12422             :   }
   12423        5740 : }
   12424             : static GEN
   12425         518 : polabstorel(GEN rnfeq, GEN x)
   12426             : {
   12427         518 :   if (typ(x) != t_POL) return x;
   12428        3500 :   pari_APPLY_pol_normalized(eltabstorel(rnfeq, gel(x,i)));
   12429             : }
   12430             : static GEN
   12431        1519 : bestapprnfrel(GEN x, GEN polabs, GEN roabs, GEN rnfeq, long prec)
   12432             : {
   12433        1519 :   x = bestapprnf(x, polabs, roabs, prec);
   12434        1519 :   if (rnfeq) x = polabstorel(rnfeq, liftpol_shallow(x));
   12435        1519 :   return x;
   12436             : }
   12437             : /* v vector of polynomials polynomial in C[X] (possibly scalar).
   12438             :  * Set *w = coeff with largest exponent and return T / *w, rationalized */
   12439             : static GEN
   12440          98 : normal(GEN v, GEN polabs, GEN roabs, GEN rnfeq, GEN *w, long prec)
   12441             : {
   12442          98 :   long i, l = lg(v), E = -(long)HIGHEXPOBIT;
   12443             :   GEN dv;
   12444        1568 :   for (i = 1; i < l; i++) improve(gel(v,i), w, &E);
   12445          98 :   v = RgV_Rg_mul(v, ginv(*w));
   12446        1568 :   for (i = 1; i < l; i++)
   12447        1470 :     gel(v,i) = bestapprnfrel(gel(v,i), polabs,roabs,rnfeq,prec);
   12448          98 :   v = Q_primitive_part(v,&dv);
   12449          98 :   if (dv) *w = gmul(*w,dv);
   12450          98 :   return v;
   12451             : }
   12452             : 
   12453             : static GEN mfpetersson_i(GEN FS, GEN GS);
   12454             : 
   12455             : GEN
   12456          42 : mfmanin(GEN FS, long bitprec)
   12457             : {
   12458          42 :   pari_sp av = avma;
   12459             :   GEN mf, M, vp, vm, cosets, CHI, vpp, vmm, f, T, P, vE, polabs, roabs, rnfeq;
   12460             :   GEN pet;
   12461             :   long N, k, lco, i, prec, lvE;
   12462             : 
   12463          42 :   if (!checkfs_i(FS))
   12464             :   {
   12465           7 :     if (checkfs2_i(FS)) pari_err_TYPE("mfmanin [need integral k > 1]",FS);
   12466           0 :     pari_err_TYPE("mfmanin",FS);
   12467             :   }
   12468          35 :   if (!mfs_iscusp(FS)) pari_err_TYPE("mfmanin [noncuspidal]",FS);
   12469          35 :   mf = fs_get_MF(FS);
   12470          35 :   vp = fs_get_pols(FS);
   12471          35 :   cosets = fs_get_cosets(FS);
   12472          35 :   bitprec = fs_get_bitprec(FS);
   12473          35 :   N = MF_get_N(mf); k = MF_get_k(mf); CHI = MF_get_CHI(mf);
   12474          35 :   lco = lg(cosets); vm = cgetg(lco, t_VEC);
   12475          35 :   prec = nbits2prec(bitprec);
   12476         476 :   for (i = 1; i < lco; i++)
   12477             :   {
   12478         441 :     GEN g = gel(cosets, i), c;
   12479         441 :     long A = itos(gcoeff(g,1,1)), B = itos(gcoeff(g,1,2));
   12480         441 :     long C = itos(gcoeff(g,2,1)), D = itos(gcoeff(g,2,2));
   12481         441 :     long Dbar, ibar = mftocoset_iD(N, mkmat22s(-B,-A,D,C), cosets, &Dbar);
   12482             : 
   12483         441 :     c = mfcharcxeval(CHI, Dbar, prec); if (odd(k)) c = gneg(c);
   12484         441 :     T = RgX_Rg_mul(gel(vp,ibar), c);
   12485         441 :     if (typ(T) == t_POL && varn(T) == 0) T = RgX_recip(T);
   12486         441 :     gel(vm,i) = T;
   12487             :   }
   12488          35 :   vpp = gadd(vp,vm);
   12489          35 :   vmm = gsub(vp,vm);
   12490             : 
   12491          35 :   vE = fs_get_vE(FS); lvE = lg(vE);
   12492          35 :   f = gel(fs_get_EF(FS), 1);
   12493          35 :   P = mf_get_field(f); if (degpol(P) == 1) P = NULL;
   12494          35 :   T = mfcharpol(CHI);  if (degpol(T) == 1) T = NULL;
   12495          35 :   if (T && P)
   12496             :   {
   12497           7 :     rnfeq = nf_rnfeqsimple(T, P);
   12498           7 :     polabs = gel(rnfeq,1);
   12499           7 :     roabs = gel(QX_complex_roots(polabs,prec), 1);
   12500             :   }
   12501             :   else
   12502             :   {
   12503          28 :     rnfeq = roabs = NULL;
   12504          28 :     polabs = P? P: T;
   12505             :   }
   12506          35 :   pet = mfpetersson_i(FS, NULL);
   12507          35 :   M = cgetg(lvE, t_VEC);
   12508          84 :   for (i = 1; i < lvE; i++)
   12509             :   {
   12510          49 :     GEN p, m, wp, wm, petdiag, r, E = gel(vE,i);
   12511          49 :     p = normal(RgXV_embed(vpp, E), polabs, roabs, rnfeq, &wp, prec);
   12512          49 :     m = normal(RgXV_embed(vmm, E), polabs, roabs, rnfeq, &wm, prec);
   12513          49 :     petdiag = typ(pet)==t_MAT? gcoeff(pet,i,i): pet;
   12514          49 :     r = gdiv(mulimag(wp, conj_i(wm)), petdiag);
   12515          49 :     r = bestapprnfrel(r, polabs, roabs, rnfeq, prec);
   12516          49 :     gel(M,i) = mkvec2(mkvec2(p,m), mkvec3(wp,wm,r));
   12517             :   }
   12518          35 :   return gc_GEN(av, lvE == 2? gel(M,1): M);
   12519             : }
   12520             : 
   12521             : /* flag = 0: full, flag = +1 or -1, odd/even */
   12522             : /* Basis of period polynomials in level 1. */
   12523             : GEN
   12524          49 : mfperiodpolbasis(long k, long flag)
   12525             : {
   12526          49 :   pari_sp av = avma;
   12527          49 :   long i, j, n = k - 2;
   12528             :   GEN M, C, v;
   12529          49 :   if (k <= 4) return cgetg(1,t_VEC);
   12530          35 :   M = cgetg(k, t_MAT);
   12531          35 :   C = matpascal(n);
   12532          35 :   if (!flag)
   12533         392 :     for (j = 0; j <= n; j++)
   12534             :     {
   12535         371 :       gel(M, j+1) = v = cgetg(k, t_COL);
   12536        4767 :       for (i = 0; i <= j; i++) gel(v, i+1) = gcoeff(C, j+1, i+1);
   12537        4396 :       for (; i <= n; i++) gel(v, i+1) = gcoeff(C, n-j+1, i-j+1);
   12538             :     }
   12539             :   else
   12540         168 :     for (j = 0; j <= n; j++)
   12541             :     {
   12542         154 :       gel(M, j+1) = v = cgetg(k, t_COL);
   12543        1848 :       for (i = 0; i <= n; i++)
   12544             :       {
   12545        1694 :         GEN a = i < j ? gcoeff(C, j+1, i+1) : gen_0;
   12546        1694 :         if (i + j >= n)
   12547             :         {
   12548         924 :           GEN b = gcoeff(C, j+1, i+j-n+1);
   12549         924 :           a = flag < 0 ? addii(a,b) : subii(a,b);
   12550             :         }
   12551        1694 :         gel(v, i+1) = a;
   12552             :       }
   12553             :     }
   12554          35 :   return gc_GEN(av, RgM_to_RgXV(ZM_ker(M), 0));
   12555             : }
   12556             : 
   12557             : static int
   12558         168 : zero_at_cusp(GEN mf, GEN F, GEN c)
   12559             : {
   12560         168 :   GEN v = evalcusp(mf, F, c, LOWDEFAULTPREC);
   12561         168 :   return (gequal0(v) || gexpo(v) <= -62);
   12562             : }
   12563             : /* Compute list E of j such that F|_k g_j vanishes at oo: return [E, s(E)] */
   12564             : static void
   12565          14 : mffvanish(GEN mf, GEN F, GEN G, GEN cosets, GEN *pres, GEN *press)
   12566             : {
   12567          14 :   long j, lc = lg(cosets), N = MF_get_N(mf);
   12568             :   GEN v, vs;
   12569          14 :   *pres = v  = zero_zv(lc-1);
   12570          14 :   *press= vs = zero_zv(lc-1);
   12571         105 :   for (j = 1; j < lc; j++)
   12572             :   {
   12573          91 :     GEN ga = gel(cosets,j), c = mat2cusp(ga);
   12574          91 :     if (zero_at_cusp(mf, F, c))
   12575          14 :       v[j] = vs[ mftocoset_i(N, ZM_mulS(ga), cosets) ] = 1;
   12576          77 :     else if (!zero_at_cusp(mf, G, c))
   12577           0 :       pari_err_IMPL("divergent Petersson product");
   12578             :   }
   12579          14 : }
   12580             : static GEN
   12581         140 : Haberland(GEN PF, GEN PG, GEN vEF, GEN vEG, long k)
   12582             : {
   12583         140 :   GEN S = gen_0, vC = vecbinomial(k-2); /* vC[n+1] = (-1)^n binom(k-2,n) */
   12584         140 :   long n, j, l = lg(PG);
   12585         406 :   for (n = 2; n < k; n+=2) gel(vC,n) = negi(gel(vC,n));
   12586        2583 :   for (j = 1; j < l; j++)
   12587             :   {
   12588        2443 :     GEN PFj = gel(PF,j), PGj = gel(PG,j);
   12589       10038 :     for (n = 0; n <= k-2; n++)
   12590             :     {
   12591        7595 :       GEN a = RgX_coeff(PGj, k-2-n), b = RgX_coeff(PFj, n);
   12592        7595 :       a = Rg_embedall(a, vEG);
   12593        7595 :       b = Rg_embedall(b, vEF);
   12594        7595 :       a = conj_i(a); if (typ(a) == t_VEC) settyp(a, t_COL);
   12595             :       /* a*b = scalar or t_VEC or t_COL or t_MAT */
   12596        7595 :       S = gadd(S, gdiv(gmul(a,b), gel(vC,n+1)));
   12597             :     }
   12598             :   }
   12599         140 :   S = mulcxpowIs(gmul2n(S, 1-k), 1+k);
   12600         140 :   return vEF==vEG? real_i(S): S;
   12601             : }
   12602             : /* F1S, F2S both symbols, same mf */
   12603             : static GEN
   12604          14 : mfpeterssonnoncusp(GEN F1S, GEN F2S)
   12605             : {
   12606          14 :   pari_sp av = avma;
   12607             :   GEN mf, F1, F2, GF1, GF2, P2, cosets, vE1, vE2, FE1, FE2, P;
   12608             :   GEN I, IP1, RHO, RHOP1, INF, res, ress;
   12609          14 :   const double height = sqrt(3.)/2;
   12610             :   long k, r, j, bitprec, prec;
   12611             : 
   12612          14 :   mf = fs_get_MF(F1S);
   12613          14 :   FE1 = fs_get_EF(F1S); F1 = gel(FE1, 1);
   12614          14 :   FE2 = fs_get_EF(F2S); F2 = gel(FE2, 1);
   12615          14 :   cosets = fs_get_cosets(F1S);
   12616          14 :   bitprec = minuu(fs_get_bitprec(F1S), fs_get_bitprec(F2S));
   12617          14 :   prec = nbits2prec(bitprec);
   12618          14 :   F1S = fs_set_expan(F1S, mfgaexpansionall(mf, FE1, cosets, height, prec));
   12619          14 :   if (F2S != F1S)
   12620          14 :     F2S = fs_set_expan(F2S, mfgaexpansionall(mf, FE2, cosets, height, prec));
   12621          14 :   k = MF_get_k(mf); r = lg(cosets) - 1;
   12622          14 :   vE1 = fs_get_vE(F1S);
   12623          14 :   vE2 = fs_get_vE(F2S);
   12624          14 :   I = gen_I();
   12625          14 :   IP1 = mkcomplex(gen_1,gen_1);
   12626          14 :   RHO = rootsof1u_cx(3, prec+EXTRAPREC64);
   12627          14 :   RHOP1 = gaddsg(1, RHO);
   12628          14 :   INF = mkoo();
   12629          14 :   mffvanish(mf, F1, F2, cosets, &res, &ress);
   12630          14 :   P2 = fs_get_pols(F2S);
   12631          14 :   GF1 = cgetg(r+1, t_VEC);
   12632          14 :   GF2 = cgetg(r+1, t_VEC); P = get_P(k, fetch_var(), prec);
   12633         105 :   for (j = 1; j <= r; j++)
   12634             :   {
   12635          91 :     GEN g = gel(cosets,j);
   12636          91 :     if (res[j]) {
   12637          14 :       gel(GF1,j) = mfsymboleval_direct(F1S, mkvec2(RHOP1,INF), g, P);
   12638          14 :       gel(GF2,j) = mfsymboleval_direct(F2S, mkvec2(I,IP1), g, P);
   12639          77 :     } else if (ress[j]) {
   12640           7 :       gel(GF1,j) = mfsymboleval_direct(F1S, mkvec2(RHOP1,RHO), g, P);
   12641           7 :       gel(GF2,j) = mfsymboleval_direct(F2S, mkvec2(I,INF), g, P);
   12642             :     } else {
   12643          70 :       gel(GF1,j) = mfsymboleval_direct(F1S, mkvec2(RHO,I), g, P);
   12644          70 :       gel(GF2,j) = gneg(gel(P2,j)); /* - symboleval(F2S, [0,oo] */
   12645             :     }
   12646             :   }
   12647          14 :   delete_var();
   12648          14 :   return gc_upto(av, gdivgu(Haberland(GF1,GF2, vE1,vE2, k), r));
   12649             : }
   12650             : 
   12651             : /* Petersson product of F and G, given by mfsymbol's [k > 1 integral] */
   12652             : static GEN
   12653         140 : mfpetersson_i(GEN FS, GEN GS)
   12654             : {
   12655         140 :   pari_sp av = avma;
   12656             :   GEN mf, ESF, ESG, PF, PG, PH, CHI, cosets, vEF, vEG;
   12657             :   long k, r, j, N, bitprec, prec;
   12658             : 
   12659         140 :   if (!checkfs_i(FS)) pari_err_TYPE("mfpetersson",FS);
   12660         140 :   mf = fs_get_MF(FS);
   12661         140 :   ESF = fs_get_vES(FS);
   12662         140 :   if (!GS) GS = FS;
   12663             :   else
   12664             :   {
   12665          35 :     if (!checkfs_i(GS)) pari_err_TYPE("mfpetersson",GS);
   12666          35 :     if (!mfs_checkmf(GS, mf))
   12667           0 :       pari_err_TYPE("mfpetersson [different mf]", mkvec2(FS,GS));
   12668             :   }
   12669         140 :   ESG = fs_get_vES(GS);
   12670         140 :   if (!gequal0(gel(ESF,1)) || !gequal0(gel(ESG,1)))
   12671          14 :     return mfpeterssonnoncusp(FS, GS);
   12672         126 :   if (gequal0(gel(ESF,2)) || gequal0(gel(ESG,2))) return gc_const(av, gen_0);
   12673         126 :   N = MF_get_N(mf);
   12674         126 :   k = MF_get_k(mf);
   12675         126 :   CHI = MF_get_CHI(mf);
   12676         126 :   PF = fs_get_pols(FS); vEF = fs_get_vE(FS);
   12677         126 :   PG = fs_get_pols(GS); vEG = fs_get_vE(GS);
   12678         126 :   cosets = fs_get_cosets(FS);
   12679         126 :   bitprec = minuu(fs_get_bitprec(FS), fs_get_bitprec(GS));
   12680         126 :   prec = nbits2prec(bitprec);
   12681         126 :   r = lg(PG)-1;
   12682         126 :   PH = cgetg(r+1, t_VEC);
   12683        2478 :   for (j = 1; j <= r; j++)
   12684             :   {
   12685        2352 :     GEN ga = gel(cosets,j), PGj1, PGjm1;
   12686             :     long iT, D;
   12687        2352 :     iT = mftocoset_iD(N, ZM_mulTi(ga), cosets, &D);
   12688        2352 :     PGj1 = RgX_Rg_translate(gel(PG, iT), gen_1);
   12689        2352 :     PGj1 = RgX_Rg_mul(PGj1, mfcharcxeval(CHI, D, prec));
   12690        2352 :     iT = mftocoset_iD(N, ZM_mulT(ga), cosets, &D);
   12691        2352 :     PGjm1 = RgX_Rg_translate(gel(PG,iT), gen_m1);
   12692        2352 :     PGjm1 = RgX_Rg_mul(PGjm1, mfcharcxeval(CHI, D, prec));
   12693        2352 :     gel(PH,j) = gsub(PGj1, PGjm1);
   12694             :   }
   12695         126 :   return gc_upto(av, gdivgu(Haberland(PF, PH, vEF, vEG, k), 6*r));
   12696             : }
   12697             : 
   12698             : /****************************************************************/
   12699             : /*           Petersson products using Nelson-Collins            */
   12700             : /****************************************************************/
   12701             : /* Compute W(k,z) = sum_{m >= 1} (mz)^{k-1}(mzK_{k-2}(mz)-K_{k-1}(mz))
   12702             :  * for z>0 and absolute accuracy < 2^{-B}.
   12703             :  * K_k(x) ~ (Pi/(2x))^{1/2} e^{-x} */
   12704             : 
   12705             : static void
   12706       10304 : Wparams(GEN *ph, long *pN, long k, double x, long prec)
   12707             : {
   12708       10304 :   double B = prec2nbits(prec) + 10;
   12709       10304 :   double C = B + k*log(x)/M_LN2 + 1, D = C*M_LN2 + 2.065;
   12710       10304 :   double F = 2 * M_LN2 * (C - 1 + dbllog2(mpfact(k))) / x;
   12711       10304 :   double T = log(F) * (1 + 2*k/x/F), PI2 = M_PI*M_PI;
   12712       10304 :   *pN = (long)ceil((T/PI2) * (D + log(D/PI2)));
   12713       10304 :   *ph = gprec_w(dbltor(T / *pN), prec);
   12714       10304 : }
   12715             : 
   12716             : static void
   12717       10304 : Wcoshall(GEN *pCH, GEN *pCHK, GEN *pCHK1, GEN h, long N, long k, long prec)
   12718             : {
   12719       10304 :   GEN CH, CHK, CHK1, z = gexp(h, prec);
   12720       10304 :   GEN PO = gpowers(z, N), POK1 = gpowers(gpowgs(z, k-1), N);
   12721       10304 :   GEN E = ginv(gel(PO, N + 1)); /* exp(-hN) */
   12722       10304 :   GEN E1 = ginv(gel(POK1, N + 1)); /* exp(-(k-1)h) */
   12723             :   long j;
   12724       10304 :   *pCH = CH = cgetg(N+1, t_VEC);
   12725       10304 :   *pCHK = CHK = cgetg(N+1, t_VEC);
   12726       10304 :   *pCHK1 = CHK1 = cgetg(N+1, t_VEC);
   12727      146048 :   for (j = 1; j <= N; j++)
   12728             :   {
   12729      135744 :     GEN eh = gel(PO, j+1), emh = gmul(gel(PO, N-j+1), E); /* e^{jh}, e^{-jh} */
   12730      135744 :     GEN ek1h = gel(POK1, j+1), ek1mh = gmul(gel(POK1, N-j+1), E1);
   12731      135744 :     gel(CH, j) = gmul2n(gadd(eh, emh), -1); /* cosh(jh) */
   12732      135744 :     gel(CHK1,j) = gmul2n(gadd(ek1h, ek1mh), -1); /* cosh((k-1)jh) */
   12733      135744 :     gel(CHK, j) = gmul2n(gadd(gmul(eh, ek1h), gmul(emh, ek1mh)), -1);
   12734             :   }
   12735       10304 : }
   12736             : 
   12737             : /* computing W(k,x) via integral */
   12738             : static GEN
   12739       10304 : Wint(long k, GEN vP, GEN x, long prec)
   12740             : {
   12741             :   GEN P, P1, S1, S, h, CH, CHK, CHK1;
   12742             :   long N, j;
   12743       10304 :   Wparams(&h, &N, k, gtodouble(x), prec);
   12744       10304 :   Wcoshall(&CH, &CHK, &CHK1, h, N, k, prec);
   12745       10304 :   P = gel(vP, k+1); P1 = gel(vP, k); S = S1 = NULL;
   12746      156352 :   for (j = 0; j <= N; j++)
   12747             :   {
   12748      146048 :     GEN eh = gexp(j? gmul(x, gel(CH, j)): x, prec);
   12749      146048 :     GEN eh1 = gsubgs(eh, 1), eh1k = gpowgs(eh1, k), t1, t;
   12750      146048 :     t = gdiv(poleval(P, eh), gmul(eh1, eh1k));
   12751      146048 :     t1 = gdiv(poleval(P1, eh), eh1k);
   12752      146048 :     if (j)
   12753             :     {
   12754      135744 :       S = gadd(S, gmul(t, gel(CHK, j)));
   12755      135744 :       S1 = gadd(S1, gmul(t1, gel(CHK1, j)));
   12756             :     }
   12757             :     else
   12758             :     {
   12759       10304 :       S = gmul2n(t, -1);
   12760       10304 :       S1 = gmul2n(t1, -1);
   12761             :     }
   12762             :   }
   12763       10304 :   return gmul(gmul(h, gpowgs(x, k-1)), gsub(gmul(x, S), gmulsg(2*k-1, S1)));
   12764             : }
   12765             : 
   12766             : static GEN
   12767          21 : get_vP(long k)
   12768             : {
   12769          21 :   GEN P, v = cgetg(k+2, t_VEC), Q = deg1pol_shallow(gen_1,gen_m1,0);
   12770             :   long j;
   12771          21 :   gel(v,1) = gen_1;
   12772          21 :   gel(v,2) = P = pol_x(0);
   12773          28 :   for (j = 2; j <= k; j++)
   12774           7 :     gel(v,j+1) = P = RgX_shift_shallow(gsub(gmulsg(j, P),
   12775             :                                             gmul(Q, ZX_deriv(P))), 1);
   12776          21 :   return v;
   12777             : }
   12778             : /* vector of (-1)^j(1/(exp(x)-1))^(j) [x = z] * z^j for 0<=j<=r */
   12779             : static GEN
   12780       63742 : VS(long r, GEN z, GEN V, long prec)
   12781             : {
   12782       63742 :   GEN e = gexp(z, prec), c = ginv(gsubgs(e,1));
   12783       63742 :   GEN T = gpowers0(gmul(c, z), r, c);
   12784             :   long j;
   12785       63742 :   V = gsubst(V, 0, e);
   12786      143864 :   for (j = 1; j <= r + 1; j++) gel(V,j) = gmul(gel(V,j), gel(T,j));
   12787       63742 :   return V;
   12788             : }
   12789             : 
   12790             : /* U(r,x)=sum_{m >= 1} (mx)^k K_k(mx), k = r+1/2 */
   12791             : static GEN
   12792       71932 : Unelson(long r, GEN V)
   12793             : {
   12794       71932 :   GEN S = gel(V,r+1), C = gen_1; /* (r+j)! / j! / (r-j)! */
   12795             :   long j;
   12796       71932 :   if (!r) return S;
   12797       40950 :   for (j = 1; j <= r; j++)
   12798             :   {
   12799       24570 :     C = gdivgu(gmulgu(C, (r+j)*(r-j+1)), j);
   12800       24570 :     S = gadd(S, gmul2n(gmul(C, gel(V, r-j+1)), -j));
   12801             :   }
   12802       16380 :   return S;
   12803             : }
   12804             : /* W(r+1/2,z) / sqrt(Pi/2) */
   12805             : static GEN
   12806       63742 : Wint2(long r, GEN vP, GEN z, long prec)
   12807             : {
   12808       63742 :   GEN R, V = VS(r, z, vP, prec);
   12809       63742 :   R = Unelson(r, V);
   12810       63742 :   if (r) R = gsub(R, gmulsg(2*r, Unelson(r-1, V)));
   12811       63742 :   return R;
   12812             : }
   12813             : typedef GEN(*Wfun_t)(long, GEN, GEN, long);
   12814             : static GEN
   12815       74046 : WfromZ(GEN Z, GEN vP, GEN gkm1, Wfun_t W, long k, GEN pi4, long prec)
   12816             : {
   12817       74046 :   pari_sp av = avma;
   12818       74046 :   GEN Zk = gpow(Z, gkm1, prec), z = gmul(pi4, gsqrt(Z,prec));
   12819       74046 :   return gc_upto(av, gdiv(W(k, vP, z, prec), Zk));
   12820             : }
   12821             : /* mf a true mf or an fs2 */
   12822             : static GEN
   12823          21 : fs2_init(GEN mf, GEN F, long bit)
   12824             : {
   12825          21 :   pari_sp av = avma;
   12826          21 :   long i, l, lim, N, k, k2, prec = nbits2prec(bit);
   12827             :   GEN DEN, cusps, tab, gk, gkm1, W0, vW, vVW, vVF, vP, al0;
   12828          21 :   GEN vE = mfgetembed(F, prec), pi4 = Pi2n(2, prec);
   12829             :   Wfun_t Wf;
   12830             : 
   12831          21 :   if (lg(mf) == 7)
   12832             :   {
   12833          21 :     vW = cusps = NULL; /* true mf */
   12834          21 :     DEN = tab = NULL; /* -Wall */
   12835             :   }
   12836             :   else
   12837             :   { /* mf already an fs2, reset if its precision is too low */
   12838           0 :     vW = (fs2_get_bitprec(mf) < bit)? NULL: fs2_get_W(mf);
   12839           0 :     cusps = fs2_get_cusps(mf);
   12840           0 :     DEN = fs2_get_den(mf);
   12841           0 :     mf = fs2_get_MF(mf);
   12842             :   }
   12843          21 :   N = MF_get_N(mf);
   12844          21 :   gk = MF_get_gk(mf); gkm1 = gsubgs(gk, 1);
   12845          21 :   k2 = itos(gmul2n(gk,1));
   12846          21 :   Wf = odd(k2)? Wint2: Wint;
   12847          21 :   k = k2 >> 1; vP = get_vP(k);
   12848          21 :   if (vW) lim = (lg(gel(vW,1)) - 2) / N; /* vW[1] attached to cusp 0, width N */
   12849             :   else
   12850             :   { /* true mf */
   12851          21 :     double B = (bit + 10)*M_LN2;
   12852          21 :     double L = (B + k2*log(B)/2 + k2*k2*log(B)/(4*B)) / (4*M_PI);
   12853             :     long n, Lw;
   12854          21 :     lim = ((long)ceil(L*L));
   12855          21 :     Lw = N*lim;
   12856          21 :     tab = cgetg(Lw+1,t_VEC);
   12857       59157 :     for (n = 1; n <= Lw; n++)
   12858       59136 :       gel(tab,n) = WfromZ(uutoQ(n,N), vP, gkm1, Wf, k, pi4, prec);
   12859          21 :     if (!cusps) cusps = mfcusps_i(N);
   12860          21 :     DEN = gmul2n(gmulgu(gpow(Pi2n(3, prec), gkm1, prec), mypsiu(N)), -2);
   12861          21 :     if (odd(k2)) DEN = gdiv(DEN, sqrtr_abs(Pi2n(-1,prec)));
   12862             :   }
   12863          21 :   l = lg(cusps);
   12864          21 :   vVF = cgetg(l, t_VEC);
   12865          21 :   vVW = cgetg(l, t_VEC);
   12866          21 :   al0 = cgetg(l, t_VECSMALL);
   12867          21 :   W0 = k2==1? ginv(pi4): gen_0;
   12868         203 :   for (i = 1; i < l; i++)
   12869             :   {
   12870             :     long A, C, w, wi, Lw, n;
   12871             :     GEN VF, W, paramsF, al;
   12872         182 :     (void)cusp_AC(gel(cusps,i), &A,&C);
   12873         182 :     wi = ugcd(N, C*C); w = N / wi; Lw = w * lim;
   12874         182 :     VF = mfslashexpansion(mf, F, cusp2mat(A,C), Lw, 0, &paramsF, prec);
   12875             :     /* paramsF[2] = w */
   12876         182 :     al = gel(paramsF, 1); if (gequal0(al)) al = NULL;
   12877      100240 :     for (n = 0; n <= Lw; n++)
   12878             :     {
   12879      100058 :       GEN a = gel(VF,n+1);
   12880      100058 :       gel(VF,n+1) = gequal0(a)? gen_0: Rg_embedall(a, vE);
   12881             :     }
   12882         182 :     if (vW)
   12883           0 :       W = gel(vW, i);
   12884             :     else
   12885             :     {
   12886         182 :       W = cgetg(Lw+2, t_VEC);
   12887      100240 :       for (n = 0; n <= Lw; n++)
   12888      100058 :         gel(W, n+1) = al? WfromZ(gadd(al,uutoQ(n,w)),vP,gkm1,Wf,k,pi4, prec)
   12889      100058 :                         : (n? gel(tab, n * wi): W0);
   12890             :     }
   12891         182 :     al0[i] = !al;
   12892         182 :     gel(vVF, i) = VF;
   12893         182 :     gel(vVW, i) = W;
   12894             :   }
   12895          21 :   if (k2 <= 1) al0 = zero_zv(l-1); /* no need to test for convergence */
   12896          21 :   return gc_GEN(av, mkvecn(7, mf,vVW,cusps,vVF,utoipos(bit),al0,DEN));
   12897             : }
   12898             : 
   12899             : static GEN
   12900          21 : mfpetersson2(GEN Fs, GEN Gs)
   12901             : {
   12902          21 :   pari_sp av = avma;
   12903          21 :   GEN VC, RES, vF, vG, vW = fs2_get_W(Fs), al0 = fs2_get_al0(Fs);
   12904          21 :   long N = MF_get_N(fs2_get_MF(Fs)), j, lC;
   12905             : 
   12906          21 :   VC = fs2_get_cusps(Fs); lC = lg(VC);
   12907          21 :   vF = fs2_get_F(Fs);
   12908          21 :   vG = Gs? fs2_get_F(Gs): vF;
   12909          21 :   RES = gen_0;
   12910         203 :   for (j = 1; j < lC; j++)
   12911             :   {
   12912         182 :     GEN W = gel(vW,j), VF = gel(vF,j), VG = gel(vG,j), T = gen_0;
   12913         182 :     long A, C, w, n, L = lg(W);
   12914         182 :     pari_sp av = avma;
   12915         182 :     (void)cusp_AC(gel(VC,j), &A,&C); w = N/ugcd(N, C*C);
   12916         182 :     if (al0[j] && !isintzero(gel(VF,1)) && !isintzero(gel(VG,1)))
   12917           0 :       pari_err_IMPL("divergent Petersson product");
   12918      100240 :     for (n = 1; n < L; n++)
   12919             :     {
   12920      100058 :       GEN b = gel(VF,n), a = gel(VG,n);
   12921      100058 :       if (!isintzero(a) && !isintzero(b))
   12922             :       {
   12923       79964 :         T = gadd(T, gmul(gel(W,n), gmul(conj_i(a),b)));
   12924       79964 :         if (gc_needed(av,2)) T = gc_upto(av,T);
   12925             :       }
   12926             :     }
   12927         182 :     if (w != 1) T = gmulgu(T,w);
   12928         182 :     RES = gc_upto(av, gadd(RES, T));
   12929             :   }
   12930          21 :   if (!Gs) RES = real_i(RES);
   12931          21 :   return gc_upto(av, gdiv(RES, fs2_get_den(Fs)));
   12932             : }
   12933             : 
   12934             : static long
   12935         161 : symbol_type(GEN F)
   12936             : {
   12937         161 :   if (checkfs_i(F)) return 1;
   12938          21 :   if (checkfs2_i(F)) return 2;
   12939           0 :   return 0;
   12940             : }
   12941             : static int
   12942          35 : symbol_same_mf(GEN F, GEN G) { return gequal(gmael(F,1,1), gmael(G,1,1)); }
   12943             : GEN
   12944         126 : mfpetersson(GEN F, GEN G)
   12945             : {
   12946         126 :   long tF = symbol_type(F);
   12947         126 :   if (!tF) pari_err_TYPE("mfpetersson",F);
   12948         126 :   if (G)
   12949             :   {
   12950          35 :     long tG = symbol_type(G);
   12951          35 :     if (!tG) pari_err_TYPE("mfpetersson",F);
   12952          35 :     if (tF != tG || !symbol_same_mf(F,G))
   12953           0 :       pari_err_TYPE("mfpetersson [incompatible symbols]", mkvec2(F,G));
   12954             :   }
   12955         126 :   return (tF == 1)? mfpetersson_i(F, G): mfpetersson2(F, G);
   12956             : }
   12957             : 
   12958             : /****************************************************************/
   12959             : /*         projective Galois representation, weight 1           */
   12960             : /****************************************************************/
   12961             : static void
   12962         392 : moreorders(long N, GEN CHI, GEN F, GEN *pP, GEN *pO, ulong *bound)
   12963             : {
   12964         392 :   pari_sp av = avma;
   12965             :   forprime_t iter;
   12966         392 :   ulong a = *bound+1, b = 2*(*bound), p;
   12967         392 :   long i = 1;
   12968         392 :   GEN P, O, V = mfcoefs_i(F, b, 1);
   12969         392 :   *bound = b;
   12970         392 :   P = cgetg(b-a+2, t_VECSMALL);
   12971         392 :   O = cgetg(b-a+2, t_VECSMALL);
   12972         392 :   u_forprime_init(&iter, a, b);
   12973        2310 :   while((p = u_forprime_next(&iter))) if (N % p)
   12974             :   {
   12975        1813 :     O[i] = mffindrootof1(V, p, CHI);
   12976        1813 :     P[i++] = p;
   12977             :   }
   12978         392 :   setlg(P, i); *pP = shallowconcat(*pP, P);
   12979         392 :   setlg(O, i); *pO = shallowconcat(*pO, O);
   12980         392 :   (void)gc_all(av, 2, pP, pO);
   12981         392 : }
   12982             : 
   12983             : static GEN
   12984         182 : search_abelian(GEN nf, long n, long k, GEN N, GEN CHI, GEN F,
   12985             :                GEN *pP, GEN *pO, ulong *bound, long prec)
   12986             : {
   12987         182 :   pari_sp av = avma;
   12988             :   GEN bnr, cond, H, cyc, gn, T, Bquo, P, E;
   12989         182 :   long sN = itos(N), r1 = nf_get_r1(nf), i, j, d;
   12990             : 
   12991         182 :   cond = idealfactor(nf, N);
   12992         182 :   P = gel(cond,1);
   12993         182 :   E = gel(cond,2);
   12994         679 :   for (i = j = 1; i < lg(P); i++)
   12995             :   {
   12996         497 :     GEN pr = gel(P,i), Ej = gen_1;
   12997         497 :     long p = itos(pr_get_p(pr));
   12998         497 :     if (p == n)
   12999             :     {
   13000          98 :       long e = pr_get_e(pr); /* 1 + [e*p/(p-1)] */
   13001          98 :       Ej = utoipos(1 + (e*p) / (p-1));
   13002             :     }
   13003             :     else
   13004             :     {
   13005         399 :       long f = pr_get_f(pr);
   13006         399 :       if (Fl_powu(p % n, f, n) != 1) continue;
   13007             :     }
   13008         462 :     gel(P,j) = pr;
   13009         462 :     gel(E,j) = Ej; j++;
   13010             :   }
   13011         182 :   setlg(P,j);
   13012         182 :   setlg(E,j);
   13013         182 :   cond = mkvec2(cond, const_vec(r1, gen_1));
   13014         182 :   bnr = Buchraymod(Buchall(nf, nf_FORCE, prec), cond, nf_INIT, utoipos(n));
   13015         182 :   cyc = bnr_get_cyc(bnr);
   13016         182 :   d = lg(cyc)-1;
   13017         182 :   H = zv_diagonal(ZV_to_Flv(cyc, n));
   13018         182 :   gn = utoi(n);
   13019         182 :   for (i = 1;;)
   13020             :   {
   13021        2646 :     for(j = 2; i < lg(*pO); i++)
   13022             :     {
   13023        2072 :       long o, q = (*pP)[i];
   13024        2072 :       GEN pr = idealprimedec_galois(nf, stoi(q));
   13025        2072 :       o = ((*pO)[i] / pr_get_f(pr)) % n;
   13026        2072 :       if (o)
   13027             :       {
   13028        1442 :         GEN v = ZV_to_Flv(isprincipalray(bnr, pr), n);
   13029        1442 :         H = vec_append(H, Flv_Fl_mul(v, o, n));
   13030             :       }
   13031             :     }
   13032         574 :     H = Flm_image(H, n); if (lg(cyc)-lg(H) <= k) break;
   13033         392 :     moreorders(sN, CHI, F, pP, pO, bound);
   13034             :   }
   13035         182 :   H = hnfmodid(shallowconcat(zm_to_ZM(H), diagonal_shallow(cyc)), gn);
   13036             : 
   13037         182 :   Bquo = cgetg(k+1, t_MAT);
   13038         812 :   for (i = j = 1; i <= d; i++)
   13039         630 :     if (!equali1(gcoeff(H,i,i))) gel(Bquo,j++) = col_ei(d,i);
   13040             : 
   13041         441 :   for (i = 1, T = NULL; i<=k; i++)
   13042             :   {
   13043         259 :     GEN Hi = hnfmodid(shallowconcat(H, vecsplice(Bquo,i)), gn);
   13044         259 :     GEN pol = rnfkummer(bnr, Hi, prec);
   13045         259 :     T = T? nfcompositum(nf, T, pol, 2): pol;
   13046             :   }
   13047         182 :   T = rnfequation(nf, T); return gc_all(av, 3, &T, pP, pO);
   13048             : }
   13049             : 
   13050             : static GEN
   13051          77 : search_solvable(GEN LG, GEN mf, GEN F, long prec)
   13052             : {
   13053          77 :   GEN N = MF_get_gN(mf), CHI = MF_get_CHI(mf), pol, O, P, nf, Nfa;
   13054          77 :   long i, l = lg(LG), v = fetch_var();
   13055          77 :   ulong bound = 1;
   13056          77 :   O = cgetg(1, t_VECSMALL); /* projective order of rho(Frob_p) */
   13057          77 :   P = cgetg(1, t_VECSMALL);
   13058          77 :   Nfa = Z_factor(N);
   13059          77 :   pol = pol_x(v);
   13060         259 :   for (i = 1; i < l; i++)
   13061             :   { /* n prime, find a (Z/nZ)^k - extension */
   13062         182 :     GEN G = gel(LG,i);
   13063         182 :     long n = G[1], k = G[2];
   13064         182 :     nf = nfinitred(mkvec2(pol,Nfa), prec);
   13065         182 :     pol = search_abelian(nf, n, k, N, CHI, F, &P, &O, &bound, prec);
   13066         182 :     setvarn(pol,v);
   13067             :   }
   13068          77 :   delete_var(); setvarn(pol,0); return pol;
   13069             : }
   13070             : 
   13071             : static GEN
   13072           0 : search_A5(GEN mf, GEN F)
   13073             : {
   13074           0 :   GEN CHI = MF_get_CHI(mf), O, P, L;
   13075           0 :   long N = MF_get_N(mf), i, j, lL, nd, r;
   13076           0 :   ulong bound = 1;
   13077           0 :   r = radicalu(N);
   13078           0 :   L = veccond_to_A5(zv_z_mul(divisorsu(N/r),r), 2); lL = lg(L); nd = lL-1;
   13079           0 :   if (nd == 1) return gmael(L,1,1);
   13080           0 :   O = cgetg(1, t_VECSMALL); /* projective order of rho(Frob_p) */
   13081           0 :   P = cgetg(1, t_VECSMALL);
   13082           0 :   for(i = 1; nd > 1; )
   13083             :   {
   13084             :     long l;
   13085           0 :     moreorders(N, CHI, F, &P, &O, &bound);
   13086           0 :     l = lg(P);
   13087           0 :     for (   ; i < l; i++)
   13088             :     {
   13089           0 :       ulong p = P[i], f = O[i];
   13090           0 :       for (j = 1; j < lL; j++)
   13091           0 :         if (gel(L,j))
   13092             :         {
   13093           0 :           GEN FE = ZpX_primedec(gmael(L,j,1), utoi(p)), F = gel(FE,1);
   13094           0 :           long nF = lg(F)-1;
   13095           0 :           if (!equaliu(gel(F, nF), f)) { gel(L,j) = NULL; nd--; }
   13096             :         }
   13097           0 :       if (nd <= 1) break;
   13098             :     }
   13099             :   }
   13100           0 :   for (j = 1; j < lL; j++)
   13101           0 :     if (gel(L,j)) return gmael(L,j,1);
   13102           0 :   return NULL;
   13103             : }
   13104             : 
   13105             : GEN
   13106          77 : mfgaloisprojrep(GEN mf, GEN F, long prec)
   13107             : {
   13108          77 :   pari_sp av = avma;
   13109          77 :   GEN LG = NULL;
   13110          77 :   if (!checkMF_i(mf) && !checkmf_i(F)) pari_err_TYPE("mfgaloisrep", F);
   13111          77 :   switch( itos(mfgaloistype(mf,F)) )
   13112             :   {
   13113          49 :     case 0: case -12:
   13114          49 :       LG = mkvec2(mkvecsmall2(3,1), mkvecsmall2(2,2)); break;
   13115          28 :     case -24:
   13116          28 :       LG = mkvec3(mkvecsmall2(2,1), mkvecsmall2(3,1), mkvecsmall2(2,2)); break;
   13117           0 :     case -60: return gc_GEN(av, search_A5(mf, F));
   13118           0 :     default: pari_err_IMPL("mfgaloisprojrep for types D_n");
   13119             :   }
   13120          77 :   return gc_GEN(av, search_solvable(LG, mf, F, prec));
   13121             : }

Generated by: LCOV version 1.16