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 - modules - stark.c (source / functions) Hit Total Coverage
Test: PARI/GP v2.18.0 lcov report (development 29804-254f602fce) Lines: 1682 1817 92.6 %
Date: 2024-12-18 09:08:59 Functions: 128 130 98.5 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* Copyright (C) 2000  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             : /*        COMPUTATION OF STARK UNITS OF TOTALLY REAL FIELDS        */
      18             : /*                                                                 */
      19             : /*******************************************************************/
      20             : #include "pari.h"
      21             : #include "paripriv.h"
      22             : 
      23             : #define DEBUGLEVEL DEBUGLEVEL_stark
      24             : 
      25             : /* ComputeCoeff */
      26             : typedef struct {
      27             :   GEN L0, L1, L11, L2; /* VECSMALL of p */
      28             :   GEN L1ray, L11ray; /* precomputed isprincipalray(pr), pr | p */
      29             :   GEN rayZ; /* precomputed isprincipalray(i), i < condZ */
      30             :   long condZ; /* generates cond(bnr) \cap Z, assumed small */
      31             : } LISTray;
      32             : 
      33             : /* Char evaluation */
      34             : typedef struct {
      35             :   long ord;
      36             :   GEN *val, chi;
      37             : } CHI_t;
      38             : 
      39             : /* RecCoeff */
      40             : typedef struct {
      41             :   GEN M, beta, B, U, nB;
      42             :   long v, G, N;
      43             : } RC_data;
      44             : 
      45             : /********************************************************************/
      46             : /*                    Miscellaneous functions                       */
      47             : /********************************************************************/
      48             : static GEN
      49       18893 : chi_get_c(GEN chi) { return gmael(chi,1,2); }
      50             : static long
      51       54488 : chi_get_deg(GEN chi) { return itou(gmael(chi,1,1)); }
      52             : 
      53             : /* Compute the image of logelt by character chi, zeta_ord(chi)^n; return n */
      54             : static ulong
      55       14763 : CharEval_n(GEN chi, GEN logelt)
      56             : {
      57       14763 :   GEN gn = ZV_dotproduct(chi_get_c(chi), logelt);
      58       14763 :   return umodiu(gn, chi_get_deg(chi));
      59             : }
      60             : /* Compute the image of logelt by character chi, as a complex number */
      61             : static GEN
      62       14707 : CharEval(GEN chi, GEN logelt)
      63             : {
      64       14707 :   ulong n = CharEval_n(chi, logelt), d = chi_get_deg(chi);
      65       14707 :   long nn = Fl_center(n,d,d>>1);
      66       14707 :   GEN x = gel(chi,2);
      67       14707 :   x = gpowgs(x, labs(nn));
      68       14707 :   if (nn < 0) x = conj_i(x);
      69       14707 :   return x;
      70             : }
      71             : 
      72             : /* return n such that C(elt) = z^n */
      73             : static ulong
      74      644963 : CHI_eval_n(CHI_t *C, GEN logelt)
      75             : {
      76      644963 :   GEN n = ZV_dotproduct(C->chi, logelt);
      77      644963 :   return umodiu(n, C->ord);
      78             : }
      79             : /* return C(elt) */
      80             : static GEN
      81      643227 : CHI_eval(CHI_t *C, GEN logelt)
      82             : {
      83      643227 :   return C->val[CHI_eval_n(C, logelt)];
      84             : }
      85             : 
      86             : static void
      87        4130 : init_CHI(CHI_t *c, GEN CHI, GEN z)
      88             : {
      89        4130 :   long i, d = chi_get_deg(CHI);
      90        4130 :   GEN *v = (GEN*)new_chunk(d);
      91        4130 :   v[0] = gen_1;
      92        4130 :   if (d != 1)
      93             :   {
      94        4130 :     v[1] = z;
      95       35511 :     for (i=2; i<d; i++) v[i] = gmul(v[i-1], z);
      96             :   }
      97        4130 :   c->chi = chi_get_c(CHI);
      98        4130 :   c->ord = d;
      99        4130 :   c->val = v;
     100        4130 : }
     101             : /* as t_POLMOD */
     102             : static void
     103        2534 : init_CHI_alg(CHI_t *c, GEN CHI) {
     104        2534 :   long d = chi_get_deg(CHI);
     105             :   GEN z;
     106        2534 :   switch(d)
     107             :   {
     108           0 :     case 1: z = gen_1; break;
     109         945 :     case 2: z = gen_m1; break;
     110        1589 :     default: z = mkpolmod(pol_x(0), polcyclo(d,0));
     111             :   }
     112        2534 :   init_CHI(c,CHI, z);
     113        2534 : }
     114             : /* as t_COMPLEX */
     115             : static void
     116        1596 : init_CHI_C(CHI_t *c, GEN CHI) {
     117        1596 :   init_CHI(c,CHI, gel(CHI,2));
     118        1596 : }
     119             : 
     120             : typedef struct {
     121             :   long r; /* rank = lg(gen) */
     122             :   GEN j; /* current elt is gen[1]^j[1] ... gen[r]^j[r] */
     123             :   GEN cyc; /* t_VECSMALL of elementary divisors */
     124             : } GROUP_t;
     125             : 
     126             : static int
     127      684019 : NextElt(GROUP_t *G)
     128             : {
     129      684019 :   long i = 1;
     130      684019 :   if (G->r == 0) return 0; /* no more elt */
     131      755524 :   while (++G->j[i] == G->cyc[i]) /* from 0 to cyc[i]-1 */
     132             :   {
     133       72303 :     G->j[i] = 0;
     134       72303 :     if (++i > G->r) return 0; /* no more elt */
     135             :   }
     136      683221 :   return i; /* we have multiplied by gen[i] */
     137             : }
     138             : 
     139             : /* enumerate all group elements; trivial elt comes last */
     140             : GEN
     141       44751 : cyc2elts(GEN cyc)
     142             : {
     143             :   long i, n;
     144             :   GEN z;
     145             :   GROUP_t G;
     146             : 
     147       44751 :   G.cyc = typ(cyc)==t_VECSMALL? cyc: gtovecsmall(cyc);
     148       44751 :   n = zv_prod(G.cyc);
     149       44751 :   G.r = lg(cyc)-1;
     150       44751 :   G.j = zero_zv(G.r);
     151             : 
     152       44751 :   z = cgetg(n+1, t_VEC);
     153       44751 :   gel(z,n) = leafcopy(G.j); /* trivial elt comes last */
     154      675045 :   for  (i = 1; i < n; i++)
     155             :   {
     156      630294 :     (void)NextElt(&G);
     157      630294 :     gel(z,i) = leafcopy(G.j);
     158             :   }
     159       44751 :   return z;
     160             : }
     161             : 
     162             : /* nchi: a character given by a vector [d, (c_i)], e.g. from char_normalize
     163             :  * such that chi(x) = e((c . log(x)) / d) where log(x) on bnr.gen */
     164             : static GEN
     165        3661 : get_Char(GEN nchi, long prec)
     166        3661 : { return mkvec2(nchi, rootsof1_cx(gel(nchi,1), prec)); }
     167             : 
     168             : /* prime divisors of conductor */
     169             : static GEN
     170         448 : divcond(GEN bnr) {GEN bid = bnr_get_bid(bnr); return gel(bid_get_fact(bid),1);}
     171             : 
     172             : /* vector of prime ideals dividing bnr but not bnrc */
     173             : static GEN
     174         161 : get_prdiff(GEN D, GEN Dc)
     175             : {
     176         161 :   long n, i, l  = lg(D);
     177         161 :   GEN diff = cgetg(l, t_COL);
     178         448 :   for (n = i = 1; i < l; i++)
     179         287 :     if (!tablesearch(Dc, gel(D,i), &cmp_prime_ideal)) gel(diff,n++) = gel(D,i);
     180         161 :   setlg(diff, n); return diff;
     181             : }
     182             : 
     183             : #define ch_prec(x) realprec(gel(x,1))
     184             : #define ch_C(x)    gel(x,1)
     185             : #define ch_bnr(x)  gel(x,2)
     186             : #define ch_3(x)    gel(x,3)
     187             : #define ch_q(x)    gel(x,3)[1]
     188             : #define ch_CHI(x)  gel(x,4)
     189             : #define ch_diff(x) gel(x,5)
     190             : #define ch_CHI0(x) gel(x,6)
     191             : #define ch_small(x) gel(x,7)
     192             : #define ch_comp(x) gel(x,7)[1]
     193             : #define ch_phideg(x) gel(x,7)[2]
     194             : static long
     195        1295 : ch_deg(GEN dtcr) { return chi_get_deg(ch_CHI(dtcr)); }
     196             : 
     197             : /********************************************************************/
     198             : /*                    1rst part: find the field K                   */
     199             : /********************************************************************/
     200             : static GEN AllStark(GEN data, long flag, long prec);
     201             : 
     202             : /* Columns of C [HNF] give the generators of a subgroup of the finite abelian
     203             :  * group A [ in terms of implicit generators ], compute data to work in A/C:
     204             :  * 1) order
     205             :  * 2) structure
     206             :  * 3) the matrix A ->> A/C
     207             :  * 4) the subgroup C */
     208             : static GEN
     209        1316 : InitQuotient(GEN C)
     210             : {
     211        1316 :   GEN U, D = ZM_snfall_i(C, &U, NULL, 1), h = ZV_prod(D);
     212        1316 :   return mkvec5(h, D, U, C, cyc_normalize(D));
     213             : }
     214             : 
     215             : /* lift chi character on A/C [Qt from InitQuotient] to character on A [cyc]*/
     216             : static GEN
     217        3612 : LiftChar(GEN Qt, GEN cyc, GEN chi)
     218             : {
     219        3612 :   GEN ncyc = gel(Qt,5), U = gel(Qt,3), nchi = char_normalize(chi, ncyc);
     220        3612 :   GEN c = ZV_ZM_mul(gel(nchi,2), U), d = gel(nchi,1);
     221        3612 :   return char_denormalize(cyc, d, c);
     222             : }
     223             : 
     224             : /* Let C be a subgroup, system of representatives of the quotient */
     225             : static GEN
     226         329 : subgroup_classes(GEN C)
     227             : {
     228         329 :   GEN U, D = ZM_snfall_i(C, &U, NULL, 1), e = cyc2elts(D);
     229         329 :   long i, l = lg(e);
     230             : 
     231         329 :   if (ZM_isidentity(U))
     232        2030 :     for (i = 1; i < l; i++) (void)vecsmall_to_vec_inplace(gel(e,i));
     233             :   else
     234             :   {
     235          14 :     GEN Ui = ZM_inv(U,NULL);
     236          84 :     for (i = 1; i < l; i++) gel(e,i) = ZM_zc_mul(Ui, gel(e,i));
     237             :   }
     238         329 :   return e;
     239             : }
     240             : 
     241             : /* Let s: A -> B given by [P,cycA,cycB] A and B, compute the kernel of s. */
     242             : GEN
     243         455 : abmap_kernel(GEN S)
     244             : {
     245         455 :   GEN U, P = gel(S,1), cycA = gel(S,2), DB = diagonal_shallow(gel(S,3));
     246         455 :   long nA = lg(cycA)-1, rk;
     247             : 
     248         455 :   rk = nA + lg(DB) - lg(ZM_hnfall_i(shallowconcat(P, DB), &U, 1));
     249         455 :   return ZM_hnfmodid(matslice(U, 1,nA, 1,rk), cycA);
     250             : }
     251             : /* let H be a subgroup of A; return s(H) */
     252             : GEN
     253        1519 : abmap_subgroup_image(GEN S, GEN H)
     254        1519 : { return ZM_hnfmodid(ZM_mul(gel(S,1), H),  gel(S,3)); }
     255             : 
     256             : /* Let m and n be two moduli such that n|m and let C be a congruence
     257             :    group modulo n, compute the corresponding congruence group modulo m
     258             :    ie the kernel of the map Clk(m) ->> Clk(n)/C */
     259             : static GEN
     260         455 : ComputeKernel(GEN bnrm, GEN bnrn, GEN dtQ)
     261             : {
     262         455 :   pari_sp av = avma;
     263         455 :   GEN S = bnrsurjection(bnrm, bnrn);
     264         455 :   GEN P = ZM_mul(gel(dtQ,3), gel(S,1));
     265         455 :   return gerepileupto(av, abmap_kernel(mkvec3(P, gel(S,2), gel(dtQ,2))));
     266             : }
     267             : 
     268             : static long
     269        1169 : cyc_is_cyclic(GEN cyc) { return lg(cyc) <= 2 || equali1(gel(cyc,2)); }
     270             : 
     271             : /* Let H be a subgroup of cl(bnr)/sugbroup, return 1 if
     272             :    cl(bnr)/subgoup/H is cyclic and the signature of the
     273             :    corresponding field is equal to sig and no finite prime
     274             :    dividing cond(bnr) is totally split in this field. Return 0
     275             :    otherwise. */
     276             : static long
     277         518 : IsGoodSubgroup(GEN H, GEN bnr, GEN map)
     278             : {
     279         518 :   pari_sp av = avma;
     280             :   GEN S, mod, modH, p1, U, P, PH, bnrH, iH, qH;
     281             :   long j;
     282             : 
     283         518 :   p1 = InitQuotient(H);
     284             :   /* quotient is non cyclic */
     285         518 :   if (!cyc_is_cyclic(gel(p1,2))) return gc_long(av,0);
     286             : 
     287         252 :   (void)ZM_hnfall_i(shallowconcat(map,H), &U, 0);
     288         252 :   setlg(U, lg(H));
     289         924 :   for (j = 1; j < lg(U); j++) setlg(gel(U,j), lg(H));
     290         252 :   p1 = ZM_hnfmodid(U, bnr_get_cyc(bnr)); /* H as a subgroup of bnr */
     291         252 :   modH = bnrconductor_raw(bnr, p1);
     292         252 :   mod  = bnr_get_mod(bnr);
     293             : 
     294             :   /* is the signature correct? */
     295         252 :   if (!gequal(gel(modH,2), gel(mod,2))) return gc_long(av, 0);
     296             : 
     297             :   /* finite part are the same: OK */
     298         182 :   if (gequal(gel(modH,1), gel(mod,1))) return gc_long(av, 1);
     299             : 
     300             :   /* need to check the splitting of primes dividing mod but not modH */
     301          63 :   bnrH = Buchray(bnr, modH, nf_INIT);
     302          63 :   P = divcond(bnr);
     303          63 :   PH = divcond(bnrH);
     304          63 :   S = bnrsurjection(bnr, bnrH);
     305             :   /* H as a subgroup of bnrH */
     306          63 :   iH = abmap_subgroup_image(S, p1);
     307          63 :   qH = InitQuotient(iH);
     308         203 :   for (j = 1; j < lg(P); j++)
     309             :   {
     310         161 :     GEN pr = gel(P, j), e;
     311             :     /* if pr divides modH, it is ramified, so it's good */
     312         161 :     if (tablesearch(PH, pr, cmp_prime_ideal)) continue;
     313             :     /* inertia degree of pr in bnr(modH)/H is charorder(cycH, e) */
     314          56 :     e = ZM_ZC_mul(gel(qH,3), isprincipalray(bnrH, pr));
     315          56 :     e = ZV_ZV_mod(e, gel(qH,2));
     316          56 :     if (ZV_equal0(e)) return gc_long(av,0); /* f = 1 */
     317             :   }
     318          42 :   return gc_long(av,1);
     319             : }
     320             : 
     321             : /* compute list of nontrivial characters trivial on H, modulo complex
     322             :  * conjugation. If flag is set, impose a nontrivial conductor at infinity */
     323             : static GEN
     324         399 : AllChars(GEN bnr, GEN dtQ, long flag)
     325             : {
     326         399 :   GEN v, vchi, cyc = bnr_get_cyc(bnr);
     327         399 :   long n, i, hD = itos(gel(dtQ,1));
     328             :   hashtable *S;
     329             : 
     330         399 :   v = cgetg(hD+1, t_VEC); /* nonconjugate chars */
     331         399 :   vchi = cyc2elts(gel(dtQ,2));
     332         399 :   S = hash_create(hD, (ulong(*)(void*))&hash_GEN,
     333             :                   (int(*)(void*,void*))&ZV_equal, 1);
     334        4011 :   for (i = n = 1; i < hD; i++) /* remove i = hD: trivial char */
     335             :   { /* lift a character of D in Clk(m) */
     336        3612 :     GEN F, lchi = LiftChar(dtQ, cyc, zv_to_ZV(gel(vchi,i))), cchi = NULL;
     337             : 
     338        3612 :     if (hash_search(S, lchi)) continue;
     339        2765 :     F = bnrconductor_raw(bnr, lchi);
     340        2765 :     if (flag && gequal0(gel(F,2))) continue; /* f_oo(chi) trivial ? */
     341             : 
     342        1309 :     if (abscmpiu(charorder(cyc,lchi), 2) > 0)
     343             :     { /* nonreal chi: add its conjugate character to S */
     344         847 :       cchi = charconj(cyc, lchi);
     345         847 :       hash_insert(S, cchi, (void*)1);
     346             :     }
     347        1309 :     gel(v, n++) = cchi? mkvec3(lchi, F, cchi): mkvec2(lchi, F);
     348             :   }
     349         399 :   setlg(v, n); return v;
     350             : }
     351             : 
     352             : static GEN InitChar(GEN bnr, GEN CR, long flag, long prec);
     353             : static void CharNewPrec(GEN data, long prec);
     354             : 
     355             : /* Given a conductor and a subgroups, return the corresponding complexity and
     356             :  * precision required using quickpol. Fill data[5] with dataCR */
     357             : static long
     358         329 : CplxModulus(GEN data, long *newprec)
     359             : {
     360         329 :   long dprec = DEFAULTPREC;
     361         329 :   pari_sp av = avma;
     362             :   for (;;)
     363           0 :   {
     364         329 :     GEN cpl, pol = AllStark(data, 1, dprec);
     365         329 :     cpl = RgX_fpnorml2(pol, LOWDEFAULTPREC);
     366         329 :     dprec = maxss(dprec, nbits2extraprec(gexpo(pol))) + EXTRAPREC64;
     367         329 :     if (!gequal0(cpl)) { *newprec = dprec; return gexpo(cpl); }
     368           0 :     set_avma(av);
     369           0 :     if (DEBUGLEVEL>1) pari_warn(warnprec, "CplxModulus", dprec);
     370           0 :     CharNewPrec(data, dprec);
     371             :   }
     372             : }
     373             : 
     374             : /* return A \cap B in abelian group defined by cyc. NULL = whole group */
     375             : static GEN
     376         567 : subgp_intersect(GEN cyc, GEN A, GEN B)
     377             : {
     378             :   GEN H, U;
     379             :   long k, lH;
     380         567 :   if (!A) return B;
     381         224 :   if (!B) return A;
     382         224 :   H = ZM_hnfall_i(shallowconcat(A,B), &U, 1);
     383         224 :   setlg(U, lg(A)); lH = lg(H);
     384         798 :   for (k = 1; k < lg(U); k++) setlg(gel(U,k), lH);
     385         224 :   return ZM_hnfmodid(ZM_mul(A,U), cyc);
     386             : }
     387             : 
     388             : static void CharNewPrec(GEN dataCR, long prec);
     389             : /* Let (f,C) be a conductor without infinite part and a congruence group mod f.
     390             :  * Compute (m,D) such that D is a congruence group of conductor m, f | m,
     391             :  * divisible by all the infinite places but one, D is a subgroup of index 2 of
     392             :  * Cm = Ker: Clk(m) -> Clk(f)/C. Consider further the subgroups H of Clk(m)/D
     393             :  * with cyclic quotient Clk(m)/H such that no place dividing m is totally split
     394             :  * in the extension KH corresponding to H: we want their intersection to be
     395             :  * trivial. These H correspond to (the kernels of Galois orbits of) characters
     396             :  * chi of Clk(m)/D such that chi(log_gen_arch(m_oo)) != 1 and for all pr | m
     397             :  * we either have
     398             :  * - chi(log_gen_pr(pr,1)) != 1 [pr | cond(chi) => ramified in KH]
     399             :  * - or [pr \nmid cond(chi)] chi lifted to Clk(m/pr^oo) is not trivial at pr.
     400             :  * We want the map from Clk(m)/D given by the vector of such characters to have
     401             :  * trivial kernel. Return bnr(m), D, Ck(m)/D and Clk(m)/Cm */
     402             : static GEN
     403         322 : FindModulus(GEN bnr, GEN dtQ, long *newprec)
     404             : {
     405         322 :   const long LIMNORM = 400;
     406         322 :   long n, i, maxnorm, minnorm, N, pr, rb, iscyc, olde = LONG_MAX;
     407         322 :   pari_sp av = avma;
     408         322 :   GEN bnf, nf, f, varch, m, rep = NULL;
     409             : 
     410         322 :   bnf = bnr_get_bnf(bnr);
     411         322 :   nf  = bnf_get_nf(bnf);
     412         322 :   N   = nf_get_degree(nf);
     413         322 :   f   = gel(bnr_get_mod(bnr), 1);
     414             : 
     415             :   /* if cpl < rb, it is not necessary to try another modulus */
     416         322 :   rb = expi( powii(mulii(nf_get_disc(nf), ZM_det_triangular(f)),
     417             :                    gmul2n(bnr_get_no(bnr), 3)) );
     418             : 
     419             :   /* Initialization of the possible infinite part */
     420         322 :   varch = cgetg(N+1,t_VEC);
     421        1106 :   for (i = 1; i <= N; i++)
     422             :   {
     423         784 :     GEN a = const_vec(N,gen_1);
     424         784 :     gel(a, N+1-i) = gen_0;
     425         784 :     gel(varch, i) = a;
     426             :   }
     427         322 :   m = cgetg(3, t_VEC);
     428             : 
     429             :   /* Go from minnorm up to maxnorm; if necessary, increase these values.
     430             :    * If the extension is cyclic then a suitable conductor exists and we go on
     431             :    * until we find it. Else, stop at norm LIMNORM. */
     432         322 :   minnorm = 1;
     433         322 :   maxnorm = 50;
     434         322 :   iscyc = cyc_is_cyclic(gel(dtQ,2));
     435             : 
     436         322 :   if (DEBUGLEVEL>1)
     437           0 :     err_printf("Looking for a modulus of norm: ");
     438             : 
     439             :   for(;;)
     440           0 :   {
     441         322 :     GEN listid = ideallist0(nf, maxnorm, 4+8); /* ideals of norm <= maxnorm */
     442         322 :     pari_sp av1 = avma;
     443        1463 :     for (n = minnorm; n <= maxnorm; n++, set_avma(av1))
     444             :     {
     445        1463 :       GEN idnormn = gel(listid,n);
     446        1463 :       long nbidnn  = lg(idnormn) - 1;
     447        1463 :       if (DEBUGLEVEL>1) err_printf(" %ld", n);
     448        2296 :       for (i = 1; i <= nbidnn; i++)
     449             :       { /* finite part of the conductor */
     450             :         long s;
     451             : 
     452        1155 :         gel(m,1) = idealmul(nf, f, gel(idnormn,i));
     453        3213 :         for (s = 1; s <= N; s++)
     454             :         { /* infinite part */
     455             :           GEN candD, Cm, bnrm;
     456             :           long lD, c;
     457             : 
     458        2380 :           gel(m,2) = gel(varch,s);
     459             :           /* compute Clk(m), check if m is a conductor */
     460        2380 :           bnrm = Buchray(bnf, m, nf_INIT);
     461        2380 :           if (!bnrisconductor(bnrm, NULL)) continue;
     462             : 
     463             :           /* compute Im(C) in Clk(m)... */
     464         448 :           Cm = ComputeKernel(bnrm, bnr, dtQ);
     465             :           /* ... and its subgroups of index 2 with conductor m */
     466         448 :           candD = subgrouplist_cond_sub(bnrm, Cm, mkvec(gen_2));
     467         448 :           lD = lg(candD);
     468         455 :           for (c = 1; c < lD; c++)
     469             :           {
     470         329 :             GEN data, CR, D = gel(candD,c), QD = InitQuotient(D);
     471         329 :             GEN ord = gel(QD,1), cyc = gel(QD,2), map = gel(QD,3);
     472             :             long e;
     473             : 
     474         329 :             if (!cyc_is_cyclic(cyc)) /* cyclic => suitable, else test */
     475             :             {
     476          77 :               GEN lH = subgrouplist(cyc, NULL), IK = NULL;
     477          77 :               long j, ok = 0;
     478         574 :               for (j = 1; j < lg(lH); j++)
     479             :               {
     480         567 :                 GEN H = gel(lH, j), IH = subgp_intersect(cyc, IK, H);
     481             :                 /* if H > IK, no need to test H */
     482         567 :                 if (IK && gidentical(IH, IK)) continue;
     483         518 :                 if (IsGoodSubgroup(H, bnrm, map))
     484             :                 {
     485         161 :                   IK = IH; /* intersection of all good subgroups */
     486         161 :                   if (equalii(ord, ZM_det_triangular(IK))) { ok = 1; break; }
     487             :                 }
     488             :               }
     489          77 :               if (!ok) continue;
     490             :             }
     491         322 :             CR = InitChar(bnrm, AllChars(bnrm, QD, 1), 0, DEFAULTPREC);
     492         322 :             data = mkvec4(bnrm, D, subgroup_classes(Cm), CR);
     493         322 :             if (DEBUGLEVEL>1)
     494           0 :               err_printf("\nTrying modulus = %Ps and subgroup = %Ps\n",
     495             :                          bnr_get_mod(bnrm), D);
     496         322 :             e = CplxModulus(data, &pr);
     497         322 :             if (DEBUGLEVEL>1) err_printf("cpl = 2^%ld\n", e);
     498         322 :             if (e < olde)
     499             :             {
     500         322 :               guncloneNULL(rep); rep = gclone(data);
     501         322 :               *newprec = pr; olde = e;
     502             :             }
     503         322 :             if (olde < rb) goto END; /* OK */
     504           0 :             if (DEBUGLEVEL>1) err_printf("Trying to find another modulus...");
     505             :           }
     506             :         }
     507         833 :         if (rep) goto END; /* OK */
     508             :       }
     509             :     }
     510             :     /* if necessary compute more ideals */
     511           0 :     minnorm = maxnorm;
     512           0 :     maxnorm <<= 1;
     513           0 :     if (!iscyc && maxnorm > LIMNORM) return NULL;
     514             :   }
     515         322 : END:
     516         322 :   if (DEBUGLEVEL>1)
     517           0 :     err_printf("No, we're done!\nModulus = %Ps and subgroup = %Ps\n",
     518           0 :                bnr_get_mod(gel(rep,1)), gel(rep,2));
     519         322 :   CharNewPrec(rep, *newprec); return gerepilecopy(av, rep);
     520             : }
     521             : 
     522             : /********************************************************************/
     523             : /*                      2nd part: compute W(X)                      */
     524             : /********************************************************************/
     525             : 
     526             : /* find ilambda s.t. Diff*f*ilambda integral and coprime to f
     527             :    and ilambda >> 0 at foo, fa = factorization of f */
     528             : static GEN
     529         798 : get_ilambda(GEN nf, GEN fa, GEN foo)
     530             : {
     531         798 :   GEN x, w, E2, P = gel(fa,1), E = gel(fa,2), D = nf_get_diff(nf);
     532         798 :   long i, l = lg(P);
     533         798 :   if (l == 1) return gen_1;
     534         665 :   w = cgetg(l, t_VEC);
     535         665 :   E2 = cgetg(l, t_COL);
     536        1456 :   for (i = 1; i < l; i++)
     537             :   {
     538         791 :     GEN pr = gel(P,i), t = pr_get_tau(pr);
     539         791 :     long e = itou(gel(E,i)), v = idealval(nf, D, pr);
     540         791 :     if (v) { D = idealdivpowprime(nf, D, pr, utoipos(v)); e += v; }
     541         791 :     gel(E2,i) = stoi(e+1);
     542         791 :     if (typ(t) == t_MAT) t = gel(t,1);
     543         791 :     gel(w,i) = gdiv(nfpow(nf, t, stoi(e)), powiu(pr_get_p(pr),e));
     544             :   }
     545         665 :   x = mkmat2(P, E2);
     546         665 :   return idealchinese(nf, mkvec2(x, foo), w);
     547             : }
     548             : /* compute the list of W(chi) such that Ld(s,chi) = W(chi) Ld(1 - s, chi*),
     549             :  * for all chi in LCHI. All chi have the same conductor (= cond(bnr)). */
     550             : static GEN
     551         994 : ArtinNumber(GEN bnr, GEN LCHI, long prec)
     552             : {
     553         994 :   long ic, i, j, nz, nChar = lg(LCHI)-1;
     554             :   pari_sp av2;
     555             :   GEN sqrtnc, cond, condZ, cond0, cond1, nf, T, cyc, vN, vB, diff, vt, idh;
     556             :   GEN zid, gen, z, nchi, indW, W, classe, s0, s, den, ilambda, sarch;
     557             :   CHI_t **lC;
     558             :   GROUP_t G;
     559             : 
     560         994 :   lC = (CHI_t**)new_chunk(nChar + 1);
     561         994 :   indW = cgetg(nChar + 1, t_VECSMALL);
     562         994 :   W = cgetg(nChar + 1, t_VEC);
     563        3451 :   for (ic = 0, i = 1; i <= nChar; i++)
     564             :   {
     565        2457 :     GEN CHI = gel(LCHI,i);
     566        2457 :     if (chi_get_deg(CHI) <= 2) { gel(W,i) = gen_1; continue; }
     567        1596 :     ic++; indW[ic] = i;
     568        1596 :     lC[ic] = (CHI_t*)new_chunk(sizeof(CHI_t));
     569        1596 :     init_CHI_C(lC[ic], CHI);
     570             :   }
     571         994 :   if (!ic) return W;
     572         798 :   nChar = ic;
     573             : 
     574         798 :   nf    = bnr_get_nf(bnr);
     575         798 :   diff  = nf_get_diff(nf);
     576         798 :   T     = nf_get_Tr(nf);
     577         798 :   cond  = bnr_get_mod(bnr);
     578         798 :   cond0 = gel(cond,1); condZ = gcoeff(cond0,1,1);
     579         798 :   cond1 = gel(cond,2);
     580             : 
     581         798 :   sqrtnc = gsqrt(idealnorm(nf, cond0), prec);
     582         798 :   ilambda = get_ilambda(nf, bid_get_fact(bnr_get_bid(bnr)), cond1);
     583         798 :   idh = idealmul(nf, ilambda, idealmul(nf, diff, cond0)); /* integral */
     584         798 :   ilambda = Q_remove_denom(ilambda, &den);
     585         798 :   z = den? rootsof1_cx(den, prec): NULL;
     586             : 
     587             :   /* compute a system of generators of (Ok/cond)^*, we'll make them
     588             :    * cond1-positive in the main loop */
     589         798 :   zid = Idealstar(nf, cond0, nf_GEN);
     590         798 :   cyc = abgrp_get_cyc(zid);
     591         798 :   gen = abgrp_get_gen(zid);
     592         798 :   nz = lg(gen) - 1;
     593         798 :   sarch = nfarchstar(nf, cond0, vec01_to_indices(cond1));
     594             : 
     595         798 :   nchi = cgetg(nChar+1, t_VEC);
     596        2394 :   for (ic = 1; ic <= nChar; ic++) gel(nchi,ic) = cgetg(nz + 1, t_VECSMALL);
     597        1631 :   for (i = 1; i <= nz; i++)
     598             :   {
     599         833 :     if (is_bigint(gel(cyc,i)))
     600           0 :       pari_err_OVERFLOW("ArtinNumber [conductor too large]");
     601         833 :     gel(gen,i) = set_sign_mod_divisor(nf, NULL, gel(gen,i), sarch);
     602         833 :     classe = isprincipalray(bnr, gel(gen,i));
     603        2569 :     for (ic = 1; ic <= nChar; ic++) {
     604        1736 :       GEN n = gel(nchi,ic);
     605        1736 :       n[i] = CHI_eval_n(lC[ic], classe);
     606             :     }
     607             :   }
     608             : 
     609             :   /* Sum chi(beta) * exp(2i * Pi * Tr(beta * ilambda) where beta
     610             :      runs through the classes of (Ok/cond0)^* and beta cond1-positive */
     611         798 :   vt = gel(T,1); /* ( Tr(w_i) )_i */
     612         798 :   if (typ(ilambda) == t_COL)
     613         574 :     vt = ZV_ZM_mul(vt, zk_multable(nf, ilambda));
     614             :   else
     615         224 :     vt = ZC_Z_mul(vt, ilambda);
     616             :   /*vt = den . (Tr(w_i * ilambda))_i */
     617         798 :   G.cyc = gtovecsmall(cyc);
     618         798 :   G.r = nz;
     619         798 :   G.j = zero_zv(nz);
     620         798 :   vN = zero_Flm_copy(nz, nChar);
     621             : 
     622         798 :   av2 = avma;
     623         798 :   vB = const_vec(nz, gen_1);
     624         798 :   s0 = z? powgi(z, modii(gel(vt,1), den)): gen_1; /* for beta = 1 */
     625         798 :   s = const_vec(nChar, s0);
     626             : 
     627       53725 :   while ( (i = NextElt(&G)) )
     628             :   {
     629       52927 :     GEN b = gel(vB,i);
     630       52927 :     b = nfmuli(nf, b, gel(gen,i));
     631       52927 :     b = typ(b) == t_COL? FpC_red(b, condZ): modii(b, condZ);
     632      106190 :     for (j=1; j<=i; j++) gel(vB,j) = b;
     633             : 
     634      225155 :     for (ic = 1; ic <= nChar; ic++)
     635             :     {
     636      172228 :       GEN v = gel(vN,ic), n = gel(nchi,ic);
     637      172228 :       v[i] = Fl_add(v[i], n[i], lC[ic]->ord);
     638      172788 :       for (j=1; j<i; j++) v[j] = v[i];
     639             :     }
     640             : 
     641       52927 :     gel(vB,i) = b = set_sign_mod_divisor(nf, NULL, b, sarch);
     642       52927 :     if (!z)
     643           0 :       s0 = gen_1;
     644             :     else
     645             :     {
     646       52927 :       b = typ(b) == t_COL? ZV_dotproduct(vt, b): mulii(gel(vt,1),b);
     647       52927 :       s0 = powgi(z, modii(b,den));
     648             :     }
     649      225155 :     for (ic = 1; ic <= nChar; ic++)
     650             :     {
     651      172228 :       GEN v = gel(vN,ic), val = lC[ic]->val[ v[i] ];
     652      172228 :       gel(s,ic) = gadd(gel(s,ic), gmul(val, s0));
     653             :     }
     654             : 
     655       52927 :     if (gc_needed(av2, 1))
     656             :     {
     657          39 :       if (DEBUGMEM > 1) pari_warn(warnmem,"ArtinNumber");
     658          39 :       gerepileall(av2, 2, &s, &vB);
     659             :     }
     660             :   }
     661             : 
     662         798 :   classe = isprincipalray(bnr, idh);
     663         798 :   z = powIs(- (lg(gel(sarch,1))-1));
     664             : 
     665        2394 :   for (ic = 1; ic <= nChar; ic++)
     666             :   {
     667        1596 :     s0 = gmul(gel(s,ic), CHI_eval(lC[ic], classe));
     668        1596 :     gel(W, indW[ic]) = gmul(gdiv(s0, sqrtnc), z);
     669             :   }
     670         798 :   return W;
     671             : }
     672             : 
     673             : static GEN
     674         721 : AllArtinNumbers(GEN CR, long prec)
     675             : {
     676         721 :   pari_sp av = avma;
     677         721 :   GEN vChar = gel(CR,1), dataCR = gel(CR,2);
     678         721 :   long j, k, cl = lg(dataCR) - 1, J = lg(vChar)-1;
     679         721 :   GEN W = cgetg(cl+1,t_VEC), WbyCond, LCHI;
     680             : 
     681        1638 :   for (j = 1; j <= J; j++)
     682             :   {
     683         917 :     GEN LChar = gel(vChar,j), ldata = vecpermute(dataCR, LChar);
     684         917 :     GEN dtcr = gel(ldata,1), bnr = ch_bnr(dtcr);
     685         917 :     long l = lg(LChar);
     686             : 
     687         917 :     if (DEBUGLEVEL>1)
     688           0 :       err_printf("* Root Number: cond. no %ld/%ld (%ld chars)\n", j, J, l-1);
     689         917 :     LCHI = cgetg(l, t_VEC);
     690        3297 :     for (k = 1; k < l; k++) gel(LCHI,k) = ch_CHI0(gel(ldata,k));
     691         917 :     WbyCond = ArtinNumber(bnr, LCHI, prec);
     692        3297 :     for (k = 1; k < l; k++) gel(W,LChar[k]) = gel(WbyCond,k);
     693             :   }
     694         721 :   return gerepilecopy(av, W);
     695             : }
     696             : 
     697             : /* compute the constant W of the functional equation of
     698             :    Lambda(chi). If flag = 1 then chi is assumed to be primitive */
     699             : GEN
     700          77 : bnrrootnumber(GEN bnr, GEN chi, long flag, long prec)
     701             : {
     702          77 :   pari_sp av = avma;
     703             :   GEN cyc, W;
     704             : 
     705          77 :   if (flag < 0 || flag > 1) pari_err_FLAG("bnrrootnumber");
     706          77 :   checkbnr(bnr);
     707          77 :   if (flag)
     708             :   {
     709           0 :     cyc = bnr_get_cyc(bnr);
     710           0 :     if (!char_check(cyc,chi)) pari_err_TYPE("bnrrootnumber [character]", chi);
     711             :   }
     712             :   else
     713             :   {
     714          77 :     bnr_char_sanitize(&bnr, &chi);
     715          77 :     cyc = bnr_get_cyc(bnr);
     716             :   }
     717          77 :   chi = char_normalize(chi, cyc_normalize(cyc));
     718          77 :   chi = get_Char(chi, prec);
     719          77 :   W = ArtinNumber(bnr, mkvec(chi), prec);
     720          77 :   return gerepilecopy(av, gel(W,1));
     721             : }
     722             : 
     723             : /********************************************************************/
     724             : /*               3rd part: initialize the characters                */
     725             : /********************************************************************/
     726             : 
     727             : /* Let chi be a character, A(chi) corresponding to the primes dividing diff
     728             :  * at s = flag. If s = 0, returns [r, A] where r is the order of vanishing
     729             :  * at s = 0 corresponding to diff. */
     730             : static GEN
     731        2128 : AChi(GEN dtcr, long *r, long flag, long prec)
     732             : {
     733        2128 :   GEN A, diff = ch_diff(dtcr), bnrc = ch_bnr(dtcr), chi  = ch_CHI0(dtcr);
     734        2128 :   long i, l = lg(diff);
     735             : 
     736        2128 :   A = gen_1; *r = 0;
     737        2233 :   for (i = 1; i < l; i++)
     738             :   {
     739         105 :     GEN B, pr = gel(diff,i), z = CharEval(chi, isprincipalray(bnrc, pr));
     740         105 :     if (flag)
     741           0 :       B = gsubsg(1, gdiv(z, pr_norm(pr)));
     742         105 :     else if (gequal1(z))
     743             :     {
     744          21 :       B = glog(pr_norm(pr), prec);
     745          21 :       (*r)++;
     746             :     }
     747             :     else
     748          84 :       B = gsubsg(1, z);
     749         105 :     A = gmul(A, B);
     750             :   }
     751        2128 :   return A;
     752             : }
     753             : /* simplified version of Achi: return 1 if L(0,chi) = 0 */
     754             : static int
     755        1071 : L_vanishes_at_0(GEN D)
     756             : {
     757        1071 :   GEN diff = ch_diff(D), bnrc = ch_bnr(D), chi  = ch_CHI0(D);
     758        1071 :   long i, l = lg(diff);
     759        1113 :   for (i = 1; i < l; i++)
     760             :   {
     761          56 :     GEN pr = gel(diff,i);
     762          56 :     if (!CharEval_n(chi, isprincipalray(bnrc, pr))) return 1;
     763             :   }
     764        1057 :   return 0;
     765             : }
     766             : 
     767             : static GEN
     768         539 : _data3(GEN arch, long r2)
     769             : {
     770         539 :   GEN z = cgetg(4, t_VECSMALL);
     771         539 :   long i, r1 = lg(arch) - 1, q = 0;
     772        1722 :   for (i = 1; i <= r1; i++) if (signe(gel(arch,i))) q++;
     773         539 :   z[1] = q;
     774         539 :   z[2] = r1 - q;
     775         539 :   z[3] = r2; return z;
     776             : }
     777             : static void
     778        1603 : ch_get3(GEN dtcr, long *a, long *b, long *c)
     779        1603 : { GEN v = ch_3(dtcr); *a = v[1]; *b = v[2]; *c = v[3]; }
     780             : /* 2^(2*r2) pi^N */
     781             : static GEN
     782         714 : get_P(GEN nf, long prec)
     783             : {
     784         714 :   long r2 = nf_get_r2(nf), N = nf_get_degree(nf);
     785         714 :   GEN P = powru(mppi(prec), N); shiftr_inplace(P, 2*r2); return P;
     786             : }
     787             : /* sort chars according to conductor */
     788             : static GEN
     789         392 : sortChars(GEN ch)
     790             : {
     791         392 :   long j, l = lg(ch);
     792         392 :   GEN F = cgetg(l, t_VEC);
     793        1701 :   for (j = 1; j < l; j++) gel(F, j) = gmael(ch,j,2);
     794         392 :   return vec_equiv(F);
     795             : }
     796             : 
     797             : /* Given a list [chi, F = cond(chi)] of characters over Cl(bnr), return
     798             :  * [vChar, dataCR], where vChar contains the equivalence classes of
     799             :  * characters with the same conductor, and dataCR contains for each character:
     800             :  * - bnr(F)
     801             :  * - the constant C(F) [t_REAL]
     802             :  * - [q, r1 - q, r2, rc] where
     803             :  *      q = number of real places in F
     804             :  *      rc = max{r1 + r2 - q + 1, r2 + q}
     805             :  * - diff(chi) primes dividing m but not F
     806             :  * - chi in bnr(m)
     807             :  * - chi in bnr(F).
     808             :  * If all is unset, only compute characters s.t. L(chi,0) != 0 */
     809             : static GEN
     810         392 : InitChar(GEN bnr, GEN ch, long all, long prec)
     811             : {
     812         392 :   GEN bnf = checkbnf(bnr), nf = bnf_get_nf(bnf), mod = bnr_get_mod(bnr);
     813         392 :   GEN P, dataCR, ncyc, dk = nf_get_disc(nf), vChar = sortChars(ch);
     814         392 :   long n, l, r2 = nf_get_r2(nf), prec2 = precdbl(prec) + EXTRAPREC64;
     815         392 :   long lv = lg(vChar);
     816             : 
     817         392 :   P = get_P(nf, prec2);
     818         392 :   ncyc = cyc_normalize(bnr_get_cyc(bnr));
     819             : 
     820         392 :   dataCR = cgetg_copy(ch, &l);
     821         931 :   for (n = 1; n < lv; n++)
     822             :   {
     823         539 :     GEN D, bnrc, v = gel(vChar, n); /* indices of chars of given cond */
     824         539 :     long a, i = v[1], lc = lg(v);
     825         539 :     GEN F = gmael(ch,i,2);
     826         539 :     gel(dataCR, i) = D = cgetg(8, t_VEC);
     827         539 :     ch_C(D) = sqrtr_abs(divir(mulii(dk, ZM_det_triangular(gel(F,1))), P));
     828         539 :     ch_3(D) = _data3(gel(F,2), r2);
     829         539 :     if (gequal(F, mod))
     830             :     {
     831         378 :       ch_bnr(D) = bnrc = bnr;
     832         378 :       ch_diff(D) = cgetg(1, t_VEC);
     833             :     }
     834             :     else
     835             :     {
     836         161 :       ch_bnr(D) = bnrc = Buchray(bnf, F, nf_INIT);
     837         161 :       ch_diff(D) = get_prdiff(divcond(bnr), divcond(bnrc));
     838             :     }
     839        1848 :     for (a = 1; a < lc; a++)
     840             :     {
     841        1309 :       long i = v[a];
     842        1309 :       GEN chi = gmael(ch,i,1);
     843             : 
     844        1309 :       if (a > 1) gel(dataCR, i) = D = leafcopy(D);
     845        1309 :       chi = char_normalize(chi,ncyc);
     846        1309 :       ch_CHI(D) = get_Char(chi, prec2);
     847        1309 :       if (bnrc == bnr)
     848        1092 :         ch_CHI0(D) = ch_CHI(D);
     849             :       else
     850             :       {
     851         217 :         chi = bnrchar_primitive(bnr, chi, bnrc);
     852         217 :         ch_CHI0(D) = get_Char(chi, prec2);
     853             :       }
     854             :       /* set last */
     855        1309 :       ch_small(D) = mkvecsmall2(all || !L_vanishes_at_0(D),
     856        1309 :                                 eulerphiu(itou(gel(chi,1))));
     857             :     }
     858             :   }
     859         392 :   return mkvec2(vChar, dataCR);
     860             : }
     861             : 
     862             : /* recompute dataCR with the new precision, in place. Don't modify bnr
     863             :  * components in place  */
     864             : static void
     865         322 : CharNewPrec(GEN data, long prec)
     866             : {
     867         322 :   long j, l, prec2 = precdbl(prec) + EXTRAPREC64;
     868         322 :   GEN P, nf, dataCR = gmael(data,4,2), D = gel(dataCR,1);
     869             : 
     870         322 :   if (ch_prec(D) >= prec2) return;
     871         322 :   nf = bnr_get_nf(ch_bnr(D));
     872         322 :   if (nf_get_prec(nf) < prec) nf = nfnewprec_shallow(nf, prec); /* not prec2 */
     873         322 :   P = get_P(nf, prec2); l = lg(dataCR);
     874        1351 :   for (j = 1; j < l; j++)
     875             :   {
     876             :     GEN f0, bnr;
     877        1029 :     D = gel(dataCR,j); f0 = gel(bnr_get_mod(ch_bnr(D)), 1);
     878        1029 :     ch_C(D) = sqrtr_abs(divir(mulii(nf_get_disc(nf),ZM_det_triangular(f0)), P));
     879        1029 :     ch_bnr(D) = bnr = shallowcopy(ch_bnr(D));
     880        1029 :     gel(bnr,1) = shallowcopy(gel(bnr,1)); gmael(bnr,1,7) = nf;
     881        1029 :     ch_CHI(D) = get_Char(gel(ch_CHI(D),1), prec2);
     882        1029 :     ch_CHI0(D)= get_Char(gel(ch_CHI0(D),1), prec2);
     883             :   }
     884             : }
     885             : 
     886             : /********************************************************************/
     887             : /*             4th part: compute the coefficients an(chi)           */
     888             : /*                                                                  */
     889             : /* matan entries are arrays of ints containing the coefficients of  */
     890             : /* an(chi) as a polmod modulo polcyclo(order(chi))                     */
     891             : /********************************************************************/
     892             : 
     893             : static void
     894     1001821 : _0toCoeff(int *rep, long deg)
     895             : {
     896             :   long i;
     897     4583479 :   for (i=0; i<deg; i++) rep[i] = 0;
     898     1001821 : }
     899             : 
     900             : /* transform a polmod into Coeff */
     901             : static void
     902      397879 : Polmod2Coeff(int *rep, GEN polmod, long deg)
     903             : {
     904             :   long i;
     905      397879 :   if (typ(polmod) == t_POLMOD)
     906             :   {
     907      293467 :     GEN pol = gel(polmod,2);
     908      293467 :     long d = degpol(pol);
     909             : 
     910      293467 :     pol += 2;
     911     1266018 :     for (i=0; i<=d; i++) rep[i] = itos(gel(pol,i));
     912      687949 :     for (   ; i<deg; i++) rep[i] = 0;
     913             :   }
     914             :   else
     915             :   {
     916      104412 :     rep[0] = itos(polmod);
     917      112644 :     for (i=1; i<deg; i++) rep[i] = 0;
     918             :   }
     919      397879 : }
     920             : 
     921             : /* initialize a deg * n matrix of ints */
     922             : static int**
     923        4165 : InitMatAn(long n, long deg, long flag)
     924             : {
     925             :   long i, j;
     926        4165 :   int *a, **A = (int**)pari_malloc((n+1)*sizeof(int*));
     927        4165 :   A[0] = NULL;
     928     6155522 :   for (i = 1; i <= n; i++)
     929             :   {
     930     6151357 :     a = (int*)pari_malloc(deg*sizeof(int));
     931     6151357 :     A[i] = a; a[0] = (i == 1 || flag);
     932    19974675 :     for (j = 1; j < deg; j++) a[j] = 0;
     933             :   }
     934        4165 :   return A;
     935             : }
     936             : 
     937             : static void
     938        6531 : FreeMat(int **A, long n)
     939             : {
     940             :   long i;
     941     6170852 :   for (i = 0; i <= n; i++)
     942     6164321 :     if (A[i]) pari_free((void*)A[i]);
     943        6531 :   pari_free((void*)A);
     944        6531 : }
     945             : 
     946             : /* initialize Coeff reduction */
     947             : static int**
     948        2366 : InitReduction(long d, long deg)
     949             : {
     950             :   long j;
     951        2366 :   pari_sp av = avma;
     952             :   int **A;
     953             :   GEN polmod, pol;
     954             : 
     955        2366 :   A   = (int**)pari_malloc(deg*sizeof(int*));
     956        2366 :   pol = polcyclo(d, 0);
     957       11165 :   for (j = 0; j < deg; j++)
     958             :   {
     959        8799 :     A[j] = (int*)pari_malloc(deg*sizeof(int));
     960        8799 :     polmod = gmodulo(pol_xn(deg+j, 0), pol);
     961        8799 :     Polmod2Coeff(A[j], polmod, deg);
     962             :   }
     963             : 
     964        2366 :   set_avma(av); return A;
     965             : }
     966             : 
     967             : #if 0
     968             : void
     969             : pan(int **an, long n, long deg)
     970             : {
     971             :   long i,j;
     972             :   for (i = 1; i <= n; i++)
     973             :   {
     974             :     err_printf("n = %ld: ",i);
     975             :     for (j = 0; j < deg; j++) err_printf("%d ",an[i][j]);
     976             :     err_printf("\n");
     977             :   }
     978             : }
     979             : #endif
     980             : 
     981             : /* returns 0 if c is zero, 1 otherwise. */
     982             : static int
     983     7661729 : IsZero(int* c, long deg)
     984             : {
     985             :   long i;
     986    26145834 :   for (i = 0; i < deg; i++)
     987    20449437 :     if (c[i]) return 0;
     988     5696397 :   return 1;
     989             : }
     990             : 
     991             : /* set c0 <-- c0 * c1 */
     992             : static void
     993     1511799 : MulCoeff(int *c0, int* c1, int** reduc, long deg)
     994             : {
     995             :   long i,j;
     996             :   int c, *T;
     997             : 
     998     1511799 :   if (IsZero(c0,deg)) return;
     999             : 
    1000      810549 :   T = (int*)new_chunk(2*deg);
    1001     8556301 :   for (i = 0; i < 2*deg; i++)
    1002             :   {
    1003     7745752 :     c = 0;
    1004    84469968 :     for (j = 0; j <= i; j++)
    1005    76724216 :       if (j < deg && j > i - deg) c += c0[j] * c1[i-j];
    1006     7745752 :     T[i] = c;
    1007             :   }
    1008     4683425 :   for (i = 0; i < deg; i++)
    1009             :   {
    1010     3872876 :     c = T[i];
    1011    40298546 :     for (j = 0; j < deg; j++) c += reduc[j][i] * T[deg+j];
    1012     3872876 :     c0[i] = c;
    1013             :   }
    1014             : }
    1015             : 
    1016             : /* c0 <- c0 + c1 * c2 */
    1017             : static void
    1018     6149930 : AddMulCoeff(int *c0, int *c1, int* c2, int** reduc, long deg)
    1019             : {
    1020             :   long i, j;
    1021             :   pari_sp av;
    1022             :   int c, *t;
    1023             : 
    1024     6149930 :   if (IsZero(c2,deg)) return;
    1025     1154783 :   if (!c1) /* c1 == 1 */
    1026             :   {
    1027      709114 :     for (i = 0; i < deg; i++) c0[i] += c2[i];
    1028      291571 :     return;
    1029             :   }
    1030      863212 :   av = avma;
    1031      863212 :   t = (int*)new_chunk(2*deg); /* = c1 * c2, not reduced */
    1032     6691468 :   for (i = 0; i < 2*deg; i++)
    1033             :   {
    1034     5828256 :     c = 0;
    1035    42776216 :     for (j = 0; j <= i; j++)
    1036    36947960 :       if (j < deg && j > i - deg) c += c1[j] * c2[i-j];
    1037     5828256 :     t[i] = c;
    1038             :   }
    1039     3777340 :   for (i = 0; i < deg; i++)
    1040             :   {
    1041     2914128 :     c = t[i];
    1042    19931044 :     for (j = 0; j < deg; j++) c += reduc[j][i] * t[deg+j];
    1043     2914128 :     c0[i] += c;
    1044             :   }
    1045      863212 :   set_avma(av);
    1046             : }
    1047             : 
    1048             : /* evaluate the Coeff. No Garbage collector */
    1049             : static GEN
    1050     3661174 : EvalCoeff(GEN z, int* c, long deg)
    1051             : {
    1052             :   long i,j;
    1053     3661174 :   GEN r, e = NULL;
    1054             : 
    1055             : #if 0
    1056             :   /* standard Horner */
    1057             :   e = stoi(c[deg - 1]);
    1058             :   for (i = deg - 2; i >= 0; i--)
    1059             :     e = gadd(stoi(c[i]), gmul(z, e));
    1060             : #else
    1061             :   /* specific attention to sparse polynomials */
    1062     5260114 :   for (i = deg-1; i >=0; i=j-1)
    1063             :   {
    1064    12405790 :     for (j=i; c[j] == 0; j--)
    1065    10806850 :       if (j==0)
    1066             :       {
    1067     3000548 :         if (!e) return NULL;
    1068      311077 :         if (i!=j) z = gpowgs(z,i-j+1);
    1069      311077 :         return gmul(e,z);
    1070             :       }
    1071     1598940 :     if (e)
    1072             :     {
    1073      627237 :       r = (i==j)? z: gpowgs(z,i-j+1);
    1074      627237 :       e = gadd(gmul(e,r), stoi(c[j]));
    1075             :     }
    1076             :     else
    1077      971703 :       e = stoi(c[j]);
    1078             :   }
    1079             : #endif
    1080      660626 :   return e;
    1081             : }
    1082             : 
    1083             : /* a2 <- copy the n x m array */
    1084             : static void
    1085      357308 : CopyCoeff(int** a, int** a2, long n, long m)
    1086             : {
    1087             :   long i,j;
    1088             : 
    1089     5261912 :   for (i = 1; i <= n; i++)
    1090             :   {
    1091     4904604 :     int *b = a[i], *b2 = a2[i];
    1092    19404982 :     for (j = 0; j < m; j++) b2[j] = b[j];
    1093             :   }
    1094      357308 : }
    1095             : 
    1096             : static void
    1097      357308 : an_AddMul(int **an,int **an2, long np, long n, long deg, GEN chi, int **reduc)
    1098             : {
    1099      357308 :   GEN chi2 = chi;
    1100             :   long q, qk, k;
    1101      357308 :   int *c, *c2 = (int*)new_chunk(deg);
    1102             : 
    1103      357308 :   CopyCoeff(an, an2, n/np, deg);
    1104      357308 :   for (q=np;;)
    1105             :   {
    1106      391202 :     if (gequal1(chi2)) c = NULL; else { Polmod2Coeff(c2, chi2, deg); c = c2; }
    1107     6541132 :     for(k = 1, qk = q; qk <= n; k++, qk += q)
    1108     6149930 :       AddMulCoeff(an[qk], c, an2[k], reduc, deg);
    1109      391202 :     if (! (q = umuluu_le(q,np, n)) ) break;
    1110             : 
    1111       33894 :     chi2 = gmul(chi2, chi);
    1112             :   }
    1113      357308 : }
    1114             : 
    1115             : /* correct the coefficients an(chi) according with diff(chi) in place */
    1116             : static void
    1117        2366 : CorrectCoeff(GEN dtcr, int** an, int** reduc, long n, long deg)
    1118             : {
    1119        2366 :   pari_sp av = avma;
    1120             :   long lg, j;
    1121             :   pari_sp av1;
    1122             :   int **an2;
    1123             :   GEN bnrc, diff;
    1124             :   CHI_t C;
    1125             : 
    1126        2366 :   diff = ch_diff(dtcr); lg = lg(diff) - 1;
    1127        2366 :   if (!lg) return;
    1128             : 
    1129         168 :   if (DEBUGLEVEL>2) err_printf("diff(CHI) = %Ps", diff);
    1130         168 :   bnrc = ch_bnr(dtcr);
    1131         168 :   init_CHI_alg(&C, ch_CHI0(dtcr));
    1132             : 
    1133         168 :   an2 = InitMatAn(n, deg, 0);
    1134         168 :   av1 = avma;
    1135         364 :   for (j = 1; j <= lg; j++)
    1136             :   {
    1137         196 :     GEN pr = gel(diff,j);
    1138         196 :     long Np = upr_norm(pr);
    1139         196 :     GEN chi  = CHI_eval(&C, isprincipalray(bnrc, pr));
    1140         196 :     an_AddMul(an,an2,Np,n,deg,chi,reduc);
    1141         196 :     set_avma(av1);
    1142             :   }
    1143         168 :   FreeMat(an2, n); set_avma(av);
    1144             : }
    1145             : 
    1146             : /* compute the coefficients an in the general case */
    1147             : static int**
    1148        1631 : ComputeCoeff(GEN dtcr, LISTray *R, long n, long deg)
    1149             : {
    1150        1631 :   pari_sp av = avma, av2;
    1151             :   long i, l;
    1152             :   int **an, **reduc, **an2;
    1153             :   GEN L;
    1154             :   CHI_t C;
    1155             : 
    1156        1631 :   init_CHI_alg(&C, ch_CHI(dtcr));
    1157        1631 :   an  = InitMatAn(n, deg, 0);
    1158        1631 :   an2 = InitMatAn(n, deg, 0);
    1159        1631 :   reduc  = InitReduction(C.ord, deg);
    1160        1631 :   av2 = avma;
    1161             : 
    1162        1631 :   L = R->L1; l = lg(L);
    1163      358743 :   for (i=1; i<l; i++, set_avma(av2))
    1164             :   {
    1165      357112 :     long np = L[i];
    1166      357112 :     GEN chi  = CHI_eval(&C, gel(R->L1ray,i));
    1167      357112 :     an_AddMul(an,an2,np,n,deg,chi,reduc);
    1168             :   }
    1169        1631 :   FreeMat(an2, n);
    1170             : 
    1171        1631 :   CorrectCoeff(dtcr, an, reduc, n, deg);
    1172        1631 :   FreeMat(reduc, deg-1);
    1173        1631 :   set_avma(av); return an;
    1174             : }
    1175             : 
    1176             : /********************************************************************/
    1177             : /*              5th part: compute L-functions at s=1                */
    1178             : /********************************************************************/
    1179             : static void
    1180         441 : deg11(LISTray *R, long p, GEN bnr, GEN pr) {
    1181         441 :   vecsmalltrunc_append(R->L1, p);
    1182         441 :   vectrunc_append(R->L1ray, isprincipalray(bnr, pr));
    1183         441 : }
    1184             : static void
    1185       24659 : deg12(LISTray *R, long p, GEN bnr, GEN pr) {
    1186       24659 :   vecsmalltrunc_append(R->L11, p);
    1187       24659 :   vectrunc_append(R->L11ray, isprincipalray(bnr, pr));
    1188       24659 : }
    1189             : static void
    1190          35 : deg0(LISTray *R, long p) { vecsmalltrunc_append(R->L0, p); }
    1191             : static void
    1192       26629 : deg2(LISTray *R, long p) { vecsmalltrunc_append(R->L2, p); }
    1193             : 
    1194             : static void
    1195         203 : InitPrimesQuad(GEN bnr, ulong N0, LISTray *R)
    1196             : {
    1197         203 :   pari_sp av = avma;
    1198         203 :   GEN bnf = bnr_get_bnf(bnr), F = gel(bnr_get_mod(bnr), 1);
    1199         203 :   GEN v, N, nf = bnf_get_nf(bnf), dk = nf_get_disc(nf);
    1200         203 :   long l = 1 + primepi_upper_bound(N0);
    1201         203 :   ulong i, p, FZ = itou(gcoeff(F,1,1)), FZ2 = itou(gcoeff(F,2,2));
    1202             :   forprime_t T;
    1203             : 
    1204         203 :   FZ2 = ugcd(FZ, FZ2); /* content(F) */
    1205         203 :   R->L0 = vecsmalltrunc_init(l);
    1206         203 :   R->L2 = vecsmalltrunc_init(l); R->condZ = FZ;
    1207         203 :   R->L1 = vecsmalltrunc_init(l); R->L1ray = vectrunc_init(l);
    1208         203 :   R->L11= vecsmalltrunc_init(l); R->L11ray= vectrunc_init(l);
    1209         203 :   N = utoipos(2);
    1210         203 :   u_forprime_init(&T, 2, N0);
    1211       51967 :   while ( (p = u_forprime_next(&T)) )
    1212             :   {
    1213       51764 :     N[2] = p;
    1214       51764 :     switch (kroiu(dk, p))
    1215             :     {
    1216       26643 :     case -1: /* inert */
    1217       26643 :       if (FZ % p == 0) deg0(R,p); else deg2(R,p);
    1218       26643 :       break;
    1219       24848 :     case 1: /* split */
    1220       24848 :       if (FZ2 % p == 0) deg0(R,p);
    1221             :       else
    1222             :       {
    1223       24848 :         GEN Lpr = idealprimedec(nf, N);
    1224       24848 :         if (FZ % p) deg12(R, p, bnr, gel(Lpr,1));
    1225             :         else
    1226             :         {
    1227         189 :           long t = ZC_prdvd(gel(F,2), gel(Lpr,1))? 2: 1;
    1228         189 :           deg11(R, p, bnr, gel(Lpr,t));
    1229             :         }
    1230             :       }
    1231       24848 :       break;
    1232         273 :     default: /* ramified */
    1233         273 :       if (FZ % p == 0)
    1234          21 :         deg0(R,p);
    1235             :       else
    1236         252 :         deg11(R, p, bnr, idealprimedec_galois(nf, N));
    1237         273 :       break;
    1238             :     }
    1239             :   }
    1240             :   /* precompute isprincipalray(x), x in Z */
    1241         203 :   v = coprimes_zv(FZ);
    1242         203 :   R->rayZ = cgetg(FZ, t_VEC);
    1243        3934 :   for (i = 1; i < FZ; i++)
    1244             :   {
    1245        3731 :     N[2] = i;
    1246        3731 :     gel(R->rayZ,i) = v[i]? isprincipalray(bnr, N): gen_0;
    1247             :   }
    1248         203 :   gerepileall(av, 7, &(R->L0), &(R->L2), &(R->rayZ),
    1249             :               &(R->L1), &(R->L1ray), &(R->L11), &(R->L11ray));
    1250         203 : }
    1251             : 
    1252             : static void
    1253         518 : InitPrimes(GEN bnr, ulong N0, LISTray *R)
    1254             : {
    1255         518 :   GEN bnf = bnr_get_bnf(bnr), cond = gel(bnr_get_mod(bnr), 1);
    1256         518 :   long p, l, condZ, N = lg(cond)-1;
    1257         518 :   GEN DL, prime, BOUND, nf = bnf_get_nf(bnf);
    1258             :   forprime_t T;
    1259             : 
    1260         518 :   R->condZ = condZ = itos(gcoeff(cond,1,1));
    1261         518 :   l = primepi_upper_bound(N0) * N;
    1262         518 :   DL = cgetg(N+1, t_VEC);
    1263         518 :   R->L1 = vecsmalltrunc_init(l);
    1264         518 :   R->L1ray = vectrunc_init(l);
    1265         518 :   u_forprime_init(&T, 2, N0);
    1266         518 :   prime = utoipos(2);
    1267         518 :   BOUND = utoi(N0);
    1268      123144 :   while ( (p = u_forprime_next(&T)) )
    1269             :   {
    1270      122626 :     pari_sp av = avma;
    1271             :     long j, k, lP;
    1272             :     GEN P;
    1273      122626 :     prime[2] = p;
    1274      122626 :     if (DEBUGLEVEL>1 && (p & 2047) == 1) err_printf("%ld ", p);
    1275      122626 :     P = idealprimedec_limit_norm(nf, prime, BOUND); lP = lg(P);
    1276      244713 :     for (j = 1; j < lP; j++)
    1277             :     {
    1278      122087 :       GEN pr  = gel(P,j), dl = NULL;
    1279      122087 :       if (condZ % p || !idealval(nf, cond, pr))
    1280             :       {
    1281      121611 :         dl = gclone( isprincipalray(bnr, pr) );
    1282      121611 :         vecsmalltrunc_append(R->L1, upowuu(p, pr_get_f(pr)));
    1283             :       }
    1284      122087 :       gel(DL,j) = dl;
    1285             :     }
    1286      122626 :     set_avma(av);
    1287      244713 :     for (k = 1; k < j; k++)
    1288             :     {
    1289      122087 :       if (!DL[k]) continue;
    1290      121611 :       vectrunc_append(R->L1ray, ZC_copy(gel(DL,k)));
    1291      121611 :       gunclone(gel(DL,k));
    1292             :     }
    1293             :   }
    1294         518 : }
    1295             : 
    1296             : static GEN /* cf polcoef */
    1297      412236 : _sercoeff(GEN x, long n)
    1298             : {
    1299      412236 :   long i = n - valser(x);
    1300      412236 :   return (i < 0)? gen_0: gel(x,i+2);
    1301             : }
    1302             : 
    1303             : static void
    1304      412236 : affect_coeff(GEN q, long n, GEN y, long t)
    1305             : {
    1306      412236 :   GEN x = _sercoeff(q,-n);
    1307      412236 :   if (x == gen_0) gel(y,n) = NULL;
    1308      210548 :   else { affgr(x, gel(y,n)); shiftr_inplace(gel(y,n), t); }
    1309      412236 : }
    1310             : /* (x-i)(x-(i+1)) */
    1311             : static GEN
    1312      105282 : d2(long i) { return deg2pol_shallow(gen_1, utoineg(2*i+1), muluu(i,i+1), 0); }
    1313             : /* x-i */
    1314             : static GEN
    1315      315902 : d1(long i) { return deg1pol_shallow(gen_1, stoi(-i), 0); }
    1316             : 
    1317             : typedef struct {
    1318             :   GEN c1, aij, bij, cS, cT, powracpi;
    1319             :   long i0, a,b,c, r, rc1, rc2;
    1320             : } ST_t;
    1321             : 
    1322             : /* compute the principal part at the integers s = 0, -1, -2, ..., -i0
    1323             :  * of Gamma((s+1)/2)^a Gamma(s/2)^b Gamma(s)^c / (s - z) with z = 0 and 1 */
    1324             : static void
    1325         308 : ppgamma(ST_t *T, long prec)
    1326             : {
    1327             :   GEN G, G1, G2, A, E, O, x, sqpi, aij, bij;
    1328         308 :   long c = T->c, r = T->r, i0 = T->i0, i, j, s, t, dx;
    1329             :   pari_sp av;
    1330             : 
    1331         308 :   T->aij = aij = cgetg(i0+1, t_VEC);
    1332         308 :   T->bij = bij = cgetg(i0+1, t_VEC);
    1333      105590 :   for (i = 1; i <= i0; i++)
    1334             :   {
    1335             :     GEN p1, p2;
    1336      105282 :     gel(aij,i) = p1 = cgetg(r+1, t_VEC);
    1337      105282 :     gel(bij,i) = p2 = cgetg(r+1, t_VEC);
    1338      311400 :     for (j=1; j<=r; j++) { gel(p1,j) = cgetr(prec); gel(p2,j) = cgetr(prec); }
    1339             :   }
    1340         308 :   av = avma; x = pol_x(0);
    1341         308 :   sqpi = sqrtr_abs(mppi(prec)); /* Gamma(1/2) */
    1342             : 
    1343         308 :   G1 = gexp(integser(psi1series(r-1, 0, prec)), prec); /* Gamma(1 + x) */
    1344         308 :   G = shallowcopy(G1); setvalser(G,-1); /* Gamma(x) */
    1345             : 
    1346             :   /* expansion of log(Gamma(u) / Gamma(1/2)) at u = 1/2 */
    1347         308 :   G2 = cgetg(r+2, t_SER);
    1348         308 :   G2[1] = evalsigne(1) | _evalvalser(1) | evalvarn(0);
    1349         308 :   gel(G2,2) = gneg(gadd(gmul2n(mplog2(prec), 1), mpeuler(prec)));
    1350         644 :   for (i = 1; i < r; i++) gel(G2,i+2) = mulri(gel(G1,i+2), int2um1(i));
    1351         308 :   G2 = gmul(sqpi, gexp(G2, prec)); /* Gamma(1/2 + x) */
    1352             : 
    1353             :  /* We simplify to get one of the following two expressions
    1354             :   * if (b > a) : sqrt(Pi)^a 2^{a-au} Gamma(u)^{a+c} Gamma(  u/2  )^{|b-a|}
    1355             :   * if (b <= a): sqrt(Pi)^b 2^{b-bu} Gamma(u)^{b+c} Gamma((u+1)/2)^{|b-a|} */
    1356         308 :   if (T->b > T->a)
    1357             :   {
    1358          56 :     t = T->a; s = T->b; dx = 1;
    1359          56 :     E = ser_unscale(G, ghalf);
    1360          56 :     O = gmul2n(gdiv(ser_unscale(G2, ghalf), d1(1)), 1); /* Gamma((x-1)/2) */
    1361             :   }
    1362             :   else
    1363             :   {
    1364         252 :     t = T->b; s = T->a; dx = 0;
    1365         252 :     E = ser_unscale(G2, ghalf);
    1366         252 :     O = ser_unscale(G, ghalf);
    1367             :   }
    1368             :   /* (sqrt(Pi) 2^{1-x})^t Gamma(x)^{t+c} */
    1369         308 :   A = gmul(gmul(powru(gmul2n(sqpi,1), t), gpowgs(G, t+c)),
    1370             :            gpow(gen_2, RgX_to_ser(gmulgs(x,-t), r+2), prec));
    1371             :   /* A * Gamma((x - dx + 1)/2)^{s-t} */
    1372         308 :   E = gmul(A, gpowgs(E, s-t));
    1373             :   /* A * Gamma((x - dx)/2)^{s-t} */
    1374         308 :   O = gdiv(gmul(A, gpowgs(O, s-t)), gpowgs(gsubgs(x, 1), t+c));
    1375       52949 :   for (i = 0; i < i0/2; i++)
    1376             :   {
    1377       52641 :     GEN p1, q1, A1 = gel(aij,2*i+1), B1 = gel(bij,2*i+1);
    1378       52641 :     GEN p2, q2, A2 = gel(aij,2*i+2), B2 = gel(bij,2*i+2);
    1379       52641 :     long t1 = i * (s+t), t2 = t1 + t;
    1380             : 
    1381       52641 :     p1 = gdiv(E, d1(2*i));
    1382       52641 :     q1 = gdiv(E, d1(2*i+1));
    1383       52641 :     p2 = gdiv(O, d1(2*i+1));
    1384       52641 :     q2 = gdiv(O, d1(2*i+2));
    1385      155700 :     for (j = 1; j <= r; j++)
    1386             :     {
    1387      103059 :       affect_coeff(p1, j, A1, t1); affect_coeff(q1, j, B1, t1);
    1388      103059 :       affect_coeff(p2, j, A2, t2); affect_coeff(q2, j, B2, t2);
    1389             :     }
    1390       52641 :     E = gdiv(E, gmul(gpowgs(d1(2*i+1+dx), s-t), gpowgs(d2(2*i+1), t+c)));
    1391       52641 :     O = gdiv(O, gmul(gpowgs(d1(2*i+2+dx), s-t), gpowgs(d2(2*i+2), t+c)));
    1392             :   }
    1393         308 :   set_avma(av);
    1394         308 : }
    1395             : 
    1396             : /* chi != 1. Return L(1, chi) if fl & 1, else [r, c] where L(s, chi) ~ c s^r
    1397             :  * at s = 0. */
    1398             : static GEN
    1399        1295 : GetValue(GEN dtcr, GEN W, GEN S, GEN T, long fl, long prec)
    1400             : {
    1401             :   GEN cf, z;
    1402             :   long q, b, c, r;
    1403        1295 :   int isreal = (ch_deg(dtcr) <= 2);
    1404             : 
    1405        1295 :   ch_get3(dtcr, &q, &b, &c);
    1406        1295 :   if (fl & 1)
    1407             :   { /* S(chi) + W(chi).T(chi)) / (C(chi) sqrt(Pi)^{r1 - q}) */
    1408         196 :     cf = gmul(ch_C(dtcr), powruhalf(mppi(prec), b));
    1409             : 
    1410         196 :     z = gadd(S, gmul(W, T));
    1411         196 :     if (isreal) z = real_i(z);
    1412         196 :     z = gdiv(z, cf);
    1413         196 :     if (fl & 2) z = gmul(z, AChi(dtcr, &r, 1, prec));
    1414             :   }
    1415             :   else
    1416             :   { /* (W(chi).S(conj(chi)) + T(chi)) / (sqrt(Pi)^q 2^{r1 - q}) */
    1417        1099 :     cf = gmul2n(powruhalf(mppi(prec), q), b);
    1418             : 
    1419        1099 :     z = gadd(gmul(W, conj_i(S)), conj_i(T));
    1420        1099 :     if (isreal) z = real_i(z);
    1421        1099 :     z = gdiv(z, cf); r = 0;
    1422        1099 :     if (fl & 2) z = gmul(z, AChi(dtcr, &r, 0, prec));
    1423        1099 :     z = mkvec2(utoi(b + c + r), z);
    1424             :   }
    1425        1295 :   return z;
    1426             : }
    1427             : 
    1428             : /* return the order and the first nonzero term of L(s, chi0)
    1429             :    at s = 0. If flag != 0, adjust the value to get L_S(s, chi0). */
    1430             : static GEN
    1431          35 : GetValue1(GEN bnr, long flag, long prec)
    1432             : {
    1433          35 :   GEN bnf = bnr_get_bnf(bnr), nf = bnf_get_nf(bnf);
    1434          35 :   GEN h = bnf_get_no(bnf), R = bnf_get_reg(bnf);
    1435          35 :   GEN c = gdivgs(mpmul(h, R), -bnf_get_tuN(bnf));
    1436          35 :   long r = lg(nf_get_roots(nf)) - 2; /* r1 + r2 - 1 */;
    1437          35 :   if (flag)
    1438             :   {
    1439           0 :     GEN diff = divcond(bnr);
    1440           0 :     long i, l = lg(diff);
    1441           0 :     r += l - 1;
    1442           0 :     for (i = 1; i < l; i++) c = gmul(c, glog(pr_norm(gel(diff,i)), prec));
    1443             :   }
    1444          35 :   return mkvec2(utoi(r), c);
    1445             : }
    1446             : 
    1447             : /********************************************************************/
    1448             : /*                6th part: recover the coefficients                */
    1449             : /********************************************************************/
    1450             : static long
    1451        2564 : TestOne(GEN plg, RC_data *d)
    1452             : {
    1453        2564 :   long j, v = d->v;
    1454        2564 :   GEN z = gsub(d->beta, gel(plg,v));
    1455        2564 :   if (expo(z) >= d->G) return 0;
    1456        7036 :   for (j = 1; j < lg(plg); j++)
    1457        4902 :     if (j != v && mpcmp(d->B, mpabs_shallow(gel(plg,j))) < 0) return 0;
    1458        2134 :   return 1;
    1459             : }
    1460             : 
    1461             : static GEN
    1462         412 : chk_reccoeff_init(FP_chk_fun *chk, GEN r, GEN mat)
    1463             : {
    1464         412 :   RC_data *d = (RC_data*)chk->data;
    1465         412 :   (void)r; d->U = mat; return d->nB;
    1466             : }
    1467             : 
    1468             : static GEN
    1469         390 : chk_reccoeff(void *data, GEN x)
    1470             : {
    1471         390 :   RC_data *d = (RC_data*)data;
    1472         390 :   GEN v = gmul(d->U, x), z = gel(v,1);
    1473             : 
    1474         390 :   if (!gequal1(z)) return NULL;
    1475         390 :   *++v = evaltyp(t_COL) | _evallg( lg(d->M) ); /* pop 1st elt */
    1476         390 :   if (TestOne(gmul(d->M, v), d)) return v;
    1477           0 :   return NULL;
    1478             : }
    1479             : 
    1480             : /* Using Cohen's method */
    1481             : static GEN
    1482         412 : RecCoeff3(GEN nf, RC_data *d, long prec)
    1483             : {
    1484             :   GEN A, M, nB, cand, p1, B2, C2, tB, beta2, nf2, Bd;
    1485         412 :   GEN beta = d->beta, B = d->B;
    1486         412 :   long N = d->N, v = d->v, e, BIG;
    1487         412 :   long i, j, k, ct = 0, prec2;
    1488         412 :   FP_chk_fun chk = { &chk_reccoeff, &chk_reccoeff_init, NULL, NULL, 0 };
    1489         412 :   chk.data = (void*)d;
    1490             : 
    1491         412 :   d->G = minss(-10, -prec >> 4);
    1492         412 :   BIG = maxss(32, -2*d->G);
    1493         412 :   tB  = sqrtnr(real2n(BIG-N,DEFAULTPREC), N-1);
    1494         412 :   Bd  = grndtoi(gmin_shallow(B, tB), &e);
    1495         412 :   if (e > 0) return NULL; /* failure */
    1496         412 :   Bd = addiu(Bd, 1);
    1497         412 :   prec2 = nbits2prec( expi(Bd) + 192 );
    1498         412 :   prec2 = maxss(precdbl(prec), prec2);
    1499         412 :   B2 = sqri(Bd);
    1500         412 :   C2 = shifti(B2, BIG<<1);
    1501             : 
    1502         412 : LABrcf: ct++;
    1503         412 :   beta2 = gprec_w(beta, prec2);
    1504         412 :   nf2 = nfnewprec_shallow(nf, prec2);
    1505         412 :   d->M = M = nf_get_M(nf2);
    1506             : 
    1507         412 :   A = cgetg(N+2, t_MAT);
    1508        1697 :   for (i = 1; i <= N+1; i++) gel(A,i) = cgetg(N+2, t_COL);
    1509             : 
    1510         412 :   gcoeff(A, 1, 1) = gadd(gmul(C2, gsqr(beta2)), B2);
    1511        1285 :   for (j = 2; j <= N+1; j++)
    1512             :   {
    1513         873 :     p1 = gmul(C2, gmul(gneg_i(beta2), gcoeff(M, v, j-1)));
    1514         873 :     gcoeff(A, 1, j) = gcoeff(A, j, 1) = p1;
    1515             :   }
    1516        1285 :   for (i = 2; i <= N+1; i++)
    1517        2256 :     for (j = i; j <= N+1; j++)
    1518             :     {
    1519        1383 :       p1 = gen_0;
    1520        4443 :       for (k = 1; k <= N; k++)
    1521             :       {
    1522        3060 :         GEN p2 = gmul(gcoeff(M, k, j-1), gcoeff(M, k, i-1));
    1523        3060 :         if (k == v) p2 = gmul(C2, p2);
    1524        3060 :         p1 = gadd(p1,p2);
    1525             :       }
    1526        1383 :       gcoeff(A, i, j) = gcoeff(A, j, i) = p1;
    1527             :     }
    1528             : 
    1529         412 :   nB = mului(N+1, B2);
    1530         412 :   d->nB = nB;
    1531         412 :   cand = fincke_pohst(A, nB, -1, prec2, &chk);
    1532             : 
    1533         412 :   if (!cand)
    1534             :   {
    1535           0 :     if (ct > 3) return NULL;
    1536           0 :     prec2 = precdbl(prec2);
    1537           0 :     if (DEBUGLEVEL>1) pari_warn(warnprec,"RecCoeff", prec2);
    1538           0 :     goto LABrcf;
    1539             :   }
    1540             : 
    1541         412 :   cand = gel(cand,1);
    1542         412 :   if (lg(cand) == 2) return gel(cand,1);
    1543             : 
    1544         217 :   if (DEBUGLEVEL>1) err_printf("RecCoeff3: no solution found!\n");
    1545         217 :   return NULL;
    1546             : }
    1547             : 
    1548             : /* Using linear dependence relations */
    1549             : static GEN
    1550        2156 : RecCoeff2(GEN nf,  RC_data *d,  long prec)
    1551             : {
    1552             :   pari_sp av;
    1553        2156 :   GEN vec, M = nf_get_M(nf), beta = d->beta;
    1554        2156 :   long bit, min, max, lM = lg(M);
    1555             : 
    1556        2156 :   d->G = minss(-20, -prec >> 4);
    1557             : 
    1558        2156 :   vec  = vec_prepend(row(M, d->v), gneg(beta));
    1559        2156 :   min = (long)prec * 0.75;
    1560        2156 :   max = (long)prec * 0.98;
    1561        2156 :   av = avma;
    1562        2705 :   for (bit = max; bit >= min; bit-=32, set_avma(av))
    1563             :   {
    1564             :     long e;
    1565        2293 :     GEN v = lindep_bit(vec, bit), z = gel(v,1);
    1566        2293 :     if (!signe(z)) continue;
    1567        2174 :     *++v = evaltyp(t_COL) | _evallg(lM); /* pop 1st elt */
    1568        2174 :     v = grndtoi(gdiv(v, z), &e);
    1569        2174 :     if (e > 0) break;
    1570        2174 :     if (TestOne(RgM_RgC_mul(M, v), d)) return v;
    1571             :   }
    1572             :   /* failure */
    1573         412 :   return RecCoeff3(nf,d,prec);
    1574             : }
    1575             : 
    1576             : /* Attempts to find a polynomial with coefficients in nf such that
    1577             :    its coefficients are close to those of pol at the place v and
    1578             :    less than a certain bound at all the other places. This bound is obtained by assuming
    1579             :    that the roots at the other places are, in absolute values, less than 2 if flag == 0,
    1580             :    and less than 1 if flag = 1. */
    1581             : static GEN
    1582         546 : RecCoeff(GEN nf,  GEN pol,  long v, long flag, long prec)
    1583             : {
    1584         546 :   long j, md, cl = degpol(pol);
    1585         546 :   pari_sp av = avma;
    1586             :   RC_data d;
    1587             : 
    1588             :   /* if precision(pol) is too low, abort */
    1589        3703 :   for (j = 2; j <= cl+1; j++)
    1590             :   {
    1591        3157 :     GEN t = gel(pol, j);
    1592        3157 :     if (precision(t) - gexpo(t) < 34) return NULL;
    1593             :   }
    1594             : 
    1595         546 :   md = cl/2;
    1596         546 :   pol = leafcopy(pol);
    1597             : 
    1598         546 :   d.N = nf_get_degree(nf);
    1599         546 :   d.v = v;
    1600             : 
    1601        2485 :   for (j = 1; j <= cl; j++)
    1602             :   { /* start with the coefficients in the middle,
    1603             :        since they are the harder to recognize! */
    1604        2156 :     long cf = md + (j%2? j/2: -j/2);
    1605        2156 :     GEN t, bound = binomial(utoipos(cl), cf);
    1606             : 
    1607        2156 :     if (flag == 0) bound = shifti(bound, cl-cf);
    1608        2156 :     if (DEBUGLEVEL>1) err_printf("RecCoeff (cf = %ld, B = %Ps)\n", cf, bound);
    1609        2156 :     d.beta = real_i( gel(pol,cf+2) );
    1610        2156 :     d.B    = bound;
    1611        2156 :     if (! (t = RecCoeff2(nf, &d, prec)) ) return NULL;
    1612        1939 :     gel(pol, cf+2) = coltoalg(nf,t);
    1613             :   }
    1614         329 :   gel(pol,cl+2) = gen_1;
    1615         329 :   return gerepilecopy(av, pol);
    1616             : }
    1617             : 
    1618             : /* an[q * i] *= chi for all (i,p)=1 */
    1619             : static void
    1620      111303 : an_mul(int **an, long p, long q, long n, long deg, GEN chi, int **reduc)
    1621             : {
    1622             :   pari_sp av;
    1623             :   long c,i;
    1624             :   int *T;
    1625             : 
    1626      111303 :   if (gequal1(chi)) return;
    1627      102549 :   av = avma;
    1628      102549 :   T = (int*)new_chunk(deg); Polmod2Coeff(T,chi, deg);
    1629     2158588 :   for (c = 1, i = q; i <= n; i += q, c++)
    1630     2056039 :     if (c == p) c = 0; else MulCoeff(an[i], T, reduc, deg);
    1631      102549 :   set_avma(av);
    1632             : }
    1633             : /* an[q * i] = 0 for all (i,p)=1 */
    1634             : static void
    1635      100787 : an_set0_coprime(int **an, long p, long q, long n, long deg)
    1636             : {
    1637             :   long c,i;
    1638     1215943 :   for (c = 1, i = q; i <= n; i += q, c++)
    1639     1115156 :     if (c == p) c = 0; else _0toCoeff(an[i], deg);
    1640      100787 : }
    1641             : /* an[q * i] = 0 for all i */
    1642             : static void
    1643         105 : an_set0(int **an, long p, long n, long deg)
    1644             : {
    1645             :   long i;
    1646       60020 :   for (i = p; i <= n; i += p) _0toCoeff(an[i], deg);
    1647         105 : }
    1648             : 
    1649             : /* compute the coefficients an for the quadratic case */
    1650             : static int**
    1651         735 : computean(GEN dtcr, LISTray *R, long n, long deg)
    1652             : {
    1653         735 :   pari_sp av = avma, av2;
    1654             :   long i, p, q, condZ, l;
    1655             :   int **an, **reduc;
    1656             :   GEN L, chi, chi1;
    1657             :   CHI_t C;
    1658             : 
    1659         735 :   init_CHI_alg(&C, ch_CHI(dtcr));
    1660         735 :   condZ= R->condZ;
    1661             : 
    1662         735 :   an = InitMatAn(n, deg, 1);
    1663         735 :   reduc = InitReduction(C.ord, deg);
    1664         735 :   av2 = avma;
    1665             : 
    1666             :   /* all pr | p divide cond */
    1667         735 :   L = R->L0; l = lg(L);
    1668         840 :   for (i=1; i<l; i++) an_set0(an,L[i],n,deg);
    1669             : 
    1670             :   /* 1 prime of degree 2 */
    1671         735 :   L = R->L2; l = lg(L);
    1672      100064 :   for (i=1; i<l; i++, set_avma(av2))
    1673             :   {
    1674       99329 :     p = L[i];
    1675       99329 :     if (condZ == 1) chi = C.val[0]; /* 1 */
    1676       99175 :     else            chi = CHI_eval(&C, gel(R->rayZ, p%condZ));
    1677       99329 :     chi1 = chi;
    1678       99329 :     for (q=p;;)
    1679             :     {
    1680      100787 :       an_set0_coprime(an, p,q,n,deg); /* v_p(q) odd */
    1681      100787 :       if (! (q = umuluu_le(q,p, n)) ) break;
    1682             : 
    1683        4614 :       an_mul(an,p,q,n,deg,chi,reduc);
    1684        4614 :       if (! (q = umuluu_le(q,p, n)) ) break;
    1685        1458 :       chi = gmul(chi, chi1);
    1686             :     }
    1687             :   }
    1688             : 
    1689             :   /* 1 prime of degree 1 */
    1690         735 :   L = R->L1; l = lg(L);
    1691        2415 :   for (i=1; i<l; i++, set_avma(av2))
    1692             :   {
    1693        1680 :     p = L[i];
    1694        1680 :     chi = CHI_eval(&C, gel(R->L1ray,i));
    1695        1680 :     chi1 = chi;
    1696        1680 :     for(q=p;;)
    1697             :     {
    1698        7760 :       an_mul(an,p,q,n,deg,chi,reduc);
    1699        7760 :       if (! (q = umuluu_le(q,p, n)) ) break;
    1700        6080 :       chi = gmul(chi, chi1);
    1701             :     }
    1702             :   }
    1703             : 
    1704             :   /* 2 primes of degree 1 */
    1705         735 :   L = R->L11; l = lg(L);
    1706       92469 :   for (i=1; i<l; i++, set_avma(av2))
    1707             :   {
    1708             :     GEN ray1, ray2, chi11, chi12, chi2;
    1709             : 
    1710       91734 :     p = L[i]; ray1 = gel(R->L11ray,i); /* use pr1 pr2 = (p) */
    1711       91734 :     if (condZ == 1)
    1712         112 :       ray2 = ZC_neg(ray1);
    1713             :     else
    1714       91622 :       ray2 = ZC_sub(gel(R->rayZ, p%condZ),  ray1);
    1715       91734 :     chi11 = CHI_eval(&C, ray1);
    1716       91734 :     chi12 = CHI_eval(&C, ray2);
    1717             : 
    1718       91734 :     chi1 = gadd(chi11, chi12);
    1719       91734 :     chi2 = chi12;
    1720       91734 :     for(q=p;;)
    1721             :     {
    1722       98929 :       an_mul(an,p,q,n,deg,chi1,reduc);
    1723       98929 :       if (! (q = umuluu_le(q,p, n)) ) break;
    1724        7195 :       chi2 = gmul(chi2, chi12);
    1725        7195 :       chi1 = gadd(chi2, gmul(chi1, chi11));
    1726             :     }
    1727             :   }
    1728             : 
    1729         735 :   CorrectCoeff(dtcr, an, reduc, n, deg);
    1730         735 :   FreeMat(reduc, deg-1);
    1731         735 :   set_avma(av); return an;
    1732             : }
    1733             : 
    1734             : /* return the vector of A^i/i for i = 1...n */
    1735             : static GEN
    1736         231 : mpvecpowdiv(GEN A, long n)
    1737             : {
    1738         231 :   pari_sp av = avma;
    1739             :   long i;
    1740         231 :   GEN v = powersr(A, n);
    1741         231 :   GEN w = cgetg(n+1, t_VEC);
    1742         231 :   gel(w,1) = rcopy(gel(v,2));
    1743      368368 :   for (i=2; i<=n; i++) gel(w,i) = divru(gel(v,i+1), i);
    1744         231 :   return gerepileupto(av, w);
    1745             : }
    1746             : 
    1747             : static void GetST0(GEN bnr, GEN *pS, GEN *pT, GEN CR, long prec);
    1748             : /* allocate memory for GetST answer */
    1749             : static void
    1750         427 : ST_alloc(GEN *pS, GEN *pT, long l, long prec)
    1751             : {
    1752             :   long j;
    1753         427 :   *pS = cgetg(l, t_VEC);
    1754         427 :   *pT = cgetg(l, t_VEC);
    1755        1932 :   for (j = 1; j < l; j++)
    1756             :   {
    1757        1505 :     gel(*pS,j) = cgetc(prec);
    1758        1505 :     gel(*pT,j) = cgetc(prec);
    1759             :   }
    1760         427 : }
    1761             : 
    1762             : /* compute S and T for the quadratic case. The following cases are:
    1763             :  * 1) bnr complex;
    1764             :  * 2) bnr real and no infinite place divide cond_chi (TODO);
    1765             :  * 3) bnr real and one infinite place divide cond_chi;
    1766             :  * 4) bnr real and both infinite places divide cond_chi (TODO) */
    1767             : static void
    1768         238 : QuadGetST(GEN bnr, GEN *pS, GEN *pT, GEN CR, long prec)
    1769             : {
    1770             :   pari_sp av, av1, av2;
    1771             :   long ncond, n, j, k, n0;
    1772         238 :   GEN vChar = gel(CR,1), dataCR = gel(CR,2), S, T, an, cs, N0, C;
    1773             :   LISTray LIST;
    1774             : 
    1775             :   /* initializations */
    1776         238 :   ST_alloc(pS, pT, lg(dataCR), prec); T = *pT; S = *pS;
    1777         238 :   av = avma;
    1778         238 :   ncond = lg(vChar)-1;
    1779         238 :   C    = cgetg(ncond+1, t_VEC);
    1780         238 :   N0   = cgetg(ncond+1, t_VECSMALL);
    1781         238 :   cs   = cgetg(ncond+1, t_VECSMALL);
    1782         238 :   n0 = 0;
    1783         469 :   for (j = 1; j <= ncond; j++)
    1784             :   {
    1785         266 :     GEN dtcr = gel(dataCR, mael(vChar,j,1)), c = ch_C(dtcr);
    1786             :     long r1, r2;
    1787             : 
    1788         266 :     gel(C,j) = c;
    1789         266 :     nf_get_sign(bnr_get_nf(ch_bnr(dtcr)), &r1, &r2);
    1790         266 :     if (r1 == 2) /* real quadratic */
    1791             :     {
    1792         252 :       cs[j] = 2 + ch_q(dtcr);
    1793         252 :       if (cs[j] == 2 || cs[j] == 4)
    1794             :       { /* NOT IMPLEMENTED YET */
    1795          35 :         GetST0(bnr, pS, pT, CR, prec);
    1796          35 :         return;
    1797             :       }
    1798             :       /* FIXME: is this value of N0 correct for the general case ? */
    1799         217 :       N0[j] = (long)prec * 0.35 * gtodouble(c);
    1800             :     }
    1801             :     else /* complex quadratic */
    1802             :     {
    1803          14 :       cs[j] = 1;
    1804          14 :       N0[j] = (long)prec * 0.7 * gtodouble(c);
    1805             :     }
    1806         231 :     if (n0 < N0[j]) n0 = N0[j];
    1807             :   }
    1808         203 :   if (DEBUGLEVEL>1) err_printf("N0 = %ld\n", n0);
    1809         203 :   InitPrimesQuad(bnr, n0, &LIST);
    1810             : 
    1811         203 :   av1 = avma;
    1812             :   /* loop over conductors */
    1813         434 :   for (j = 1; j <= ncond; j++)
    1814             :   {
    1815         231 :     GEN c0 = gel(C,j), c1 = divur(1, c0), c2 = divur(2, c0);
    1816         231 :     GEN ec1 = mpexp(c1), ec2 = mpexp(c2), LChar = gel(vChar,j);
    1817             :     GEN vf0, vf1, cf0, cf1;
    1818         231 :     const long nChar = lg(LChar)-1, NN = N0[j];
    1819             : 
    1820         231 :     if (DEBUGLEVEL>1)
    1821           0 :       err_printf("* conductor no %ld/%ld (N = %ld)\n\tInit: ", j,ncond,NN);
    1822         231 :     if (realprec(ec1) > prec) ec1 = rtor(ec1, prec);
    1823         231 :     if (realprec(ec2) > prec) ec2 = rtor(ec2, prec);
    1824         231 :     switch(cs[j])
    1825             :     {
    1826          14 :     case 1:
    1827          14 :       cf0 = gen_1;
    1828          14 :       cf1 = c0;
    1829          14 :       vf0 = mpveceint1(rtor(c1, prec), ec1, NN);
    1830          14 :       vf1 = mpvecpowdiv(invr(ec1), NN); break;
    1831             : 
    1832         217 :     case 3:
    1833         217 :       cf0 = sqrtr(mppi(prec));
    1834         217 :       cf1 = gmul2n(cf0, 1);
    1835         217 :       cf0 = gmul(cf0, c0);
    1836         217 :       vf0 = mpvecpowdiv(invr(ec2), NN);
    1837         217 :       vf1 = mpveceint1(rtor(c2, prec), ec2, NN); break;
    1838             : 
    1839           0 :     default:
    1840           0 :       cf0 = cf1 = NULL; /* FIXME: not implemented */
    1841           0 :       vf0 = vf1 = NULL;
    1842             :     }
    1843         973 :     for (k = 1; k <= nChar; k++)
    1844             :     {
    1845         742 :       long u = LChar[k], d, c;
    1846         742 :       GEN dtcr = gel(dataCR, u), z, s, t;
    1847             :       int **matan;
    1848             : 
    1849         742 :       if (!ch_comp(dtcr)) continue;
    1850         735 :       if (DEBUGLEVEL>1) err_printf("\tchar no: %ld (%ld/%ld)\n", u,k,nChar);
    1851         735 :       d = ch_phideg(dtcr);
    1852         735 :       z = gel(ch_CHI(dtcr), 2); s = t = gen_0; av2 = avma;
    1853         735 :       matan = computean(gel(dataCR,u), &LIST, NN, d);
    1854     1255374 :       for (n = 1, c = 0; n <= NN; n++)
    1855     1254639 :         if ((an = EvalCoeff(z, matan[n], d)))
    1856             :         {
    1857      330118 :           s = gadd(s, gmul(an, gel(vf0,n)));
    1858      330118 :           t = gadd(t, gmul(an, gel(vf1,n)));
    1859      330118 :           if (++c == 256) { gerepileall(av2,2, &s,&t); c = 0; }
    1860             :         }
    1861         735 :       gaffect(gmul(cf0, s), gel(S,u));
    1862         735 :       gaffect(gmul(cf1, conj_i(t)), gel(T,u));
    1863         735 :       FreeMat(matan,NN); set_avma(av2);
    1864             :     }
    1865         231 :     if (DEBUGLEVEL>1) err_printf("\n");
    1866         231 :     set_avma(av1);
    1867             :   }
    1868         203 :   set_avma(av);
    1869             : }
    1870             : 
    1871             : /* s += t*u. All 3 of them t_REAL, except we allow s or u = NULL (for 0) */
    1872             : static GEN
    1873    51886310 : _addmulrr(GEN s, GEN t, GEN u)
    1874             : {
    1875    51886310 :   if (u)
    1876             :   {
    1877    51618378 :     GEN v = mulrr(t, u);
    1878    51618378 :     return s? addrr(s, v): v;
    1879             :   }
    1880      267932 :   return s;
    1881             : }
    1882             : /* s += t. Both real, except we allow s or t = NULL (for exact 0) */
    1883             : static GEN
    1884   105518127 : _addrr(GEN s, GEN t)
    1885   105518127 : { return t? (s? addrr(s, t): t) : s; }
    1886             : 
    1887             : /* S & T for the general case. This is time-critical: optimize */
    1888             : static void
    1889      521409 : get_cS_cT(ST_t *T, long n)
    1890             : {
    1891             :   pari_sp av;
    1892             :   GEN csurn, nsurc, lncsurn, A, B, s, t, Z, aij, bij;
    1893             :   long i, j, r, i0;
    1894             : 
    1895      521409 :   if (gel(T->cS,n)) return;
    1896             : 
    1897      244167 :   av = avma;
    1898      244167 :   aij = T->aij; i0= T->i0;
    1899      244167 :   bij = T->bij; r = T->r;
    1900      244167 :   Z = cgetg(r+1, t_VEC);
    1901      244167 :   gel(Z,1) = NULL; /* unused */
    1902             : 
    1903      244167 :   csurn = divru(T->c1, n);
    1904      244167 :   nsurc = invr(csurn);
    1905      244167 :   lncsurn = logr_abs(csurn);
    1906             : 
    1907      244167 :   if (r > 1)
    1908             :   {
    1909      243992 :     gel(Z,2) = lncsurn; /* r >= 2 */
    1910      248087 :     for (i = 3; i <= r; i++)
    1911        4095 :       gel(Z,i) = divru(mulrr(gel(Z,i-1), lncsurn), i-1);
    1912             :     /* Z[i] = ln^(i-1)(c1/n) / (i-1)! */
    1913             :   }
    1914             : 
    1915             :   /* i = i0 */
    1916      244167 :     A = gel(aij,i0); t = _addrr(NULL, gel(A,1));
    1917      244167 :     B = gel(bij,i0); s = _addrr(NULL, gel(B,1));
    1918      492254 :     for (j = 2; j <= r; j++)
    1919             :     {
    1920      248087 :       s = _addmulrr(s, gel(Z,j),gel(B,j));
    1921      248087 :       t = _addmulrr(t, gel(Z,j),gel(A,j));
    1922             :     }
    1923    52392813 :   for (i = i0 - 1; i > 1; i--)
    1924             :   {
    1925    52148646 :     A = gel(aij,i); if (t) t = mulrr(t, nsurc);
    1926    52148646 :     B = gel(bij,i); if (s) s = mulrr(s, nsurc);
    1927    77595627 :     for (j = odd(i)? T->rc2: T->rc1; j > 1; j--)
    1928             :     {
    1929    25446981 :       s = _addmulrr(s, gel(Z,j),gel(B,j));
    1930    25446981 :       t = _addmulrr(t, gel(Z,j),gel(A,j));
    1931             :     }
    1932    52148646 :     s = _addrr(s, gel(B,1));
    1933    52148646 :     t = _addrr(t, gel(A,1));
    1934             :   }
    1935             :   /* i = 1 */
    1936      244167 :     A = gel(aij,1); if (t) t = mulrr(t, nsurc);
    1937      244167 :     B = gel(bij,1); if (s) s = mulrr(s, nsurc);
    1938      244167 :     s = _addrr(s, gel(B,1));
    1939      244167 :     t = _addrr(t, gel(A,1));
    1940      492254 :     for (j = 2; j <= r; j++)
    1941             :     {
    1942      248087 :       s = _addmulrr(s, gel(Z,j),gel(B,j));
    1943      248087 :       t = _addmulrr(t, gel(Z,j),gel(A,j));
    1944             :     }
    1945      244167 :   s = _addrr(s, T->b? mulrr(csurn, gel(T->powracpi,T->b+1)): csurn);
    1946      244167 :   if (!s) s = gen_0;
    1947      244167 :   if (!t) t = gen_0;
    1948      244167 :   gel(T->cS,n) = gclone(s);
    1949      244167 :   gel(T->cT,n) = gclone(t); set_avma(av);
    1950             : }
    1951             : 
    1952             : static void
    1953         497 : clear_cScT(ST_t *T, long N)
    1954             : {
    1955         497 :   GEN cS = T->cS, cT = T->cT;
    1956             :   long i;
    1957     1643058 :   for (i=1; i<=N; i++)
    1958     1642561 :     if (cS[i]) {
    1959      244167 :       gunclone(gel(cS,i));
    1960      244167 :       gunclone(gel(cT,i)); gel(cS,i) = gel(cT,i) = NULL;
    1961             :     }
    1962         497 : }
    1963             : 
    1964             : static void
    1965         308 : init_cScT(ST_t *T, GEN dtcr, long N, long prec)
    1966             : {
    1967         308 :   ch_get3(dtcr, &T->a, &T->b, &T->c);
    1968         308 :   T->rc1 = T->a + T->c;
    1969         308 :   T->rc2 = T->b + T->c;
    1970         308 :   T->r   = maxss(T->rc2+1, T->rc1); /* >= 2 */
    1971         308 :   ppgamma(T, prec);
    1972         308 :   clear_cScT(T, N);
    1973         308 : }
    1974             : 
    1975             : /* return a t_REAL */
    1976             : static GEN
    1977         518 : zeta_get_limx(long r1, long r2, long bit)
    1978             : {
    1979         518 :   pari_sp av = avma;
    1980             :   GEN p1, p2, c0, c1, A0;
    1981         518 :   long r = r1 + r2, N = r + r2;
    1982             : 
    1983             :   /* c1 = N 2^(-2r2 / N) */
    1984         518 :   c1 = mulrs(powrfrac(real2n(1, DEFAULTPREC), -2*r2, N), N);
    1985             : 
    1986         518 :   p1 = powru(Pi2n(1, DEFAULTPREC), r - 1);
    1987         518 :   p2 = mulir(powuu(N,r), p1); shiftr_inplace(p2, -r2);
    1988         518 :   c0 = sqrtr( divrr(p2, powru(c1, r+1)) );
    1989             : 
    1990         518 :   A0 = logr_abs( gmul2n(c0, bit) ); p2 = divrr(A0, c1);
    1991         518 :   p1 = divrr(mulur(N*(r+1), logr_abs(p2)), addsr(2*(r+1), gmul2n(A0,2)));
    1992         518 :   return gerepileuptoleaf(av, divrr(addrs(p1, 1), powruhalf(p2, N)));
    1993             : }
    1994             : /* N_0 = floor( C_K / limx ). Large */
    1995             : static long
    1996         637 : zeta_get_N0(GEN C,  GEN limx)
    1997             : {
    1998             :   long e;
    1999         637 :   pari_sp av = avma;
    2000         637 :   GEN z = gcvtoi(gdiv(C, limx), &e); /* avoid truncation error */
    2001         637 :   if (e >= 0 || is_bigint(z))
    2002           0 :     pari_err_OVERFLOW("zeta_get_N0 [need too many primes]");
    2003         637 :   return gc_long(av, itos(z));
    2004             : }
    2005             : 
    2006             : static GEN
    2007        1897 : eval_i(long r1, long r2, GEN limx, long i)
    2008             : {
    2009        1897 :   GEN t = powru(limx, i);
    2010        1897 :   if (!r1)      t = mulrr(t, powru(mpfactr(i  , DEFAULTPREC), r2));
    2011        1897 :   else if (!r2) t = mulrr(t, powru(mpfactr(i/2, DEFAULTPREC), r1));
    2012             :   else {
    2013           0 :     GEN u1 = mpfactr(i/2, DEFAULTPREC);
    2014           0 :     GEN u2 = mpfactr(i,   DEFAULTPREC);
    2015           0 :     if (r1 == r2) t = mulrr(t, powru(mulrr(u1,u2), r1));
    2016           0 :     else t = mulrr(t, mulrr(powru(u1,r1), powru(u2,r2)));
    2017             :   }
    2018        1897 :   return t;
    2019             : }
    2020             : 
    2021             : /* "small" even i such that limx^i ( (i\2)! )^r1 ( i! )^r2 > B. */
    2022             : static long
    2023         189 : get_i0(long r1, long r2, GEN B, GEN limx)
    2024             : {
    2025         189 :   long imin = 1, imax = 1400;
    2026         196 :   while (mpcmp(eval_i(r1,r2,limx, imax), B) < 0) { imin = imax; imax *= 2; }
    2027        1890 :   while(imax - imin >= 4)
    2028             :   {
    2029        1701 :     long m = (imax + imin) >> 1;
    2030        1701 :     if (mpcmp(eval_i(r1,r2,limx, m), B) >= 0) imax = m; else imin = m;
    2031             :   }
    2032         189 :   return imax & ~1; /* make it even */
    2033             : }
    2034             : /* limx = zeta_get_limx(r1, r2, bit), a t_REAL */
    2035             : static long
    2036         189 : zeta_get_i0(long r1, long r2, long bit, GEN limx)
    2037             : {
    2038         189 :   pari_sp av = avma;
    2039         189 :   GEN B = gmul(sqrtr( divrr(powrs(mppi(DEFAULTPREC), r2-3), limx) ),
    2040             :                gmul2n(powuu(5, r1), bit + r2));
    2041         189 :   return gc_long(av, get_i0(r1, r2, B, limx));
    2042             : }
    2043             : 
    2044             : static void
    2045         189 : GetST0(GEN bnr, GEN *pS, GEN *pT, GEN CR, long prec)
    2046             : {
    2047             :   pari_sp av, av1, av2;
    2048             :   long n, j, k, jc, n0, prec2, i0, r1, r2, ncond;
    2049         189 :   GEN nf = bnr_get_nf(bnr);
    2050         189 :   GEN vChar = gel(CR,1), dataCR = gel(CR,2), N0, C, an, limx, S, T;
    2051             :   LISTray LIST;
    2052             :   ST_t cScT;
    2053             : 
    2054         189 :   ST_alloc(pS, pT, lg(dataCR), prec); T = *pT; S = *pS;
    2055         189 :   av = avma;
    2056         189 :   nf_get_sign(nf,&r1,&r2);
    2057         189 :   ncond = lg(vChar)-1;
    2058         189 :   C  = cgetg(ncond+1, t_VEC);
    2059         189 :   N0 = cgetg(ncond+1, t_VECSMALL);
    2060         189 :   n0 = 0;
    2061         189 :   limx = zeta_get_limx(r1, r2, prec);
    2062         497 :   for (j = 1; j <= ncond; j++)
    2063             :   {
    2064         308 :     GEN dtcr = gel(dataCR, mael(vChar,j,1)), c = ch_C(dtcr);
    2065         308 :     gel(C,j) = c;
    2066         308 :     N0[j] = zeta_get_N0(c, limx);
    2067         308 :     if (n0 < N0[j]) n0  = N0[j];
    2068             :   }
    2069         189 :   cScT.i0 = i0 = zeta_get_i0(r1, r2, prec, limx);
    2070         189 :   if (DEBUGLEVEL>1) err_printf("i0 = %ld, N0 = %ld\n",i0, n0);
    2071         189 :   InitPrimes(bnr, n0, &LIST);
    2072         189 :   prec2 = precdbl(prec) + EXTRAPREC64;
    2073         189 :   cScT.powracpi = powersr(sqrtr(mppi(prec2)), r1);
    2074         189 :   cScT.cS = cgetg(n0+1, t_VEC);
    2075         189 :   cScT.cT = cgetg(n0+1, t_VEC);
    2076      794107 :   for (j=1; j<=n0; j++) gel(cScT.cS,j) = gel(cScT.cT,j) = NULL;
    2077             : 
    2078         189 :   av1 = avma;
    2079         497 :   for (jc = 1; jc <= ncond; jc++)
    2080             :   {
    2081         308 :     GEN LChar = gel(vChar,jc);
    2082         308 :     long nChar = lg(LChar)-1, N = N0[jc];
    2083             : 
    2084             :     /* Can discard completely a conductor if all chars satisfy L(0,chi) = 0
    2085             :      * Not sure whether this is possible. */
    2086         308 :     if (DEBUGLEVEL>1)
    2087           0 :       err_printf("* conductor no %ld/%ld (N = %ld)\n\tInit: ", jc,ncond,N);
    2088             : 
    2089         308 :     cScT.c1 = gel(C,jc);
    2090         308 :     init_cScT(&cScT, gel(dataCR, LChar[1]), N, prec2);
    2091         308 :     av2 = avma;
    2092         875 :     for (k = 1; k <= nChar; k++)
    2093             :     {
    2094         567 :       long d, c, u = LChar[k];
    2095         567 :       GEN dtcr = gel(dataCR, u), z, s, t;
    2096             :       int **matan;
    2097             : 
    2098         567 :       if (!ch_comp(dtcr)) continue;
    2099         560 :       if (DEBUGLEVEL>1) err_printf("\tchar no: %ld (%ld/%ld)\n", u,k,nChar);
    2100         560 :       z = gel(ch_CHI(dtcr), 2);
    2101         560 :       d = ch_phideg(dtcr); s = t = gen_0;
    2102         560 :       matan = ComputeCoeff(dtcr, &LIST, N, d);
    2103     2094167 :       for (n = 1, c = 0; n <= N; n++)
    2104     2093607 :         if ((an = EvalCoeff(z, matan[n], d)))
    2105             :         {
    2106      521409 :           get_cS_cT(&cScT, n);
    2107      521409 :           s = gadd(s, gmul(an, gel(cScT.cS,n)));
    2108      521409 :           t = gadd(t, gmul(an, gel(cScT.cT,n)));
    2109      521409 :           if (++c == 256) { gerepileall(av2,2, &s,&t); c = 0; }
    2110             :         }
    2111         560 :       gaffect(s,         gel(S,u));
    2112         560 :       gaffect(conj_i(t), gel(T,u));
    2113         560 :       FreeMat(matan, N); set_avma(av2);
    2114             :     }
    2115         308 :     if (DEBUGLEVEL>1) err_printf("\n");
    2116         308 :     set_avma(av1);
    2117             :   }
    2118         189 :   clear_cScT(&cScT, n0);
    2119         189 :   set_avma(av);
    2120         189 : }
    2121             : 
    2122             : static void
    2123         392 : GetST(GEN bnr, GEN *pS, GEN *pT, GEN CR, long prec)
    2124             : {
    2125         392 :   if (nf_get_degree(bnr_get_nf(bnr)) == 2)
    2126         238 :     QuadGetST(bnr, pS, pT, CR, prec);
    2127             :   else
    2128         154 :     GetST0(bnr, pS, pT, CR, prec);
    2129         392 : }
    2130             : 
    2131             : /*******************************************************************/
    2132             : /*                                                                 */
    2133             : /*     Class fields of real quadratic fields using Stark units     */
    2134             : /*                                                                 */
    2135             : /*******************************************************************/
    2136             : /* compute the Hilbert class field using genus class field theory when
    2137             :    the exponent of the class group is exactly 2 (trivial group not covered) */
    2138             : /* Cf Herz, Construction of class fields, LNM 21, Theorem 1 (VII-6) */
    2139             : static GEN
    2140          14 : GenusFieldQuadReal(GEN disc)
    2141             : {
    2142          14 :   GEN T = NULL, p0 = NULL, P = gel(Z_factor(disc), 1);
    2143          14 :   long i, i0 = 0, l = lg(P);
    2144             : 
    2145          42 :   for (i = 1; i < l; i++)
    2146             :   {
    2147          35 :     GEN p = gel(P,i);
    2148          35 :     if (mod4(p) == 3) { p0 = p; i0 = i; break; }
    2149             :   }
    2150          14 :   l--; /* remove last prime */
    2151          14 :   if (i0 == l) l--; /* ... remove p0 and last prime */
    2152          49 :   for (i = 1; i < l; i++)
    2153             :   {
    2154          35 :     GEN p = gel(P,i), d, t;
    2155          35 :     if (i == i0) continue;
    2156          28 :     if (absequaliu(p, 2))
    2157          14 :       switch (mod32(disc))
    2158             :       {
    2159          14 :         case  8: d = gen_2; break;
    2160           0 :         case 24: d = shifti(p0, 1); break;
    2161           0 :         default: d = p0; break;
    2162             :       }
    2163             :     else
    2164          14 :       d = (mod4(p) == 1)? p: mulii(p0, p);
    2165          28 :     t = mkpoln(3, gen_1, gen_0, negi(d)); /* x^2 - d */
    2166          28 :     T = T? ZX_composedsum(T, t): t;
    2167             :   }
    2168          14 :   return polredbest(T, 0);
    2169             : }
    2170             : static GEN
    2171         406 : GenusFieldQuadImag(GEN disc)
    2172             : {
    2173         406 :   GEN T = NULL, P = gel(absZ_factor(disc), 1);
    2174         406 :   long i, n = lg(P)-1;
    2175        1183 :   for (i = 1; i < n; i++) /* remove last prime */
    2176             :   {
    2177         777 :     GEN p = gel(P,i), d, t;
    2178         777 :     if (absequaliu(p, 2))
    2179         231 :       switch (mod32(disc))
    2180             :       {
    2181          56 :         case 24: d = gen_2; break;  /* disc = 8 mod 32 */
    2182          42 :         case  8: d = gen_m2; break; /* disc =-8 mod 32 */
    2183         133 :         default: d = gen_m1; break;
    2184             :       }
    2185             :     else
    2186         546 :       d = (mod4(p) == 1)? p: negi(p);
    2187         777 :     t = mkpoln(3, gen_1, gen_0, negi(d)); /* x^2 - d */
    2188         777 :     T = T? ZX_composedsum(T, t): t;
    2189             :   }
    2190         406 :   return polredbest(T, 0);
    2191             : }
    2192             : 
    2193             : /* if flag = 1, computes a fast and crude approximation of the trace of the Stark unit
    2194             :    if flag = 2, computes the Stark unit */
    2195             : static GEN
    2196         658 : AllStark(GEN data, long flag, long newprec)
    2197             : {
    2198         658 :   const long BND = 300;
    2199         658 :   long cl, i, j, cpt = 0, N, h, v, n, r1, r2, den;
    2200             :   pari_sp av, av2;
    2201             :   int **matan;
    2202         658 :   GEN bnr = gel(data,1), nf = bnr_get_nf(bnr), p1, p2, S, T;
    2203         658 :   GEN CR = gel(data,4), dataCR = gel(CR,2);
    2204             :   GEN polrelnum, polrel, Lp, W, vzeta, C, cond1, L1, an;
    2205             :   LISTray LIST;
    2206             :   pari_timer ti;
    2207             : 
    2208         658 :   nf_get_sign(nf, &r1,&r2);
    2209         658 :   N     = nf_get_degree(nf);
    2210         658 :   cond1 = gel(bnr_get_mod(bnr), 2);
    2211             : 
    2212         658 :   v = 1;
    2213        1372 :   while (gequal1(gel(cond1,v))) v++;
    2214         658 :   cl = lg(dataCR)-1;
    2215         658 :   h  = itos(ZM_det_triangular(gel(data,2))) >> 1;
    2216             : 
    2217         658 : LABDOUB:
    2218         658 :   if (DEBUGLEVEL) timer_start(&ti);
    2219         658 :   av = avma;
    2220         658 :   W = AllArtinNumbers(CR, newprec);
    2221         658 :   if (DEBUGLEVEL) timer_printf(&ti,"Compute W");
    2222         658 :   Lp = cgetg(cl + 1, t_VEC);
    2223         658 :   if (flag != 1)
    2224             :   {
    2225         329 :     GetST(bnr, &S, &T, CR, newprec);
    2226         329 :     if (DEBUGLEVEL) timer_printf(&ti, "S&T");
    2227        1400 :     for (i = 1; i <= cl; i++)
    2228             :     {
    2229        1071 :       GEN chi = gel(dataCR, i), vv = gen_0;
    2230        1071 :       if (ch_comp(chi))
    2231        1057 :         vv = gel(GetValue(chi, gel(W,i), gel(S,i), gel(T,i), 2, newprec), 2);
    2232        1071 :       gel(Lp, i) = vv;
    2233             :     }
    2234             :   }
    2235             :   else
    2236             :   { /* compute a crude approximation of the result */
    2237         329 :     C = cgetg(cl + 1, t_VEC);
    2238        1400 :     for (i = 1; i <= cl; i++) gel(C,i) = ch_C(gel(dataCR, i));
    2239         329 :     n = zeta_get_N0(vecmax(C), zeta_get_limx(r1, r2, newprec));
    2240         329 :     if (n > BND) n = BND;
    2241         329 :     if (DEBUGLEVEL) err_printf("N0 in QuickPol: %ld \n", n);
    2242         329 :     InitPrimes(bnr, n, &LIST);
    2243             : 
    2244         329 :     L1 = cgetg(cl+1, t_VEC);
    2245             :     /* use L(1) = sum (an / n) */
    2246        1400 :     for (i = 1; i <= cl; i++)
    2247             :     {
    2248        1071 :       GEN dtcr = gel(dataCR,i);
    2249        1071 :       long d = ch_phideg(dtcr);
    2250        1071 :       matan = ComputeCoeff(dtcr, &LIST, n, d);
    2251        1071 :       av2 = avma;
    2252        1071 :       p1 = real_0(newprec); p2 = gel(ch_CHI(dtcr), 2);
    2253      313999 :       for (j = 1; j <= n; j++)
    2254      312928 :         if ( (an = EvalCoeff(p2, matan[j], d)) )
    2255      120176 :           p1 = gadd(p1, gdivgu(an, j));
    2256        1071 :       gel(L1,i) = gerepileupto(av2, p1);
    2257        1071 :       FreeMat(matan, n);
    2258             :     }
    2259         329 :     p1 = gmul2n(powruhalf(mppi(newprec), N-2), 1);
    2260             : 
    2261        1400 :     for (i = 1; i <= cl; i++)
    2262             :     {
    2263             :       long r;
    2264        1071 :       GEN WW, A = AChi(gel(dataCR,i), &r, 0, newprec);
    2265        1071 :       WW = gmul(gel(C,i), gmul(A, gel(W,i)));
    2266        1071 :       gel(Lp,i) = gdiv(gmul(WW, conj_i(gel(L1,i))), p1);
    2267             :     }
    2268             :   }
    2269             : 
    2270         658 :   p1 = gel(data,3);
    2271         658 :   den = (flag == 0) ? 2*h: h;
    2272         658 :   if (flag == 2)
    2273           7 :     vzeta = cgetg(2*h + 1, t_VEC);
    2274             :   else
    2275         651 :     vzeta = cgetg(h + 1,t_VEC);
    2276        4228 :   for (i = 1; i <= h; i++)
    2277             :   {
    2278        3570 :     GEN z = gen_0, sig = gel(p1,i);
    2279       18172 :     for (j = 1; j <= cl; j++)
    2280             :     {
    2281       14602 :       GEN dtcr = gel(dataCR,j), CHI = ch_CHI(dtcr);
    2282       14602 :       GEN t = mulreal(gel(Lp,j), CharEval(CHI, sig));
    2283       14602 :       if (chi_get_deg(CHI) != 2) t = gmul2n(t, 1); /* character not real */
    2284       14602 :       z = gadd(z, t);
    2285             :     }
    2286        3570 :     z = gdivgu(z, den);
    2287        3570 :     if (flag == 2)
    2288             :     {
    2289          77 :       gel(vzeta, i) = gexp(gmul2n(z,1), newprec);
    2290          77 :       gel(vzeta, i+h) = ginv(gel(vzeta,i));
    2291             :     }
    2292             :     /* if flag == 0, we first try with the square-root of the Stark unit */
    2293        3493 :     else gel(vzeta,i) = gmul2n(gcosh(z, newprec), 1);
    2294             :   }
    2295         658 :   polrelnum = roots_to_pol(vzeta, 0);
    2296         658 :   if (DEBUGLEVEL)
    2297             :   {
    2298           0 :     if (DEBUGLEVEL>1) {
    2299           0 :       err_printf("polrelnum = %Ps\n", polrelnum);
    2300           0 :       err_printf("zetavalues = %Ps\n", vzeta);
    2301           0 :       if (flag == 0)
    2302           0 :         err_printf("Checking the square-root of the Stark unit...\n");
    2303             :     }
    2304           0 :     timer_printf(&ti, "Compute %s", flag? "quickpol": "polrelnum");
    2305             :   }
    2306         658 :   if (flag == 1) return gerepilecopy(av, polrelnum);
    2307             :   /* try to recognize this polynomial */
    2308         329 :   polrel = RecCoeff(nf, polrelnum, v, flag, newprec);
    2309             : 
    2310             :   /* if this doesn't work, maybe the Stark unit is not a square */
    2311         329 :   if (!polrel && flag == 0)
    2312             :   {
    2313         217 :     if (DEBUGLEVEL)
    2314             :     {
    2315           0 :       if (DEBUGLEVEL>1) {
    2316           0 :         err_printf("It's not a square...\n");
    2317           0 :         err_printf("polrelnum = %Ps\n", polrelnum);
    2318             :       }
    2319           0 :       timer_printf(&ti, "Compute polrelnum");
    2320             :     }
    2321        1512 :     for (j = 1; j <= h; j++)
    2322        1295 :       gel(vzeta,j) = gsubgs(gsqr(gel(vzeta,j)), 2);
    2323         217 :     polrelnum = roots_to_pol(vzeta, 0);
    2324             : 
    2325         217 :     polrel = RecCoeff(nf, polrelnum, v, 0, newprec);
    2326             :   }
    2327         329 :   if (!polrel) /* FAILED */
    2328             :   {
    2329           0 :     const long EXTRA_BITS = 64;
    2330             :     long incr_pr;
    2331           0 :     if (++cpt >= 3) pari_err_PREC( "stark (computation impossible)");
    2332             :     /* estimate needed precision */
    2333           0 :     incr_pr = gprecision(polrelnum) - gexpo(polrelnum);
    2334           0 :     if (incr_pr < 0) incr_pr = -incr_pr + EXTRA_BITS;
    2335           0 :     newprec += nbits2extraprec(maxss(3*EXTRA_BITS, cpt*incr_pr));
    2336           0 :     if (DEBUGLEVEL) pari_warn(warnprec, "AllStark", newprec);
    2337           0 :     CharNewPrec(data, newprec);
    2338           0 :     nf = bnr_get_nf(ch_bnr(gel(dataCR,1)));
    2339           0 :     goto LABDOUB;
    2340             :   }
    2341             : 
    2342         329 :   if (DEBUGLEVEL) {
    2343           0 :     if (DEBUGLEVEL>1) err_printf("polrel = %Ps\n", polrel);
    2344           0 :     timer_printf(&ti, "Recpolnum");
    2345             :   }
    2346         329 :   return gerepilecopy(av, polrel);
    2347             : }
    2348             : 
    2349             : /********************************************************************/
    2350             : /*                        Main functions                            */
    2351             : /********************************************************************/
    2352             : static GEN
    2353           0 : bnrstark_cyclic(GEN bnr, GEN dtQ, long prec)
    2354             : {
    2355           0 :   GEN v, vH, cyc = gel(dtQ,2), U = gel(dtQ,3), M = ZM_inv(U, NULL);
    2356           0 :   long i, j, l = lg(M);
    2357             : 
    2358             :   /* M = indep. generators of Cl_f/subgp, restrict to cyclic components */
    2359           0 :   vH = cgetg(l, t_VEC);
    2360           0 :   for (i = j = 1; i < l; i++)
    2361             :   {
    2362           0 :     if (is_pm1(gel(cyc,i))) break;
    2363           0 :     gel(vH, j++) = ZM_hnfmodid(vecsplice(M,i), cyc);
    2364             :   }
    2365           0 :   setlg(vH, j); v = cgetg(l, t_VEC);
    2366           0 :   for (i = 1; i < j; i++) gel(v,i) = bnrstark(bnr, gel(vH,i), prec);
    2367           0 :   return v;
    2368             : }
    2369             : GEN
    2370         203 : bnrstark(GEN bnr, GEN subgrp, long prec)
    2371             : {
    2372             :   long newprec;
    2373         203 :   pari_sp av = avma;
    2374             :   GEN nf, data, dtQ;
    2375             : 
    2376             :   /* check the bnr */
    2377         203 :   checkbnr(bnr); nf  = bnr_get_nf(bnr);
    2378         203 :   if (nf_get_degree(nf) == 1) return galoissubcyclo(bnr, subgrp, 0, 0);
    2379         203 :   if (!nf_get_varn(nf))
    2380           0 :     pari_err_PRIORITY("bnrstark", nf_get_pol(nf), "=", 0);
    2381         203 :   if (nf_get_r2(nf)) pari_err_DOMAIN("bnrstark", "r2", "!=", gen_0, nf);
    2382             : 
    2383             :   /* compute bnr(conductor) */
    2384         196 :   bnr_subgroup_sanitize(&bnr, &subgrp);
    2385         196 :   if (gequal1(ZM_det_triangular(subgrp))) { set_avma(av); return pol_x(0); }
    2386             : 
    2387             :   /* check the class field */
    2388         196 :   if (!gequal0(gel(bnr_get_mod(bnr), 2)))
    2389           7 :     pari_err_DOMAIN("bnrstark", "r2(class field)", "!=", gen_0, bnr);
    2390             : 
    2391             :   /* find a suitable extension N */
    2392         189 :   dtQ = InitQuotient(subgrp);
    2393         189 :   data  = FindModulus(bnr, dtQ, &newprec);
    2394         189 :   if (!data) return gerepileupto(av, bnrstark_cyclic(bnr, dtQ, prec));
    2395         189 :   if (DEBUGLEVEL>1 && newprec > prec)
    2396           0 :     err_printf("new precision: %ld\n", newprec);
    2397         189 :   return gerepileupto(av, AllStark(data, 0, newprec));
    2398             : }
    2399             : 
    2400             : GEN
    2401          14 : bnrstarkunit(GEN bnr, GEN subgrp)
    2402             : {
    2403             :   long newprec, deg;
    2404          14 :   pari_sp av = avma;
    2405             :   GEN nf, data, dtQ, bnf, bnrf, Cm, candD, D, QD, CR;
    2406             : 
    2407             :   /* check the input */
    2408          14 :   checkbnr(bnr); bnf = bnr_get_bnf(bnr); nf = bnf_get_nf(bnf);
    2409          14 :   if (!nf_get_varn(nf))
    2410           0 :     pari_err_PRIORITY("bnrstarkunit", nf_get_pol(nf), "=", 0);
    2411          14 :   deg = nf_get_degree(nf);
    2412          14 :   if (deg == 1) pari_err_IMPL("bnrstarkunit for basefield Q");
    2413          14 :   if (nf_get_r2(nf)) pari_err_DOMAIN("bnrstarkunit", "r2", "!=", gen_0, nf);
    2414          14 :   bnr_subgroup_sanitize(&bnr, &subgrp);
    2415          14 :   if (lg(bid_get_archp(bnr_get_bid(bnr)))-1 != deg-1)
    2416           7 :     pari_err_DOMAIN("bnrstarkunit", "# unramified places", "!=", gen_1, bnr);
    2417           7 :   bnrf = Buchray(bnf, gel(bnr_get_mod(bnr), 1), nf_INIT);
    2418           7 :   subgrp = abmap_subgroup_image(bnrsurjection(bnr, bnrf), subgrp);
    2419           7 :   dtQ  = InitQuotient(subgrp);
    2420           7 :   Cm   = ComputeKernel(bnr, bnrf, dtQ);
    2421           7 :   candD = subgrouplist_cond_sub(bnr, Cm, mkvec(gen_2));
    2422           7 :   if (lg(candD) != 2) pari_err(e_MISC, "incorrect modulus in bnrstark");
    2423           7 :   D    = gel(candD, 1);
    2424           7 :   QD   = InitQuotient(D);
    2425           7 :   CR   = InitChar(bnr, AllChars(bnr, QD, 1), 0, DEFAULTPREC);
    2426           7 :   data = mkvec4(bnr, D, subgroup_classes(Cm), CR);
    2427           7 :   CplxModulus(data, &newprec);
    2428           7 :   return gerepileupto(av, AllStark(data, 2, newprec));
    2429             : }
    2430             : 
    2431             : /* For each character of Cl(bnr)/subgp, compute L(1, chi) (or equivalently
    2432             :  * the first nonzero term c(chi) of the expansion at s = 0).
    2433             :  * If flag & 1: compute the value at s = 1 (for nontrivial characters),
    2434             :  * else compute the term c(chi) and return [r(chi), c(chi)] where r(chi) is
    2435             :  *   the order of L(s, chi) at s = 0.
    2436             :  * If flag & 2: compute the value of the L-function L_S(s, chi) where S is the
    2437             :  *   set of places dividing the modulus of bnr (and the infinite places),
    2438             :  * else
    2439             :  *   compute the value of the primitive L-function attached to chi,
    2440             :  * If flag & 4: return also the character */
    2441             : GEN
    2442          70 : bnrL1(GEN bnr, GEN subgp, long flag, long prec)
    2443             : {
    2444             :   GEN L1, ch, Qt, z;
    2445             :   long l, h;
    2446          70 :   pari_sp av = avma;
    2447             : 
    2448          70 :   checkbnr(bnr);
    2449          70 :   if (flag < 0 || flag > 8) pari_err_FLAG("bnrL1");
    2450             : 
    2451          70 :   subgp = bnr_subgroup_check(bnr, subgp, NULL);
    2452          70 :   if (!subgp) subgp = diagonal_shallow(bnr_get_cyc(bnr));
    2453             : 
    2454          70 :   Qt = InitQuotient(subgp);
    2455          70 :   ch = AllChars(bnr, Qt, 0); l = lg(ch);
    2456          70 :   h = itou(gel(Qt,1));
    2457          70 :   L1 = cgetg((flag&1)? h: h+1, t_VEC);
    2458          70 :   if (l > 1)
    2459             :   {
    2460          63 :     GEN W, S, T, CR = InitChar(bnr, ch, 1, prec), dataCR = gel(CR,2);
    2461             :     long i, j;
    2462             : 
    2463          63 :     GetST(bnr, &S, &T, CR, prec);
    2464          63 :     W = AllArtinNumbers(CR, prec);
    2465         301 :     for (i = j = 1; i < l; i++)
    2466             :     {
    2467         238 :       GEN chi = gel(ch,i);
    2468         238 :       z = GetValue(gel(dataCR,i), gel(W,i), gel(S,i), gel(T,i), flag, prec);
    2469         238 :       gel(L1,j++) = (flag & 4)? mkvec2(gel(chi,1), z): z;
    2470         238 :       if (lg(chi) == 4)
    2471             :       { /* nonreal */
    2472         133 :         z = conj_i(z);
    2473         133 :         gel(L1, j++) = (flag & 4)? mkvec2(gel(chi,3), z): z;
    2474             :       }
    2475             :     }
    2476             :   }
    2477          70 :   if (!(flag & 1))
    2478             :   { /* trivial character */
    2479          35 :     z = GetValue1(bnr, flag & 2, prec);
    2480          35 :     if (flag & 4)
    2481             :     {
    2482           0 :       GEN chi = zerovec(lg(bnr_get_cyc(bnr))-1);
    2483           0 :       z = mkvec2(chi, z);
    2484             :     }
    2485          35 :     gel(L1,h) = z;
    2486             :   }
    2487          70 :   return gerepilecopy(av, L1);
    2488             : }
    2489             : 
    2490             : /*******************************************************************/
    2491             : /*                                                                 */
    2492             : /*       Hilbert and Ray Class field using Stark                   */
    2493             : /*                                                                 */
    2494             : /*******************************************************************/
    2495             : /* P in A[x,y], deg_y P < 2, return P0 and P1 in A[x] such that P = P0 + P1 y */
    2496             : static void
    2497         133 : split_pol_quad(GEN P, GEN *gP0, GEN *gP1)
    2498             : {
    2499         133 :   long i, l = lg(P);
    2500         133 :   GEN P0 = cgetg(l, t_POL), P1 = cgetg(l, t_POL);
    2501         133 :   P0[1] = P1[1] = P[1];
    2502        1211 :   for (i = 2; i < l; i++)
    2503             :   {
    2504        1078 :     GEN c = gel(P,i), c0 = c, c1 = gen_0;
    2505        1078 :     if (typ(c) == t_POL) /* write c = c1 y + c0 */
    2506         945 :       switch(degpol(c))
    2507             :       {
    2508           0 :         case -1: c0 = gen_0; break;
    2509         945 :         default: c1 = gel(c,3); /* fall through */
    2510         945 :         case  0: c0 = gel(c,2); break;
    2511             :       }
    2512        1078 :     gel(P0,i) = c0; gel(P1,i) = c1;
    2513             :   }
    2514         133 :   *gP0 = normalizepol_lg(P0, l);
    2515         133 :   *gP1 = normalizepol_lg(P1, l);
    2516         133 : }
    2517             : 
    2518             : /* k = nf quadratic field, P relative equation of H_k (Hilbert class field)
    2519             :  * return T in Z[X], such that H_k / Q is the compositum of Q[X]/(T) and k */
    2520             : static GEN
    2521         133 : makescind(GEN nf, GEN P)
    2522             : {
    2523         133 :   GEN Pp, p, pol, G, L, a, roo, P0,P1, Ny,Try, nfpol = nf_get_pol(nf);
    2524             :   long i, is_P;
    2525             : 
    2526         133 :   P = lift_shallow(P);
    2527         133 :   split_pol_quad(P, &P0, &P1);
    2528             :   /* P = P0 + y P1, Norm_{k/Q}(P) = P0^2 + Tr y P0P1 + Ny P1^2, irreducible/Q */
    2529         133 :   Ny = gel(nfpol, 2);
    2530         133 :   Try = negi(gel(nfpol, 3));
    2531         133 :   pol = RgX_add(RgX_sqr(P0), RgX_Rg_mul(RgX_sqr(P1), Ny));
    2532         133 :   if (signe(Try)) pol = RgX_add(pol, RgX_Rg_mul(RgX_mul(P0,P1), Try));
    2533             :   /* pol = rnfequation(nf, P); */
    2534         133 :   G = galoisinit(pol, NULL);
    2535         133 :   L = gal_get_group(G);
    2536         133 :   p = gal_get_p(G);
    2537         133 :   a = FpX_oneroot(nfpol, p);
    2538             :   /* P mod a prime \wp above p (which splits) */
    2539         133 :   Pp = FpXY_evalx(P, a, p);
    2540         133 :   roo = gal_get_roots(G);
    2541         133 :   is_P = gequal0( FpX_eval(Pp, remii(gel(roo,1),p), p) );
    2542             :   /* each roo[i] mod p is a root of P or (exclusive) tau(P) mod \wp */
    2543             :   /* record whether roo[1] is a root of P or tau(P) */
    2544             : 
    2545        1022 :   for (i = 1; i < lg(L); i++)
    2546             :   {
    2547        1022 :     GEN perm = gel(L,i);
    2548        1022 :     long k = perm[1]; if (k == 1) continue;
    2549         889 :     k = gequal0( FpX_eval(Pp, remii(gel(roo,k),p), p) );
    2550             :     /* roo[k] is a root of the other polynomial */
    2551         889 :     if (k != is_P)
    2552             :     {
    2553         133 :       ulong o = perm_orderu(perm);
    2554         133 :       if (o != 2) perm = perm_powu(perm, o >> 1);
    2555             :       /* perm has order two and doesn't belong to Gal(H_k/k) */
    2556         133 :       return polredbest(galoisfixedfield(G, perm, 1, varn(P)), 0);
    2557             :     }
    2558             :   }
    2559           0 :   pari_err_BUG("makescind");
    2560             :   return NULL; /*LCOV_EXCL_LINE*/
    2561             : }
    2562             : 
    2563             : /* pbnf = NULL if no bnf is needed, f = NULL may be passed for a trivial
    2564             :  * conductor */
    2565             : static void
    2566         847 : quadray_init(GEN *pD, GEN *pbnf, long prec)
    2567             : {
    2568         847 :   GEN D = *pD, nf, bnf = NULL;
    2569         847 :   if (typ(D) == t_INT)
    2570             :   {
    2571             :     int isfund;
    2572         812 :     if (pbnf) {
    2573         252 :       bnf = Buchall(quadpoly0(D, 1), nf_FORCE, prec);
    2574         252 :       nf = bnf_get_nf(bnf);
    2575         252 :       isfund = equalii(D, nf_get_disc(nf));
    2576             :     }
    2577             :     else
    2578         560 :       isfund = Z_isfundamental(D);
    2579         812 :     if (!isfund) pari_err_DOMAIN("quadray", "isfundamental(D)", "=",gen_0, D);
    2580             :   }
    2581             :   else
    2582             :   {
    2583          35 :     bnf = checkbnf(D);
    2584          35 :     nf = bnf_get_nf(bnf);
    2585          35 :     if (nf_get_degree(nf) != 2)
    2586           7 :       pari_err_DOMAIN("quadray", "degree", "!=", gen_2, nf_get_pol(nf));
    2587          28 :     D = nf_get_disc(nf);
    2588             :   }
    2589         833 :   if (pbnf) *pbnf = bnf;
    2590         833 :   *pD = D;
    2591         833 : }
    2592             : 
    2593             : /* compute the polynomial over Q of the Hilbert class field of
    2594             :    Q(sqrt(D)) where D is a positive fundamental discriminant */
    2595             : static GEN
    2596         147 : quadhilbertreal(GEN D, long prec)
    2597             : {
    2598             :   GEN bnf, bnr, dtQ, data, M;
    2599             :   long newprec;
    2600             :   pari_timer T;
    2601             : 
    2602         147 :   quadray_init(&D, &bnf, prec);
    2603         147 :   switch(itou_or_0(cyc_get_expo(bnf_get_cyc(bnf))))
    2604             :   {
    2605           0 :     case 1: return pol_x(0);
    2606          14 :     case 2: return GenusFieldQuadReal(D);
    2607             :   }
    2608         133 :   bnr  = Buchray(bnf, gen_1, nf_INIT);
    2609         133 :   M = diagonal_shallow(bnr_get_cyc(bnr));
    2610         133 :   dtQ = InitQuotient(M);
    2611             : 
    2612         133 :   if (DEBUGLEVEL) timer_start(&T);
    2613         133 :   data = FindModulus(bnr, dtQ, &newprec);
    2614         133 :   if (DEBUGLEVEL) timer_printf(&T,"FindModulus");
    2615         133 :   if (!data) return bnrstark_cyclic(bnr, dtQ, prec);
    2616         133 :   return makescind(bnf_get_nf(bnf), AllStark(data, 0, newprec));
    2617             : }
    2618             : 
    2619             : /*******************************************************************/
    2620             : /*                                                                 */
    2621             : /*       Hilbert and Ray Class field using CM (Schertz)            */
    2622             : /*                                                                 */
    2623             : /*******************************************************************/
    2624             : /* form^2 = 1 ? */
    2625             : static int
    2626         813 : hasexp2(GEN form)
    2627             : {
    2628         813 :   GEN a = gel(form,1), b = gel(form,2), c = gel(form,3);
    2629         813 :   return !signe(b) || absequalii(a,b) || equalii(a,c);
    2630             : }
    2631             : static int
    2632        1323 : uhasexp2(GEN form)
    2633             : {
    2634        1323 :   long a = form[1], b = form[2], c = form[3];
    2635        1323 :   return !b || a == labs(b) || a == c;
    2636             : }
    2637             : 
    2638             : GEN
    2639         455 : qfbforms(GEN D)
    2640             : {
    2641         455 :   ulong d = itou(D), dover3 = d/3, t, b2, a, b, c, h;
    2642         455 :   GEN L = cgetg((long)(sqrt((double)d) * log2(d)), t_VEC);
    2643         455 :   b2 = b = (d&1); h = 0;
    2644         455 :   if (!b) /* b = 0 treated separately to avoid special cases */
    2645             :   {
    2646         252 :     t = d >> 2; /* (b^2 - D) / 4*/
    2647        2954 :     for (a=1; a*a<=t; a++)
    2648        2702 :       if (c = t/a, t == c*a) gel(L,++h) = mkvecsmall3(a,0,c);
    2649         252 :     b = 2; b2 = 4;
    2650             :   }
    2651             :   /* now b > 0, b = D (mod 2) */
    2652        8078 :   for ( ; b2 <= dover3; b += 2, b2 = b*b)
    2653             :   {
    2654        7623 :     t = (b2 + d) >> 2; /* (b^2 - D) / 4*/
    2655             :     /* b = a */
    2656        7623 :     if (c = t/b, t == c*b) gel(L,++h) = mkvecsmall3(b,b,c);
    2657             :     /* b < a < c */
    2658     1912029 :     for (a = b+1; a*a < t; a++)
    2659     1904406 :       if (c = t/a, t == c*a)
    2660             :       {
    2661        1057 :         gel(L,++h) = mkvecsmall3(a, b,c);
    2662        1057 :         gel(L,++h) = mkvecsmall3(a,-b,c);
    2663             :       }
    2664             :     /* a = c */
    2665        7623 :     if (a * a == t) gel(L,++h) = mkvecsmall3(a,b,a);
    2666             :   }
    2667         455 :   setlg(L,h+1); return L;
    2668             : }
    2669             : 
    2670             : /* gcd(n, 24) */
    2671             : static long
    2672         813 : GCD24(long n)
    2673             : {
    2674         813 :   switch(n % 24)
    2675             :   {
    2676          35 :     case 0: return 24;
    2677          35 :     case 1: return 1;
    2678          28 :     case 2: return 2;
    2679           0 :     case 3: return 3;
    2680         119 :     case 4: return 4;
    2681           0 :     case 5: return 1;
    2682         105 :     case 6: return 6;
    2683           0 :     case 7: return 1;
    2684           0 :     case 8: return 8;
    2685           0 :     case 9: return 3;
    2686          91 :     case 10: return 2;
    2687           0 :     case 11: return 1;
    2688         119 :     case 12: return 12;
    2689           0 :     case 13: return 1;
    2690           0 :     case 14: return 2;
    2691           0 :     case 15: return 3;
    2692          91 :     case 16: return 8;
    2693           0 :     case 17: return 1;
    2694          92 :     case 18: return 6;
    2695           0 :     case 19: return 1;
    2696           0 :     case 20: return 4;
    2697           0 :     case 21: return 3;
    2698          98 :     case 22: return 2;
    2699           0 :     case 23: return 1;
    2700           0 :     default: return 0;
    2701             :   }
    2702             : }
    2703             : 
    2704             : struct gpq_data {
    2705             :   long p, q;
    2706             :   GEN sqd; /* sqrt(D), t_REAL */
    2707             :   GEN u, D;
    2708             :   GEN pq, pq2; /* p*q, 2*p*q */
    2709             :   GEN qfpq ; /* class of \P * \Q */
    2710             : };
    2711             : 
    2712             : /* find P and Q two non principal prime ideals (above p <= q) such that
    2713             :  *   cl(P) = cl(Q) if P,Q have order 2 in Cl(K).
    2714             :  *   Ensure that e = 24 / gcd(24, (p-1)(q-1)) = 1 */
    2715             : /* D t_INT, discriminant */
    2716             : static void
    2717          49 : init_pq(GEN D, struct gpq_data *T)
    2718             : {
    2719          49 :   const long Np = 6547; /* N.B. primepi(50000) = 5133 */
    2720          49 :   const ulong maxq = 50000;
    2721          49 :   GEN listp = cgetg(Np + 1, t_VECSMALL); /* primes p */
    2722          49 :   GEN listP = cgetg(Np + 1, t_VEC); /* primeform(p) if of order 2, else NULL */
    2723          49 :   GEN gcd24 = cgetg(Np + 1, t_VECSMALL); /* gcd(p-1, 24) */
    2724             :   forprime_t S;
    2725          49 :   long l = 1;
    2726          49 :   double best = 0.;
    2727             :   ulong q;
    2728             : 
    2729          49 :   u_forprime_init(&S, 2, ULONG_MAX);
    2730          49 :   T->D = D;
    2731          49 :   T->p = T->q = 0;
    2732             :   for(;;)
    2733        1777 :   {
    2734             :     GEN Q;
    2735             :     long i, gcdq, mod;
    2736             :     int order2, store;
    2737             :     double t;
    2738             : 
    2739        1826 :     q = u_forprime_next(&S);
    2740        1826 :     if (best > 0 && q >= maxq)
    2741             :     {
    2742           0 :       if (DEBUGLEVEL)
    2743           0 :         pari_warn(warner,"possibly suboptimal (p,q) for D = %Ps", D);
    2744           0 :       break;
    2745             :     }
    2746        1826 :     if (kroiu(D, q) < 0) continue; /* inert */
    2747         890 :     Q = qfbred_i(primeform_u(D, q));
    2748         890 :     if (is_pm1(gel(Q,1))) continue; /* Q | q is principal */
    2749             : 
    2750         813 :     store = 1;
    2751         813 :     order2 = hasexp2(Q);
    2752         813 :     gcd24[l] = gcdq = GCD24(q-1);
    2753         813 :     mod = 24 / gcdq; /* mod must divide p-1 otherwise e > 1 */
    2754         813 :     listp[l] = q;
    2755         813 :     gel(listP,l) = order2 ? Q : NULL;
    2756         813 :     t = (q+1)/(double)(q-1);
    2757        2129 :     for (i = 1; i < l; i++) /* try all (p, q), p < q in listp */
    2758             :     {
    2759        1660 :       long p = listp[i], gcdp = gcd24[i];
    2760             :       double b;
    2761             :       /* P,Q order 2 => cl(Q) = cl(P) */
    2762        1660 :       if (order2 && gel(listP,i) && !gequal(gel(listP,i), Q)) continue;
    2763        1653 :       if (gcdp % gcdq == 0) store = 0; /* already a better one in the list */
    2764        1653 :       if ((p-1) % mod) continue;
    2765             : 
    2766         344 :       b = (t*(p+1)) / (p-1); /* (p+1)(q+1) / (p-1)(q-1) */
    2767         344 :       if (b > best) {
    2768          98 :         store = 0; /* (p,q) always better than (q,r) for r >= q */
    2769          98 :         best = b; T->q = q; T->p = p;
    2770          98 :         if (DEBUGLEVEL>2) err_printf("p,q = %ld,%ld\n", p, q);
    2771             :       }
    2772             :       /* won't improve with this q as largest member */
    2773         344 :       if (best > 0) break;
    2774             :     }
    2775             :     /* if !store or (q,r) won't improve on current best pair, forget that q */
    2776         813 :     if (store && t*t > best)
    2777         119 :       if (++l >= Np) pari_err_BUG("quadhilbert (not enough primes)");
    2778         813 :     if (!best) /* (p,q) with p < q always better than (q,q) */
    2779             :     { /* try (q,q) */
    2780         140 :       if (gcdq >= 12 && umodiu(D, q)) /* e = 1 and unramified */
    2781             :       {
    2782           7 :         double b = (t*q) / (q-1); /* q(q+1) / (q-1)^2 */
    2783           7 :         if (b > best) {
    2784           7 :           best = b; T->q = T->p = q;
    2785           7 :           if (DEBUGLEVEL>2) err_printf("p,q = %ld,%ld\n", q, q);
    2786             :         }
    2787             :       }
    2788             :     }
    2789             :     /* If (p1+1)(q+1) / (p1-1)(q-1) <= best, we can no longer improve
    2790             :      * even with best p : stop */
    2791         813 :     if ((listp[1]+1)*t <= (listp[1]-1)*best) break;
    2792             :   }
    2793          49 :   if (DEBUGLEVEL>1)
    2794           0 :     err_printf("(p, q) = %ld, %ld; gain = %f\n", T->p, T->q, 12*best);
    2795          49 : }
    2796             : 
    2797             : static GEN
    2798        4102 : gpq(GEN form, struct gpq_data *T)
    2799             : {
    2800        4102 :   pari_sp av = avma;
    2801        4102 :   long a = form[1], b = form[2], c = form[3], p = T->p, q = T->q;
    2802             :   GEN form2, w, z;
    2803        4102 :   int fl, real = 0;
    2804             : 
    2805        4102 :   form2 = qfbcomp_i(T->qfpq, mkqfb(stoi(a), stoi(-b), stoi(c), T->D));
    2806             :   /* form2 and form yield complex conjugate roots : only compute for the
    2807             :    * lexicographically smallest of the 2 */
    2808        4102 :   fl = cmpis(gel(form2,1), a);
    2809        4102 :   if (fl <= 0)
    2810             :   {
    2811        2156 :     if (fl < 0) return NULL;
    2812         210 :     fl = cmpis(gel(form2,2), b);
    2813         210 :     if (fl <= 0)
    2814             :     {
    2815         147 :       if (fl < 0) return NULL;
    2816             :       /* form == form2 : real root */
    2817          84 :       real = 1;
    2818             :     }
    2819             :   }
    2820             : 
    2821        2093 :   if (p == 2) { /* (a,b,c) = (1,1,0) mod 2 ? */
    2822         203 :     if (a % q == 0 && (a & b & 1) && !(c & 1))
    2823             :     { /* apply S : make sure that (a,b,c) represents odd values */
    2824           0 :       lswap(a,c); b = -b;
    2825             :     }
    2826             :   }
    2827        2093 :   if (a % p == 0 || a % q == 0)
    2828             :   { /* apply T^k, look for c' = a k^2 + b k + c coprime to N */
    2829         595 :     while (c % p == 0 || c % q == 0)
    2830             :     {
    2831          98 :       c += a + b;
    2832          98 :       b += a << 1;
    2833             :     }
    2834         497 :     lswap(a, c); b = -b; /* apply S */
    2835             :   }
    2836             :   /* now (a,b,c) ~ form and (a,pq) = 1 */
    2837             : 
    2838             :   /* gcd(2a, u) = 2,  w = u mod 2pq, -b mod 2a */
    2839        2093 :   w = Z_chinese(T->u, stoi(-b), T->pq2, utoipos(a << 1));
    2840        2093 :   z = double_eta_quotient(utoipos(a), w, T->D, T->p, T->q, T->pq, T->sqd);
    2841        2093 :   if (real && typ(z) == t_COMPLEX) z = gcopy(gel(z, 1));
    2842        2093 :   return gerepileupto(av, z);
    2843             : }
    2844             : 
    2845             : /* returns an equation for the Hilbert class field of Q(sqrt(D)), D < 0
    2846             :  * fundamental discriminant */
    2847             : static GEN
    2848         462 : quadhilbertimag(GEN D)
    2849             : {
    2850             :   GEN L, P, Pi, Pr, qfp, u;
    2851             :   long h, i, prec;
    2852             :   struct gpq_data T;
    2853             :   pari_timer ti;
    2854             : 
    2855         462 :   if (DEBUGLEVEL>1) timer_start(&ti);
    2856         462 :   if (lgefint(D) == 3)
    2857         462 :     switch (D[2]) { /* = |D|; special cases where e > 1 */
    2858           7 :       case 3:
    2859             :       case 4:
    2860             :       case 7:
    2861             :       case 8:
    2862             :       case 11:
    2863             :       case 19:
    2864             :       case 43:
    2865             :       case 67:
    2866           7 :       case 163: return pol_x(0);
    2867             :     }
    2868         455 :   L = qfbforms(D);
    2869         455 :   h = lg(L)-1;
    2870         455 :   if (! (h & (h - 1))) /* power of 2 */
    2871             :   { /* check whether > |Cl|/2 elements have order <= 2 ==> 2-elementary */
    2872         413 :     long lim = (h>>1) + 1;
    2873        1729 :     for (i=1; i <= lim; i++)
    2874        1323 :       if (!uhasexp2(gel(L,i))) break;
    2875         413 :     if (i > lim) return GenusFieldQuadImag(D);
    2876             :   }
    2877          49 :   if (DEBUGLEVEL>1) timer_printf(&ti,"class number = %ld",h);
    2878          49 :   init_pq(D, &T);
    2879          49 :   qfp = primeform_u(D, T.p);
    2880          49 :   T.pq =  muluu(T.p, T.q);
    2881          49 :   T.pq2 = shifti(T.pq,1);
    2882          49 :   if (T.p == T.q)
    2883             :   {
    2884           0 :     GEN qfbp2 = qfbcompraw(qfp, qfp);
    2885           0 :     u = gel(qfbp2,2);
    2886           0 :     T.u = modii(u, T.pq2);
    2887           0 :     T.qfpq = qfbred_i(qfbp2);
    2888             :   }
    2889             :   else
    2890             :   {
    2891          49 :     GEN qfq = primeform_u(D, T.q), bp = gel(qfp,2), bq = gel(qfq,2);
    2892          49 :     T.u = Z_chinese(bp, bq, utoipos(T.p << 1), utoipos(T.q << 1));
    2893             :     /* T.u = bp (mod 2p), T.u = bq (mod 2q) */
    2894          49 :     T.qfpq = qfbcomp_i(qfp, qfq);
    2895             :   }
    2896             :   /* u modulo 2pq */
    2897          49 :   prec = LOWDEFAULTPREC;
    2898          49 :   Pr = cgetg(h+1,t_VEC);
    2899          49 :   Pi = cgetg(h+1,t_VEC);
    2900             :   for(;;)
    2901          14 :   {
    2902          63 :     long ex, exmax = 0, r1 = 0, r2 = 0;
    2903          63 :     pari_sp av0 = avma;
    2904          63 :     T.sqd = sqrtr_abs(itor(D, prec));
    2905        4165 :     for (i=1; i<=h; i++)
    2906             :     {
    2907        4102 :       GEN s = gpq(gel(L,i), &T);
    2908        4102 :       if (DEBUGLEVEL>3) err_printf("%ld ", i);
    2909        4102 :       if (!s) continue;
    2910        2093 :       if (typ(s) != t_COMPLEX) gel(Pr, ++r1) = s; /* real root */
    2911        2009 :       else                     gel(Pi, ++r2) = s;
    2912        2093 :       ex = gexpo(s); if (ex > 0) exmax += ex;
    2913             :     }
    2914          63 :     if (DEBUGLEVEL>1) timer_printf(&ti,"roots");
    2915          63 :     setlg(Pr, r1+1);
    2916          63 :     setlg(Pi, r2+1);
    2917          63 :     P = roots_to_pol_r1(shallowconcat(Pr,Pi), 0, r1);
    2918          63 :     P = grndtoi(P,&exmax);
    2919          63 :     if (DEBUGLEVEL>1) timer_printf(&ti,"product, error bits = %ld",exmax);
    2920          63 :     if (exmax <= -10) break;
    2921          14 :     set_avma(av0); prec += nbits2extraprec(DEFAULTPREC + exmax);
    2922          14 :     if (DEBUGLEVEL) pari_warn(warnprec,"quadhilbertimag",prec);
    2923             :   }
    2924          49 :   return P;
    2925             : }
    2926             : 
    2927             : GEN
    2928         574 : quadhilbert(GEN D, long prec)
    2929             : {
    2930         574 :   pari_sp av = avma;
    2931         574 :   GEN d = D;
    2932         574 :   quadray_init(&d, NULL, 0);
    2933         973 :   return gerepileupto(av, signe(d)>0? quadhilbertreal(D,prec)
    2934         413 :                                     : quadhilbertimag(d));
    2935             : }
    2936             : 
    2937             : /* return a vector of all roots of 1 in bnf [not necessarily quadratic] */
    2938             : static GEN
    2939          70 : getallrootsof1(GEN bnf)
    2940             : {
    2941          70 :   GEN T, u, nf = bnf_get_nf(bnf), tu;
    2942          70 :   long i, n = bnf_get_tuN(bnf);
    2943             : 
    2944          70 :   if (n == 2) {
    2945          56 :     long N = nf_get_degree(nf);
    2946          56 :     return mkvec2(scalarcol_shallow(gen_m1, N),
    2947             :                   scalarcol_shallow(gen_1, N));
    2948             :   }
    2949          14 :   tu = poltobasis(nf, bnf_get_tuU(bnf));
    2950          14 :   T = zk_multable(nf, tu);
    2951          14 :   u = cgetg(n+1, t_VEC); gel(u,1) = tu;
    2952          56 :   for (i=2; i <= n; i++) gel(u,i) = ZM_ZC_mul(T, gel(u,i-1));
    2953          14 :   return u;
    2954             : }
    2955             : /* assume bnr has the right conductor */
    2956             : static GEN
    2957          70 : get_lambda(GEN bnr)
    2958             : {
    2959          70 :   GEN bnf = bnr_get_bnf(bnr), nf = bnf_get_nf(bnf), pol = nf_get_pol(nf);
    2960          70 :   GEN f = gel(bnr_get_mod(bnr), 1), labas, lamodf, u;
    2961          70 :   long a, b, f2, i, lu, v = varn(pol);
    2962             : 
    2963          70 :   f2 = 2 * itos(gcoeff(f,1,1));
    2964          70 :   u = getallrootsof1(bnf); lu = lg(u);
    2965         238 :   for (i=1; i<lu; i++)
    2966         168 :     gel(u,i) = ZC_hnfrem(gel(u,i), f); /* roots of 1, mod f */
    2967          70 :   if (DEBUGLEVEL>1)
    2968           0 :     err_printf("quadray: looking for [a,b] != unit mod 2f\n[a,b] = ");
    2969         168 :   for (a=0; a<f2; a++)
    2970        2576 :     for (b=0; b<f2; b++)
    2971             :     {
    2972        2478 :       GEN la = deg1pol_shallow(stoi(a), stoi(b), v); /* ax + b */
    2973        2478 :       if (umodiu(gnorm(mkpolmod(la, pol)), f2) != 1) continue;
    2974         224 :       if (DEBUGLEVEL>1) err_printf("[%ld,%ld] ",a,b);
    2975             : 
    2976         224 :       labas = poltobasis(nf, la);
    2977         224 :       lamodf = ZC_hnfrem(labas, f);
    2978         469 :       for (i=1; i<lu; i++)
    2979         399 :         if (ZV_equal(lamodf, gel(u,i))) break;
    2980         224 :       if (i < lu) continue; /* la = unit mod f */
    2981          70 :       if (DEBUGLEVEL)
    2982             :       {
    2983           0 :         if (DEBUGLEVEL>1) err_printf("\n");
    2984           0 :         err_printf("lambda = %Ps\n",la);
    2985             :       }
    2986          70 :       return labas;
    2987             :     }
    2988           0 :   pari_err_BUG("get_lambda");
    2989             :   return NULL;/*LCOV_EXCL_LINE*/
    2990             : }
    2991             : 
    2992             : static GEN
    2993        8778 : to_approx(GEN nf, GEN a)
    2994             : {
    2995        8778 :   GEN M = nf_get_M(nf);
    2996        8778 :   return gadd(gel(a,1), gmul(gcoeff(M,1,2),gel(a,2)));
    2997             : }
    2998             : /* Z-basis for a (over C) */
    2999             : static GEN
    3000        4354 : get_om(GEN nf, GEN a) {
    3001        4354 :   return mkvec2(to_approx(nf,gel(a,2)),
    3002        4354 :                 to_approx(nf,gel(a,1)));
    3003             : }
    3004             : 
    3005             : /* Compute all elts in class group G = [|G|,c,g], c=cyclic factors, g=gens.
    3006             :  * Set list[j + 1] = g1^e1...gk^ek where j is the integer
    3007             :  *   ek + ck [ e(k-1) + c(k-1) [... + c2 [e1]]...] */
    3008             : static GEN
    3009          70 : getallelts(GEN bnr)
    3010             : {
    3011             :   GEN nf, C, c, g, list, pows, gk;
    3012             :   long lc, i, j, no;
    3013             : 
    3014          70 :   nf = bnr_get_nf(bnr);
    3015          70 :   no = itos( bnr_get_no(bnr) );
    3016          70 :   c = bnr_get_cyc(bnr);
    3017          70 :   g = bnr_get_gen_nocheck(bnr); lc = lg(c)-1;
    3018          70 :   list = cgetg(no+1,t_VEC);
    3019          70 :   gel(list,1) = matid(nf_get_degree(nf)); /* (1) */
    3020          70 :   if (!no) return list;
    3021             : 
    3022          70 :   pows = cgetg(lc+1,t_VEC);
    3023          70 :   c = leafcopy(c); settyp(c, t_VECSMALL);
    3024         140 :   for (i=1; i<=lc; i++)
    3025             :   {
    3026          70 :     long k = itos(gel(c,i));
    3027          70 :     c[i] = k;
    3028          70 :     gk = cgetg(k, t_VEC); gel(gk,1) = gel(g,i);
    3029        4284 :     for (j=2; j<k; j++)
    3030        4214 :       gel(gk,j) = idealmoddivisor(bnr, idealmul(nf, gel(gk,j-1), gel(gk,1)));
    3031          70 :     gel(pows,i) = gk; /* powers of g[i] */
    3032             :   }
    3033             : 
    3034          70 :   C = cgetg(lc+1, t_VECSMALL); C[1] = c[lc];
    3035          70 :   for (i=2; i<=lc; i++) C[i] = C[i-1] * c[lc-i+1];
    3036             :   /* C[i] = c(k-i+1) * ... * ck */
    3037             :   /* j < C[i+1] <==> j only involves g(k-i)...gk */
    3038          70 :   i = 1;
    3039        4354 :   for (j=1; j < C[1]; j++)
    3040        4284 :     gel(list, j+1) = gmael(pows,lc,j);
    3041          70 :   while(j<no)
    3042             :   {
    3043             :     long k;
    3044             :     GEN a;
    3045           0 :     if (j == C[i+1]) i++;
    3046           0 :     a = gmael(pows,lc-i,j/C[i]);
    3047           0 :     k = j%C[i] + 1;
    3048           0 :     if (k > 1) a = idealmoddivisor(bnr, idealmul(nf, a, gel(list,k)));
    3049           0 :     gel(list, ++j) = a;
    3050             :   }
    3051          70 :   return list;
    3052             : }
    3053             : 
    3054             : /* x quadratic integer (approximate), recognize it. If error return NULL */
    3055             : static GEN
    3056        4424 : findbezk(GEN nf, GEN x)
    3057             : {
    3058        4424 :   GEN a,b, M = nf_get_M(nf), u = gcoeff(M,1,2);
    3059             :   long ea, eb;
    3060             : 
    3061             :   /* u t_COMPLEX generator of nf.zk, write x ~ a + b u, a,b in Z */
    3062        4424 :   b = grndtoi(mpdiv(imag_i(x), gel(u,2)), &eb);
    3063        4424 :   if (eb > -20) return NULL;
    3064        4424 :   a = grndtoi(mpsub(real_i(x), mpmul(b,gel(u,1))), &ea);
    3065        4424 :   if (ea > -20) return NULL;
    3066        4424 :   return signe(b)? coltoalg(nf, mkcol2(a,b)): a;
    3067             : }
    3068             : 
    3069             : static GEN
    3070          70 : findbezk_pol(GEN nf, GEN x)
    3071             : {
    3072          70 :   long i, lx = lg(x);
    3073          70 :   GEN y = cgetg(lx,t_POL);
    3074        4494 :   for (i=2; i<lx; i++)
    3075        4424 :     if (! (gel(y,i) = findbezk(nf,gel(x,i))) ) return NULL;
    3076          70 :   y[1] = x[1]; return y;
    3077             : }
    3078             : 
    3079             : /* P approximation computed at initial precision prec. Compute needed prec
    3080             :  * to know P with 1 word worth of trailing decimals */
    3081             : static long
    3082           0 : get_prec(GEN P, long prec)
    3083             : {
    3084           0 :   long k = gprecision(P);
    3085           0 :   if (k == 3) return precdbl(prec); /* approximation not trustworthy */
    3086           0 :   k = prec - k; /* lost precision when computing P */
    3087           0 :   if (k < 0) k = 0;
    3088           0 :   k += nbits2prec(gexpo(P) + 128);
    3089           0 :   if (k <= prec) k = precdbl(prec); /* dubious: old prec should have worked */
    3090           0 :   return k;
    3091             : }
    3092             : 
    3093             : /* Compute data for ellphist */
    3094             : static GEN
    3095        4354 : ellphistinit(GEN om, long prec)
    3096             : {
    3097        4354 :   GEN res,om1b,om2b, om1 = gel(om,1), om2 = gel(om,2);
    3098             : 
    3099        4354 :   if (gsigne(imag_i(gdiv(om1,om2))) < 0) { swap(om1,om2); om = mkvec2(om1,om2); }
    3100        4354 :   om1b = conj_i(om1);
    3101        4354 :   om2b = conj_i(om2); res = cgetg(4,t_VEC);
    3102        4354 :   gel(res,1) = gdivgu(elleisnum(om,2,0,prec),12);
    3103        4354 :   gel(res,2) = gdiv(PiI2(prec), gmul(om2, imag_i(gmul(om1b,om2))));
    3104        4354 :   gel(res,3) = om2b; return res;
    3105             : }
    3106             : 
    3107             : /* Computes log(phi^*(z,om)), using res computed by ellphistinit */
    3108             : static GEN
    3109        8708 : ellphist(GEN om, GEN res, GEN z, long prec)
    3110             : {
    3111        8708 :   GEN u = imag_i(gmul(z, gel(res,3)));
    3112        8708 :   GEN zst = gsub(gmul(u, gel(res,2)), gmul(z,gel(res,1)));
    3113        8708 :   return gsub(ellsigma(om,z,1,prec),gmul2n(gmul(z,zst),-1));
    3114             : }
    3115             : 
    3116             : /* Computes phi^*(la,om)/phi^*(1,om) where (1,om) is an oriented basis of the
    3117             :    ideal gf*gc^{-1} */
    3118             : static GEN
    3119        4354 : computeth2(GEN om, GEN la, long prec)
    3120             : {
    3121        4354 :   GEN p1,p2,res = ellphistinit(om,prec);
    3122             : 
    3123        4354 :   p1 = gsub(ellphist(om,res,la,prec), ellphist(om,res,gen_1,prec));
    3124        4354 :   p2 = imag_i(p1);
    3125        4354 :   if (gexpo(real_i(p1)) > 20 || gexpo(p2) > minss(prec,realprec(p2)) - 10)
    3126           0 :     return NULL;
    3127        4354 :   return gexp(p1,prec);
    3128             : }
    3129             : 
    3130             : /* Computes P_2(X)=polynomial in Z_K[X] closest to prod_gc(X-th2(gc)) where
    3131             :    the product is over the ray class group bnr.*/
    3132             : static GEN
    3133          70 : computeP2(GEN bnr, long prec)
    3134             : {
    3135          70 :   long clrayno, i, first = 1;
    3136          70 :   pari_sp av=avma, av2;
    3137          70 :   GEN listray, P0, P, lanum, la = get_lambda(bnr);
    3138          70 :   GEN nf = bnr_get_nf(bnr), f = gel(bnr_get_mod(bnr), 1);
    3139          70 :   listray = getallelts(bnr);
    3140          70 :   clrayno = lg(listray)-1; av2 = avma;
    3141          70 : PRECPB:
    3142          70 :   if (!first)
    3143             :   {
    3144           0 :     if (DEBUGLEVEL) pari_warn(warnprec,"computeP2",prec);
    3145           0 :     nf = gerepilecopy(av2, nfnewprec_shallow(bnr_get_nf(bnr),prec));
    3146             :   }
    3147          70 :   first = 0; lanum = to_approx(nf,la);
    3148          70 :   P = cgetg(clrayno+1,t_VEC);
    3149        4424 :   for (i=1; i<=clrayno; i++)
    3150             :   {
    3151        4354 :     GEN om = get_om(nf, idealdiv(nf,f,gel(listray,i)));
    3152        4354 :     GEN s = computeth2(om,lanum,prec);
    3153        4354 :     if (!s) { prec = precdbl(prec); goto PRECPB; }
    3154        4354 :     gel(P,i) = s;
    3155             :   }
    3156          70 :   P0 = roots_to_pol(P, 0);
    3157          70 :   P = findbezk_pol(nf, P0);
    3158          70 :   if (!P) { prec = get_prec(P0, prec); goto PRECPB; }
    3159          70 :   return gerepilecopy(av, P);
    3160             : }
    3161             : 
    3162             : #define nexta(a) (a>0 ? -a : 1-a)
    3163             : static GEN
    3164          49 : do_compo(GEN A0, GEN B)
    3165             : {
    3166          49 :   long a, i, l = lg(B), v = fetch_var_higher();
    3167             :   GEN A, z;
    3168             :   /* now v > x = pol_x(0) > nf variable */
    3169          49 :   B = leafcopy(B); setvarn(B, v);
    3170         210 :   for (i = 2; i < l; i++) gel(B,i) = monomial(gel(B,i), l-i-1, 0);
    3171             :   /* B := x^deg(B) B(v/x) */
    3172          49 :   A = A0 = leafcopy(A0); setvarn(A0, v);
    3173          56 :   for  (a = 0;; a = nexta(a))
    3174             :   {
    3175          56 :     if (a) A = RgX_translate(A0, stoi(a));
    3176          56 :     z = resultant(A,B); /* in variable 0 */
    3177          56 :     if (issquarefree(z)) break;
    3178             :   }
    3179          49 :   (void)delete_var(); return z;
    3180             : }
    3181             : #undef nexta
    3182             : 
    3183             : static GEN
    3184          14 : galoisapplypol(GEN nf, GEN s, GEN x)
    3185             : {
    3186          14 :   long i, lx = lg(x);
    3187          14 :   GEN y = cgetg(lx,t_POL);
    3188             : 
    3189          56 :   for (i=2; i<lx; i++) gel(y,i) = galoisapply(nf,s,gel(x,i));
    3190          14 :   y[1] = x[1]; return y;
    3191             : }
    3192             : /* x quadratic, write it as ua + v, u,v rational */
    3193             : static GEN
    3194          70 : findquad(GEN a, GEN x, GEN p)
    3195             : {
    3196             :   long tu, tv;
    3197          70 :   pari_sp av = avma;
    3198             :   GEN u,v;
    3199          70 :   if (typ(x) == t_POLMOD) x = gel(x,2);
    3200          70 :   if (typ(a) == t_POLMOD) a = gel(a,2);
    3201          70 :   u = poldivrem(x, a, &v);
    3202          70 :   u = simplify_shallow(u); tu = typ(u);
    3203          70 :   v = simplify_shallow(v); tv = typ(v);
    3204          70 :   if (!is_scalar_t(tu)) pari_err_TYPE("findquad", u);
    3205          70 :   if (!is_scalar_t(tv)) pari_err_TYPE("findquad", v);
    3206          70 :   x = deg1pol(u, v, varn(a));
    3207          70 :   if (typ(x) == t_POL) x = gmodulo(x,p);
    3208          70 :   return gerepileupto(av, x);
    3209             : }
    3210             : static GEN
    3211          14 : findquad_pol(GEN p, GEN a, GEN x)
    3212             : {
    3213          14 :   long i, lx = lg(x);
    3214          14 :   GEN y = cgetg(lx,t_POL);
    3215          84 :   for (i=2; i<lx; i++) gel(y,i) = findquad(a, gel(x,i), p);
    3216          14 :   y[1] = x[1]; return y;
    3217             : }
    3218             : /* m is 3, 4 or 12 */
    3219             : static GEN
    3220          35 : compocyclo(GEN D, long m)
    3221          35 : { return do_compo(quadhilbertimag(D), polcyclo(m,0)); }
    3222             : /* m is prime or 4 * prime */
    3223             : static GEN
    3224          14 : compocyclop(GEN nf, long m)
    3225             : {
    3226          14 :   GEN sb,a,b,s,p1,p2,p3,res,polL,polLK,nfL, D = nf_get_disc(nf);
    3227             :   long ell,vx;
    3228             : 
    3229          14 :   p1 = quadhilbertimag(D);
    3230          14 :   p2 = polcyclo(m,0);
    3231          14 :   ell = odd(m)? m: (m>>2); /* prime */
    3232          14 :   if (absequalui(ell,D)) /* ell = |D| */
    3233             :   {
    3234           0 :     p2 = gcoeff(nffactor(nf,p2),1,1);
    3235           0 :     return do_compo(p1,p2);
    3236             :   }
    3237          14 :   if (ell%4 == 3) ell = -ell;
    3238             :   /* nf = K = Q(a), L = K(b) quadratic extension = Q(t) */
    3239          14 :   polLK = quadpoly_i(stoi(ell)); /* relative polynomial */
    3240          14 :   res = rnfequation2(nf, polLK);
    3241          14 :   vx = nf_get_varn(nf);
    3242          14 :   polL = gsubst(gel(res,1),0,pol_x(vx)); /* = charpoly(t) */
    3243          14 :   a = gsubst(lift_shallow(gel(res,2)), 0,pol_x(vx));
    3244          14 :   b = gsub(pol_x(vx), gmul(gel(res,3), a));
    3245          14 :   nfL = nfinit(polL, DEFAULTPREC);
    3246          14 :   p1 = gcoeff(nffactor(nfL,p1),1,1);
    3247          14 :   p2 = gcoeff(nffactor(nfL,p2),1,1);
    3248          14 :   p3 = do_compo(p1,p2); /* relative equation over L */
    3249             :   /* compute non trivial s in Gal(L / K) */
    3250          14 :   sb= gneg(gadd(b, RgX_coeff(polLK,1))); /* s(b) = Tr(b) - b */
    3251          14 :   s = gadd(pol_x(vx), gsub(sb, b)); /* s(t) = t + s(b) - b */
    3252          14 :   p3 = gmul(p3, galoisapplypol(nfL, s, p3));
    3253          14 :   return findquad_pol(nf_get_pol(nf), a, p3);
    3254             : }
    3255             : 
    3256             : /* I integral ideal in HNF. (x) = I, x small in Z ? */
    3257             : static long
    3258         119 : isZ(GEN I)
    3259             : {
    3260         119 :   GEN x = gcoeff(I,1,1);
    3261         119 :   if (signe(gcoeff(I,1,2)) || !equalii(x, gcoeff(I,2,2))) return 0;
    3262         105 :   return is_bigint(x)? -1: itos(x);
    3263             : }
    3264             : 
    3265             : /* Treat special cases directly. return NULL if not special case */
    3266             : static GEN
    3267         119 : treatspecialsigma(GEN bnr)
    3268             : {
    3269         119 :   GEN bnf = bnr_get_bnf(bnr), nf = bnf_get_nf(bnf);
    3270         119 :   GEN f = gel(bnr_get_mod(bnr), 1),  D = nf_get_disc(nf);
    3271             :   GEN p1, p2;
    3272         119 :   long Ds, fl, tryf, i = isZ(f);
    3273             : 
    3274         119 :   if (i == 1) return quadhilbertimag(D); /* f = 1 */
    3275             : 
    3276         119 :   if (absequaliu(D,3)) /* Q(j) */
    3277             :   {
    3278           0 :     if (i == 4 || i == 5 || i == 7) return polcyclo(i,0);
    3279           0 :     if (!absequaliu(gcoeff(f,1,1),9) || !absequaliu(Z_content(f),3)) return NULL;
    3280             :     /* f = P_3^3 */
    3281           0 :     p1 = mkpolmod(bnf_get_tuU(bnf), nf_get_pol(nf));
    3282           0 :     return gadd(pol_xn(3,0), p1); /* x^3+j */
    3283             :   }
    3284         119 :   if (absequaliu(D,4)) /* Q(i) */
    3285             :   {
    3286          14 :     if (i == 3 || i == 5) return polcyclo(i,0);
    3287          14 :     if (i != 4) return NULL;
    3288           0 :     p1 = mkpolmod(bnf_get_tuU(bnf), nf_get_pol(nf));
    3289           0 :     return gadd(pol_xn(2,0), p1); /* x^2+i */
    3290             :   }
    3291         105 :   Ds = smodis(D,48);
    3292         105 :   if (i)
    3293             :   {
    3294          91 :     if (i==2 && Ds%16== 8) return compocyclo(D, 4);
    3295          84 :     if (i==3 && Ds% 3== 1) return compocyclo(D, 3);
    3296          70 :     if (i==4 && Ds% 8== 1) return compocyclo(D, 4);
    3297          63 :     if (i==6 && Ds   ==40) return compocyclo(D,12);
    3298          56 :     return NULL;
    3299             :   }
    3300             : 
    3301          14 :   p1 = gcoeff(f,1,1); /* integer > 0 */
    3302          14 :   tryf = itou_or_0(p1); if (!tryf) return NULL;
    3303          14 :   p2 = gcoeff(f,2,2); /* integer > 0 */
    3304          14 :   if (is_pm1(p2)) fl = 0;
    3305             :   else {
    3306           0 :     if (Ds % 16 != 8 || !absequaliu(Z_content(f),2)) return NULL;
    3307           0 :     fl = 1; tryf >>= 1;
    3308             :   }
    3309          14 :   if (tryf <= 3 || umodiu(D, tryf) || !uisprime(tryf)) return NULL;
    3310          14 :   if (fl) tryf <<= 2;
    3311          14 :   return compocyclop(nf, tryf);
    3312             : }
    3313             : 
    3314             : GEN
    3315         161 : quadray(GEN D, GEN f, long prec)
    3316             : {
    3317         161 :   GEN bnr, y, bnf, H = NULL;
    3318         161 :   pari_sp av = avma;
    3319             : 
    3320         161 :   if (isint1(f)) return quadhilbert(D, prec);
    3321         126 :   if (typ(D) == t_INT && typ(f) != t_INT)
    3322           0 :     pari_err_TYPE("quadray [conductor]", f);
    3323         126 :   quadray_init(&D, &bnf, prec);
    3324         126 :   bnr = Buchray(bnf, f, nf_INIT|nf_GEN);
    3325         126 :   if (is_pm1(bnr_get_no(bnr))) { set_avma(av); return pol_x(0); }
    3326         126 :   if (signe(D) > 0)
    3327           7 :     y = bnrstark(bnr, H, prec);
    3328             :   else
    3329             :   {
    3330         119 :     bnr_subgroup_sanitize(&bnr, &H);
    3331         119 :     y = treatspecialsigma(bnr);
    3332         119 :     if (!y) y = computeP2(bnr, prec);
    3333             :   }
    3334         126 :   return gerepileupto(av, y);
    3335             : }

Generated by: LCOV version 1.16