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 : #include "pari.h"
15 : #include "paripriv.h"
16 :
17 : /*********************************************************************/
18 : /** **/
19 : /** GENERIC ABELIAN CHARACTERS **/
20 : /** **/
21 : /*********************************************************************/
22 : /* check whether G is a znstar */
23 : int
24 3868929 : checkznstar_i(GEN G)
25 : {
26 3868712 : return (typ(G) == t_VEC && lg(G) == 6
27 3868334 : && typ(znstar_get_N(G)) == t_INT
28 3868327 : && typ(znstar_get_faN(G)) == t_VEC
29 7737641 : && typ(gel(G,1)) == t_VEC && lg(gel(G,1)) == 3);
30 : }
31 :
32 : int
33 120057 : char_check(GEN cyc, GEN chi)
34 120057 : { return typ(chi) == t_VEC && lg(chi) == lg(cyc) && RgV_is_ZV(chi); }
35 :
36 : /* Shallow; return [ d[1], d[1]/d[2],...,d[1]/d[n] ] */
37 : GEN
38 252637 : cyc_normalize(GEN d)
39 : {
40 252637 : long i, l = lg(d);
41 : GEN C, D;
42 252637 : if (l == 1) return mkvec(gen_1);
43 252616 : D = cgetg(l, t_VEC); gel(D,1) = C = gel(d,1);
44 597408 : for (i = 2; i < l; i++) gel(D,i) = diviiexact(C, gel(d,i));
45 252616 : return D;
46 : }
47 :
48 : /* chi character [D,C] given by chi(g_i) = \zeta_D^C[i] for all i, return
49 : * [d,c] such that chi(g_i) = \zeta_d^c[i] for all i and d minimal */
50 : GEN
51 293804 : char_simplify(GEN D, GEN C)
52 : {
53 293804 : GEN d = D;
54 293804 : if (lg(C) == 1) d = gen_1;
55 : else
56 : {
57 293258 : GEN t = gcdii(d, ZV_content(C));
58 293258 : if (!equali1(t))
59 : {
60 204001 : long tc = typ(C);
61 204001 : C = ZC_Z_divexact(C, t); settyp(C, tc);
62 204001 : d = diviiexact(d, t);
63 : }
64 : }
65 293804 : return mkvec2(d,C);
66 : }
67 :
68 : /* Shallow; ncyc from cyc_normalize(): ncyc[1] = cyc[1],
69 : * ncyc[i] = cyc[i]/cyc[1] for i > 1; chi character on G ~ cyc.
70 : * Return [d,c] such that: chi( g_i ) = e(chi[i] / cyc[i]) = e(c[i]/ d) */
71 : GEN
72 292901 : char_normalize(GEN chi, GEN ncyc)
73 : {
74 292901 : long i, l = lg(chi);
75 292901 : GEN c = cgetg(l, t_VEC);
76 292901 : if (l > 1) {
77 292880 : gel(c,1) = gel(chi,1);
78 739004 : for (i = 2; i < l; i++) gel(c,i) = mulii(gel(chi,i), gel(ncyc,i));
79 : }
80 292901 : return char_simplify(gel(ncyc,1), c);
81 : }
82 :
83 : /* Called by function 's'. x is a group object affording ".cyc" method, and
84 : * chi an abelian character. Return NULL if the group is (Z/nZ)^* [special
85 : * case more character types allowed] and x.cyc otherwise */
86 : static GEN
87 532 : get_cyc(GEN x, GEN chi, const char *s)
88 : {
89 532 : switch(nftyp(x))
90 : {
91 448 : case typ_BIDZ:
92 448 : if (!zncharcheck(x, chi)) pari_err_TYPE(s, chi);
93 448 : return NULL;
94 0 : case typ_GCHAR:
95 0 : x = gchar_get_cyc(x);
96 0 : if (!is_vec_t(typ(chi)) || lg(chi) != lg(x) || !RgV_is_ZV(chi))
97 0 : pari_err_TYPE(s, chi); /* FIXME: handle norm component */
98 0 : return x;
99 84 : default:
100 84 : if (typ(x) != t_VEC || !RgV_is_ZV(x)) x = member_cyc(x);
101 84 : if (!char_check(x, chi)) pari_err_TYPE(s, chi);
102 84 : return x;
103 : }
104 : }
105 :
106 : /* conjugate character [ZV/ZC] */
107 : GEN
108 6664 : charconj(GEN cyc, GEN chi)
109 : {
110 : long i, l;
111 6664 : GEN z = cgetg_copy(chi, &l);
112 11998 : for (i = 1; i < l; i++)
113 : {
114 5334 : GEN c = gel(chi,i);
115 5334 : gel(z,i) = signe(c)? subii(gel(cyc,i), c): gen_0;
116 : }
117 6664 : return z;
118 : }
119 : GEN
120 28 : charconj0(GEN x, GEN chi)
121 : {
122 28 : GEN cyc = get_cyc(x, chi, "charconj");
123 28 : return cyc? charconj(cyc, chi): zncharconj(x, chi);
124 : }
125 :
126 : GEN
127 1382136 : charorder(GEN cyc, GEN x)
128 : {
129 1382136 : pari_sp av = avma;
130 1382136 : long i, l = lg(cyc);
131 1382136 : GEN f = gen_1;
132 3613197 : for (i = 1; i < l; i++)
133 2231061 : if (signe(gel(x,i)))
134 : {
135 1274406 : GEN c, o = gel(cyc,i);
136 1274406 : if (!signe(o))
137 0 : return gerepileupto(av,mkoo());
138 1274406 : c = gcdii(o, gel(x,i));
139 1274406 : if (!is_pm1(c)) o = diviiexact(o,c);
140 1274406 : f = lcmii(f, o);
141 : }
142 1382136 : return gerepileuptoint(av, f);
143 : }
144 : GEN
145 210 : charorder0(GEN x, GEN chi)
146 : {
147 210 : GEN cyc = get_cyc(x, chi, "charorder");
148 210 : return cyc? charorder(cyc, chi): zncharorder(x, chi);
149 : }
150 :
151 : /* chi character of abelian G: chi[i] = chi(z_i), where G = \oplus Z/cyc[i] z_i.
152 : * Return Ker chi */
153 : GEN
154 97881 : charker(GEN cyc, GEN chi)
155 : {
156 97881 : long i, l = lg(cyc);
157 : GEN nchi, ncyc, m, U;
158 :
159 97881 : if (l == 1) return cgetg(1,t_MAT); /* trivial subgroup */
160 97832 : ncyc = cyc_normalize(cyc);
161 97832 : nchi = char_normalize(chi, ncyc);
162 97832 : m = shallowconcat(gel(nchi,2), gel(nchi,1));
163 97832 : U = gel(ZV_extgcd(m), 2); setlg(U,l);
164 267526 : for (i = 1; i < l; i++) setlg(U[i], l);
165 97832 : return hnfmodid(U, gel(ncyc,1));
166 : }
167 : GEN
168 35 : charker0(GEN x, GEN chi)
169 : {
170 35 : GEN cyc = get_cyc(x, chi, "charker");
171 35 : return cyc? charker(cyc, chi): zncharker(x, chi);
172 : }
173 :
174 : GEN
175 189 : charpow(GEN cyc, GEN a, GEN N)
176 : {
177 : long i, l;
178 189 : GEN v = cgetg_copy(a, &l);
179 350 : for (i = 1; i < l; i++) gel(v,i) = Fp_mul(gel(a,i), N, gel(cyc,i));
180 189 : return v;
181 : }
182 : GEN
183 302008 : charmul(GEN cyc, GEN a, GEN b)
184 : {
185 : long i, l;
186 302008 : GEN v = cgetg_copy(a, &l);
187 1350328 : for (i = 1; i < l; i++) gel(v,i) = Fp_add(gel(a,i), gel(b,i), gel(cyc,i));
188 302008 : return v;
189 : }
190 : GEN
191 5376 : chardiv(GEN cyc, GEN a, GEN b)
192 : {
193 : long i, l;
194 5376 : GEN v = cgetg_copy(a, &l);
195 11739 : for (i = 1; i < l; i++) gel(v,i) = Fp_sub(gel(a,i), gel(b,i), gel(cyc,i));
196 5376 : return v;
197 : }
198 : GEN
199 63 : charpow0(GEN x, GEN a, GEN N)
200 : {
201 63 : GEN cyc = get_cyc(x, a, "charpow");
202 63 : return cyc? charpow(cyc, a, N): zncharpow(x, a, N);
203 : }
204 : GEN
205 154 : charmul0(GEN x, GEN a, GEN b)
206 : {
207 154 : const char *s = "charmul";
208 154 : GEN cyc = get_cyc(x, a, s);
209 154 : if (!cyc)
210 : {
211 154 : if (!zncharcheck(x, b)) pari_err_TYPE(s, b);
212 154 : return zncharmul(x, a, b);
213 : }
214 : else
215 : {
216 0 : if (!char_check(cyc, b)) pari_err_TYPE(s, b);
217 0 : return charmul(cyc, a, b);
218 : }
219 : }
220 : GEN
221 42 : chardiv0(GEN x, GEN a, GEN b)
222 : {
223 42 : const char *s = "chardiv";
224 42 : GEN cyc = get_cyc(x, a, s);
225 42 : if (!cyc)
226 : {
227 42 : if (!zncharcheck(x, b)) pari_err_TYPE(s, b);
228 42 : return znchardiv(x, a, b);
229 : }
230 : else
231 : {
232 0 : if (!char_check(cyc, b)) pari_err_TYPE(s, b);
233 0 : return chardiv(cyc, a, b);
234 : }
235 : }
236 :
237 : static GEN
238 2520420 : chareval_i(GEN nchi, GEN dlog, GEN z)
239 : {
240 2520420 : GEN o, q, r, b = gel(nchi,1);
241 2520420 : GEN a = FpV_dotproduct(gel(nchi,2), dlog, b);
242 : /* image is a/b in Q/Z */
243 2520420 : if (!z) return gdiv(a,b);
244 2520042 : if (typ(z) == t_INT)
245 : {
246 2508646 : q = dvmdii(z, b, &r);
247 2508646 : if (signe(r)) pari_err_TYPE("chareval", z);
248 2508646 : return mulii(a, q);
249 : }
250 : /* return z^(a*o/b), assuming z^o = 1 and b | o */
251 11396 : if (typ(z) != t_VEC || lg(z) != 3) pari_err_TYPE("chareval", z);
252 11396 : o = gel(z,2); if (typ(o) != t_INT) pari_err_TYPE("chareval", z);
253 11396 : q = dvmdii(o, b, &r); if (signe(r)) pari_err_TYPE("chareval", z);
254 11396 : q = mulii(a, q); /* in [0, o[ since a is reduced mod b */
255 11396 : z = gel(z,1);
256 11396 : if (typ(z) == t_VEC)
257 : {
258 10080 : if (itos_or_0(o) != lg(z)-1) pari_err_TYPE("chareval", z);
259 10080 : return gcopy(gel(z, itos(q)+1));
260 : }
261 : else
262 1316 : return gpow(z, q, DEFAULTPREC);
263 : }
264 :
265 : static GEN
266 1855 : not_coprime(GEN z)
267 1855 : { return (!z || typ(z) == t_INT)? gen_m1: gen_0; }
268 :
269 : static GEN
270 35 : get_chi(GEN cyc, GEN chi)
271 : {
272 35 : if (!char_check(cyc,chi)) pari_err_TYPE("chareval", chi);
273 35 : return char_normalize(chi, cyc_normalize(cyc));
274 : }
275 : /* G a bnr. FIXME: horribly inefficient to check that (x,N)=1, what to do ? */
276 : static int
277 42 : bnr_coprime(GEN G, GEN x)
278 : {
279 42 : GEN t, N = gel(bnr_get_mod(G), 1);
280 42 : if (typ(x) == t_INT) /* shortcut */
281 : {
282 14 : t = gcdii(gcoeff(N,1,1), x);
283 14 : if (equali1(t)) return 1;
284 0 : t = idealadd(G, N, x);
285 0 : return equali1(gcoeff(t,1,1));
286 : }
287 28 : x = idealnumden(G, x);
288 28 : t = idealadd(G, N, gel(x,1));
289 28 : if (!equali1(gcoeff(t,1,1))) return 0;
290 21 : t = idealadd(G, N, gel(x,2));
291 21 : return equali1(gcoeff(t,1,1));
292 : }
293 : GEN
294 3605 : chareval(GEN G, GEN chi, GEN x, GEN z)
295 : {
296 3605 : pari_sp av = avma;
297 : GEN nchi, L;
298 :
299 3605 : switch(nftyp(G))
300 : {
301 42 : case typ_BNR:
302 42 : if (!bnr_coprime(G, x)) return not_coprime(z);
303 28 : L = isprincipalray(G, x);
304 28 : nchi = get_chi(bnr_get_cyc(G), chi);
305 28 : break;
306 7 : case typ_BNF:
307 7 : L = isprincipal(G, x);
308 7 : nchi = get_chi(bnf_get_cyc(G), chi);
309 7 : break;
310 3542 : case typ_BIDZ:
311 3542 : if (checkznstar_i(G)) return gerepileupto(av, znchareval(G, chi, x, z));
312 : /* don't implement chars on general bid: need an nf... */
313 : case typ_GCHAR:
314 7 : pari_err_TYPE("chareval [use gchareval to evaluate a grossencharacter]", G);
315 7 : default:
316 7 : pari_err_TYPE("chareval", G);
317 : return NULL;/* LCOV_EXCL_LINE */
318 : }
319 35 : return gerepileupto(av, chareval_i(nchi, L, z));
320 : }
321 :
322 : /* nchi = [ord,D] a quasi-normalized character (ord may be a multiple of
323 : * the character order); return v such that v[n] = -1 if (n,N) > 1 else
324 : * chi(n) = e(v[n]/ord), 1 <= n <= N */
325 : GEN
326 36505 : ncharvecexpo(GEN G, GEN nchi)
327 : {
328 36505 : long N = itou(znstar_get_N(G)), ord = itou(gel(nchi,1)), i, j, l;
329 : GEN cyc, gen, d, t, t1, t2, t3, e, u, u1, u2, u3;
330 36505 : GEN D = gel(nchi,2), v = const_vecsmall(N,-1);
331 36505 : pari_sp av = avma;
332 36505 : if (typ(D) == t_COL) {
333 36505 : cyc = znstar_get_conreycyc(G);
334 36505 : gen = znstar_get_conreygen(G);
335 : } else {
336 0 : cyc = znstar_get_cyc(G);
337 0 : gen = znstar_get_gen(G);
338 : }
339 36505 : l = lg(cyc);
340 36505 : e = u = cgetg(N+1,t_VECSMALL);
341 36505 : d = t = cgetg(N+1,t_VECSMALL);
342 36505 : *++d = 1;
343 36505 : *++e = 0; v[*d] = *e;
344 77420 : for (i = 1; i < l; i++)
345 : {
346 40915 : ulong g = itou(gel(gen,i)), c = itou(gel(cyc,i)), x = itou(gel(D,i));
347 549262 : for (t1=t,u1=u,j=c-1; j; j--,t1=t2,u1=u2)
348 1128526 : for (t2=d,u2=e, t3=t1,u3=u1; t3<t2; )
349 : {
350 620179 : *++d = Fl_mul(*++t3, g, N);
351 620179 : *++e = Fl_add(*++u3, x, ord); v[*d] = *e;
352 : }
353 : }
354 36505 : set_avma(av); return v;
355 : }
356 :
357 : /*****************************************************************************/
358 :
359 : static ulong
360 257075 : lcmuu(ulong a, ulong b) { return (a/ugcd(a,b)) * b; }
361 : static ulong
362 101248 : zv_charorder(GEN cyc, GEN x)
363 : {
364 101248 : long i, l = lg(cyc);
365 101248 : ulong f = 1;
366 332115 : for (i = 1; i < l; i++)
367 230867 : if (x[i])
368 : {
369 155827 : ulong o = cyc[i];
370 155827 : f = lcmuu(f, o / ugcd(o, x[i]));
371 : }
372 101248 : return f;
373 : }
374 :
375 : /* N > 0 */
376 : GEN
377 549969 : coprimes_zv(ulong N)
378 : {
379 549969 : GEN v = const_vecsmall(N,1);
380 549969 : pari_sp av = avma;
381 549969 : GEN P = gel(factoru(N),1);
382 549969 : long i, l = lg(P);
383 1282309 : for (i = 1; i < l; i++)
384 : {
385 732340 : ulong p = P[i], j;
386 3414453 : for (j = p; j <= N; j += p) v[j] = 0;
387 : }
388 549969 : set_avma(av); return v;
389 : }
390 : /* cf zv_cyc_minimal: return k such that g*k is minimal (wrt lex) */
391 : long
392 46627 : zv_cyc_minimize(GEN cyc, GEN g, GEN coprime)
393 : {
394 46627 : pari_sp av = avma;
395 46627 : long d, k, e, i, maxi, k0, bestk, l = lg(g), o = lg(coprime)-1;
396 : GEN best, gk, gd;
397 : ulong t;
398 46627 : if (o == 1) return 1;
399 54187 : for (i = 1; i < l; i++)
400 54187 : if (g[i]) break;
401 46627 : if (g[i] == 1) return 1;
402 38423 : k0 = Fl_invgen(g[i], cyc[i], &t);
403 38423 : d = cyc[i] / (long)t;
404 38423 : if (k0 > 1) g = vecmoduu(Flv_Fl_mul(g, k0, cyc[i]), cyc);
405 50785 : for (i++; i < l; i++)
406 44751 : if (g[i]) break;
407 38423 : if (i == l) return k0;
408 32389 : cyc = vecslice(cyc,i,l-1);
409 32389 : g = vecslice(g, i,l-1);
410 32389 : e = cyc[1];
411 32389 : gd = Flv_Fl_mul(g, d, e);
412 32389 : bestk = 1; best = g; maxi = e/ugcd(d,e);
413 48643 : for (gk = g, k = d+1, i = 1; i < maxi; k += d, i++)
414 : {
415 16254 : long ko = k % o;
416 16254 : gk = Flv_add(gk, gd, e); if (!ko || !coprime[ko]) continue;
417 7357 : gk = vecmoduu(gk, cyc);
418 7357 : if (vecsmall_lexcmp(gk, best) < 0) { best = gk; bestk = k; }
419 : }
420 32389 : return gc_long(av, bestk == 1? k0: (long) Fl_mul(k0, bestk, o));
421 : }
422 : /* g of order o in abelian group G attached to cyc. Is g a minimal generator
423 : * [wrt lex order] of the cyclic subgroup it generates;
424 : * coprime = coprimes_zv(o) */
425 : long
426 100800 : zv_cyc_minimal(GEN cyc, GEN g, GEN coprime)
427 : {
428 100800 : pari_sp av = avma;
429 100800 : long i, maxi, d, k, e, l = lg(g), o = lg(coprime)-1; /* elt order */
430 : GEN gd, gk;
431 100800 : if (o == 1) return 1;
432 106407 : for (k = 1; k < l; k++)
433 106407 : if (g[k]) break;
434 100800 : if (g[k] == 1) return 1;
435 80332 : if (cyc[k] % g[k]) return 0;
436 80332 : d = cyc[k] / g[k]; /* > 1 */
437 97069 : for (k++; k < l; k++) /* skip following 0s */
438 97069 : if (g[k]) break;
439 80332 : if (k == l) return 1;
440 80332 : cyc = vecslice(cyc,k,l-1);
441 80332 : g = vecslice(g, k,l-1);
442 80332 : e = cyc[1];
443 : /* find k in (Z/e)^* such that g*k mod cyc is lexicographically minimal,
444 : * k = 1 mod d to fix the first nonzero entry */
445 80332 : gd = Flv_Fl_mul(g, d, e); maxi = e/ugcd(d,e);
446 138159 : for (gk = g, k = d+1, i = 1; i < maxi; i++, k += d)
447 : {
448 67823 : long ko = k % o;
449 67823 : gk = Flv_add(gk, gd, e); if (!coprime[ko]) continue;
450 36141 : gk = vecmoduu(gk, cyc);
451 36141 : if (vecsmall_lexcmp(gk, g) < 0) return gc_long(av,0);
452 : }
453 70336 : return gc_long(av,1);
454 : }
455 :
456 : static GEN
457 8008 : coprime_tables(long N)
458 : {
459 8008 : GEN D = divisorsu(N), v = const_vec(N, NULL);
460 8008 : long i, l = lg(D);
461 47481 : for (i = 1; i < l; i++) gel(v, D[i]) = coprimes_zv(D[i]);
462 8008 : return v;
463 : }
464 : /* enumerate all group elements, modulo (Z/cyc[1])^* */
465 : static GEN
466 8036 : cyc2elts_normal(GEN cyc, long maxord, GEN ORD)
467 : {
468 8036 : long i, n, o, N, j = 1;
469 : GEN z, vcoprime;
470 :
471 8036 : if (typ(cyc) != t_VECSMALL) cyc = vec_to_vecsmall(cyc);
472 8036 : n = lg(cyc)-1;
473 8036 : N = zv_prod(cyc);
474 8036 : z = cgetg(N+1, t_VEC);
475 8036 : if (1 <= maxord && (!ORD|| zv_search(ORD,1)))
476 7651 : gel(z,j++) = zero_zv(n);
477 8036 : if (n == 0) { setlg(z, j); return z; }
478 8008 : vcoprime = coprime_tables(cyc[1]);
479 23184 : for (i = n; i > 0; i--)
480 : {
481 15176 : GEN cyc0 = vecslice(cyc,i+1,n), pre = zero_zv(i);
482 15176 : GEN D = divisorsu(cyc[i]), C = cyc2elts(cyc0);
483 15176 : long s, t, lD = lg(D), nC = lg(C)-1; /* remove last element */
484 54810 : for (s = 1; s < lD-1; s++)
485 : {
486 39634 : long o0 = D[lD-s]; /* cyc[i] / D[s] */
487 39634 : if (o0 > maxord) continue;
488 38143 : pre[i] = D[s];
489 38143 : if (!ORD || zv_search(ORD,o0))
490 : {
491 37842 : GEN c = vecsmall_concat(pre, zero_zv(n-i));
492 37842 : gel(z,j++) = c;
493 : }
494 139391 : for (t = 1; t < nC; t++)
495 : {
496 101248 : GEN chi0 = gel(C,t);
497 101248 : o = lcmuu(o0, zv_charorder(cyc0,chi0));
498 101248 : if (o <= maxord && (!ORD || zv_search(ORD,o)))
499 : {
500 100800 : GEN c = vecsmall_concat(pre, chi0);
501 100800 : if (zv_cyc_minimal(cyc, c, gel(vcoprime,o))) gel(z,j++) = c;
502 : }
503 : }
504 : }
505 : }
506 8008 : setlg(z,j); return z;
507 : }
508 :
509 : GEN
510 10479 : chargalois(GEN G, GEN ORD)
511 : {
512 10479 : pari_sp av = avma;
513 : long maxord, i, l;
514 10479 : GEN v, cyc = (typ(G) == t_VEC && RgV_is_ZVpos(G))? G: member_cyc(G);
515 10479 : if (lg(cyc) == 1 && !ORD) retmkvec(cgetg(1,t_VEC));
516 8043 : maxord = itou(cyc_get_expo(cyc));
517 8043 : if (ORD)
518 441 : switch(typ(ORD))
519 : {
520 : long l;
521 42 : case t_VEC:
522 42 : ORD = ZV_to_zv(ORD);
523 406 : case t_VECSMALL:
524 406 : l = lg(ORD);
525 406 : if (l > 2)
526 : {
527 371 : ORD = leafcopy(ORD);
528 371 : vecsmall_sort(ORD);
529 : }
530 406 : if (l == 1) { set_avma(av); return cgetg(1, t_VEC); }
531 399 : maxord = minss(maxord, ORD[l-1]);
532 399 : break;
533 35 : case t_INT:
534 35 : maxord = minss(maxord, itos(ORD));
535 35 : ORD = NULL;
536 35 : break;
537 0 : default: pari_err_TYPE("chargalois", ORD);
538 : }
539 8036 : v = cyc2elts_normal(cyc, maxord, ORD); l = lg(v);
540 144333 : for(i = 1; i < l; i++) gel(v,i) = zv_to_ZV(gel(v,i));
541 8036 : return gerepileupto(av, v);
542 : }
543 :
544 : /*********************************************************************/
545 : /** **/
546 : /** (Z/NZ)^* AND DIRICHLET CHARACTERS **/
547 : /** **/
548 : /*********************************************************************/
549 :
550 : GEN
551 130933 : znstar0(GEN N, long flag)
552 : {
553 130933 : GEN F = NULL, P, E, cyc, gen, mod, G;
554 : long i, i0, l, nbprimes;
555 130933 : pari_sp av = avma;
556 :
557 130933 : if (flag && flag != 1) pari_err_FLAG("znstar");
558 130933 : if ((F = check_arith_all(N,"znstar")))
559 : {
560 10003 : F = clean_Z_factor(F);
561 10003 : N = typ(N) == t_VEC? gel(N,1): factorback(F);
562 : }
563 130933 : if (!signe(N))
564 : {
565 21 : if (flag) pari_err_IMPL("znstar(0,1)");
566 14 : set_avma(av);
567 14 : retmkvec3(gen_2, mkvec(gen_2), mkvec(gen_m1));
568 : }
569 130912 : N = absi_shallow(N);
570 130912 : if (abscmpiu(N,2) <= 0)
571 : {
572 10535 : G = mkvec3(gen_1, cgetg(1,t_VEC), cgetg(1,t_VEC));
573 10535 : if (flag)
574 : {
575 10472 : GEN v = const_vec(6,cgetg(1,t_VEC));
576 10472 : gel(v,3) = cgetg(1,t_MAT);
577 20909 : F = equali1(N)? mkvec2(cgetg(1,t_COL),cgetg(1,t_VECSMALL))
578 10472 : : mkvec2(mkcol(gen_2), mkvecsmall(1));
579 10472 : G = mkvec5(mkvec2(N,mkvec(gen_0)), G, F, v, cgetg(1,t_MAT));
580 : }
581 10535 : return gerepilecopy(av,G);
582 : }
583 120377 : if (!F) F = Z_factor(N);
584 120377 : P = gel(F,1); nbprimes = lg(P)-1;
585 120377 : E = ZV_to_nv( gel(F,2) );
586 120377 : switch(mod8(N))
587 : {
588 20354 : case 0:
589 20354 : P = shallowconcat(gen_2,P);
590 20354 : E = vecsmall_prepend(E, E[1]); /* add a copy of p=2 row */
591 20354 : i = 2; /* 2 generators at 2 */
592 20354 : break;
593 19235 : case 4:
594 19235 : i = 1; /* 1 generator at 2 */
595 19235 : break;
596 7014 : case 2: case 6:
597 7014 : P = vecsplice(P,1);
598 7014 : E = vecsplice(E,1); /* remove 2 */
599 7014 : i = 0; /* no generator at 2 */
600 7014 : break;
601 73774 : default:
602 73774 : i = 0; /* no generator at 2 */
603 73774 : break;
604 : }
605 120377 : l = lg(P);
606 120377 : cyc = cgetg(l,t_VEC);
607 120377 : gen = cgetg(l,t_VEC);
608 120377 : mod = cgetg(l,t_VEC);
609 : /* treat p=2 first */
610 120377 : if (i == 2)
611 : {
612 20354 : long v2 = E[1];
613 20354 : GEN q = int2n(v2);
614 20354 : gel(cyc,1) = gen_2;
615 20354 : gel(gen,1) = subiu(q,1); /* -1 */
616 20354 : gel(mod,1) = q;
617 20354 : gel(cyc,2) = int2n(v2-2);
618 20354 : gel(gen,2) = utoipos(5); /* Conrey normalization */
619 20354 : gel(mod,2) = q;
620 20354 : i0 = 3;
621 : }
622 100023 : else if (i == 1)
623 : {
624 19235 : gel(cyc,1) = gen_2;
625 19235 : gel(gen,1) = utoipos(3);
626 19235 : gel(mod,1) = utoipos(4);
627 19235 : i0 = 2;
628 : }
629 : else
630 80788 : i0 = 1;
631 : /* odd primes, fill remaining entries */
632 290932 : for (i = i0; i < l; i++)
633 : {
634 170555 : long e = E[i];
635 170555 : GEN p = gel(P,i), q = powiu(p, e-1), Q = mulii(p, q);
636 170555 : gel(cyc,i) = subii(Q, q); /* phi(p^e) */
637 170555 : gel(gen,i) = pgener_Zp(p);/* Conrey normalization, for e = 1 also */
638 170555 : gel(mod,i) = Q;
639 : }
640 : /* gen[i] has order cyc[i] and generates (Z/mod[i]Z)^* */
641 120377 : if (nbprimes > 1) /* lift generators to (Z/NZ)^*, = 1 mod N/mod[i] */
642 243907 : for (i=1; i<l; i++)
643 : {
644 175614 : GEN Q = gel(mod,i), g = gel(gen,i), qinv = Fp_inv(Q, diviiexact(N,Q));
645 175614 : g = addii(g, mulii(mulii(subsi(1,g),qinv),Q));
646 175614 : gel(gen,i) = modii(g, N);
647 : }
648 :
649 : /* cyc[i] > 1 and remain so in the loop, gen[i] = 1 mod (N/mod[i]) */
650 120377 : if (!flag)
651 : { /* update generators in place; about twice faster */
652 35802 : G = gen;
653 36019 : for (i=l-1; i>=2; i--)
654 : {
655 217 : GEN ci = gel(cyc,i), gi = gel(G,i);
656 : long j;
657 518 : for (j=i-1; j>=1; j--) /* we want cyc[i] | cyc[j] */
658 : {
659 301 : GEN cj = gel(cyc,j), gj, qj, v, d;
660 :
661 301 : d = bezout(ci,cj,NULL,&v); /* > 1 */
662 455 : if (absequalii(ci, d)) continue; /* ci | cj */
663 189 : if (absequalii(cj, d)) { /* cj | ci */
664 154 : swap(gel(G,j),gel(G,i));
665 154 : gi = gel(G,i);
666 154 : swap(gel(cyc,j),gel(cyc,i));
667 154 : ci = gel(cyc,i); continue;
668 : }
669 :
670 35 : qj = diviiexact(cj,d);
671 35 : gel(cyc,j) = mulii(ci,qj);
672 35 : gel(cyc,i) = d;
673 :
674 : /* [1,v*cj/d; 0,1]*[1,0;-1,1]*diag(cj,ci)*[ci/d,-v; cj/d,u]
675 : * = diag(lcm,gcd), with u ci + v cj = d */
676 35 : gj = gel(G,j);
677 : /* (gj, gi) *= [1,0; -1,1]^-1 */
678 35 : gj = Fp_mul(gj, gi, N); /* order ci*qj = lcm(ci,cj) */
679 : /* (gj,gi) *= [1,v*qj; 0,1]^-1 */
680 35 : togglesign_safe(&v);
681 35 : if (signe(v) < 0) v = modii(v,ci); /* >= 0 to avoid inversions */
682 35 : gel(G,i) = gi = Fp_mul(gi, Fp_pow(gj, mulii(qj, v), N), N);
683 35 : gel(G,j) = gj;
684 35 : ci = d; if (absequaliu(ci, 2)) break;
685 : }
686 : }
687 35802 : G = mkvec3(ZV_prod(cyc), cyc, FpV_to_mod(G,N));
688 : }
689 : else
690 : { /* keep matrices between generators, return an 'init' structure */
691 84575 : GEN D, U, Ui, fao = cgetg(l, t_VEC), lo = cgetg(l, t_VEC);
692 84575 : F = mkvec2(P, E);
693 84575 : D = ZV_snf_group(cyc,&U,&Ui);
694 279054 : for (i = 1; i < l; i++)
695 : {
696 194479 : GEN t = gen_0, p = gel(P,i), p_1 = subiu(p,1);
697 194479 : long e = E[i];
698 194479 : gel(fao,i) = get_arith_ZZM(p_1);
699 194479 : if (e >= 2 && !absequaliu(p,2))
700 : {
701 12923 : GEN q = gel(mod,i), g = Fp_pow(gel(gen,i),p_1,q);
702 12923 : if (e == 2)
703 10550 : t = Fp_inv(diviiexact(subiu(g,1), p), p);
704 : else
705 2373 : t = ginv(Qp_log(cvtop(g,p,e)));
706 : }
707 194479 : gel(lo,i) = t;
708 : }
709 84575 : G = cgetg(l, t_VEC);
710 279054 : for (i = 1; i < l; i++) gel(G,i) = FpV_factorback(gen, gel(Ui,i), N);
711 84575 : G = mkvec3(ZV_prod(D), D, G);
712 84575 : G = mkvec5(mkvec2(N,mkvec(gen_0)), G, F,
713 : mkvecn(6,mod,fao,Ui,gen,cyc,lo), U);
714 : }
715 120377 : return gerepilecopy(av, G);
716 : }
717 : GEN
718 35788 : znstar(GEN N) { return znstar0(N, 0); }
719 :
720 : /* g has order 2^(e-2), g,h = 1 (mod 4); return x s.t. g^x = h (mod 2^e) */
721 : static GEN
722 949576 : Zideallog_2k(GEN h, GEN g, long e, GEN pe)
723 : {
724 949576 : GEN a = Fp_log(h, g, int2n(e-2), pe);
725 949576 : if (typ(a) != t_INT) return NULL;
726 949576 : return a;
727 : }
728 :
729 : /* ord = get_arith_ZZM(p-1), simplified form of znlog_rec: g is known
730 : * to be a primitive root mod p^e; lo = 1/log_p(g^(p-1)) */
731 : static GEN
732 2000050 : Zideallog_pk(GEN h, GEN g, GEN p, long e, GEN pe, GEN ord, GEN lo)
733 : {
734 2000050 : GEN gp = (e == 1)? g: modii(g, p);
735 2000050 : GEN hp = (e == 1)? h: modii(h, p);
736 2000050 : GEN a = Fp_log(hp, gp, ord, p);
737 2000050 : if (typ(a) != t_INT) return NULL;
738 2000043 : if (e > 1)
739 : { /* find a s.t. g^a = h (mod p^e), p odd prime, e > 0, (h,p) = 1 */
740 : /* use p-adic log: O(log p + e) mul*/
741 53432 : GEN b, p_1 = gel(ord,1);
742 53432 : h = Fp_mul(h, Fp_pow(g, negi(a), pe), pe);
743 : /* g,h = 1 mod p; compute b s.t. h = g^b */
744 53432 : if (e == 2) /* simpler */
745 46131 : b = Fp_mul(diviiexact(subiu(h,1), p), lo, p);
746 : else
747 7301 : b = padic_to_Q(gmul(Qp_log(cvtop(h, p, e)), lo));
748 53432 : a = addii(a, mulii(p_1, b));
749 : }
750 2000043 : return a;
751 : }
752 :
753 : int
754 902601 : znconrey_check(GEN cyc, GEN chi)
755 902601 : { return typ(chi) == t_COL && lg(chi) == lg(cyc) && RgV_is_ZV(chi); }
756 :
757 : int
758 207088 : zncharcheck(GEN G, GEN chi)
759 : {
760 207088 : switch(typ(chi))
761 : {
762 875 : case t_INT: return 1;
763 197715 : case t_COL: return znconrey_check(znstar_get_conreycyc(G), chi);
764 8498 : case t_VEC: return char_check(znstar_get_cyc(G), chi);
765 : }
766 0 : return 0;
767 : }
768 :
769 : GEN
770 153055 : znconreyfromchar_normalized(GEN bid, GEN chi)
771 : {
772 153055 : GEN nchi, U = znstar_get_U(bid);
773 153055 : long l = lg(chi);
774 153055 : if (l == 1) retmkvec2(gen_1,cgetg(1,t_VEC));
775 150612 : if (!RgV_is_ZV(chi) || lgcols(U) != l) pari_err_TYPE("lfunchiZ", chi);
776 150605 : nchi = char_normalize(chi, cyc_normalize(znstar_get_cyc(bid)));
777 150605 : gel(nchi,2) = ZV_ZM_mul(gel(nchi,2),U); return nchi;
778 : }
779 :
780 : GEN
781 142919 : znconreyfromchar(GEN bid, GEN chi)
782 : {
783 142919 : GEN nchi = znconreyfromchar_normalized(bid, chi);
784 142912 : GEN v = char_denormalize(znstar_get_conreycyc(bid), gel(nchi,1), gel(nchi,2));
785 142912 : settyp(v, t_COL); return v;
786 : }
787 :
788 : /* discrete log on canonical "primitive root" generators
789 : * Allow log(x) instead of x [usual discrete log on bid's generators] */
790 : GEN
791 2605884 : znconreylog(GEN bid, GEN x)
792 : {
793 2605884 : pari_sp av = avma;
794 : GEN N, L, F, P,E, y, pe, fao, gen, lo, cycg;
795 : long i, l;
796 2605884 : if (!checkznstar_i(bid)) pari_err_TYPE("znconreylog", bid);
797 2605877 : N = znstar_get_N(bid);
798 2605877 : if (abscmpiu(N, 2) <= 0)
799 : {
800 12376 : switch(typ(x))
801 : {
802 2674 : case t_INT: break;
803 7 : case t_INTMOD:
804 7 : if (!equalii(N, gel(x,1))) pari_err_TYPE("znconreylog", x);
805 0 : x = gel(x,2); break;
806 9688 : case t_COL:
807 : case t_VEC:
808 9688 : if (lg(x) != 1) pari_err_TYPE("znconreylog", x);
809 9674 : break;
810 7 : default: pari_err_TYPE("znconreylog", x);
811 : }
812 12348 : return cgetg(1, t_COL);
813 : }
814 2593501 : cycg = znstar_get_conreycyc(bid);
815 2593501 : switch(typ(x))
816 : {
817 : GEN Ui;
818 0 : case t_INTMOD:
819 0 : if (!equalii(N, gel(x,1))) pari_err_TYPE("znconreylog", x);
820 0 : x = gel(x,2); /* fall through */
821 2559383 : case t_INT:
822 2559383 : if (!signe(x)) pari_err_COPRIME("znconreylog", x, N);
823 2559376 : break;
824 35 : case t_COL: /* log_bid(x) */
825 35 : Ui = znstar_get_Ui(bid);
826 35 : if (!RgV_is_ZV(x) || lg(x) != lg(Ui)) pari_err_TYPE("znconreylog", x);
827 35 : return gerepileupto(av, ZV_ZV_mod(ZM_ZC_mul(Ui,x), cycg));
828 34083 : case t_VEC:
829 34083 : return gerepilecopy(av, znconreyfromchar(bid, x));
830 0 : default: pari_err_TYPE("znconreylog", x);
831 : }
832 2559376 : F = znstar_get_faN(bid); /* factor(N) */
833 2559376 : P = gel(F, 1); /* prime divisors of N */
834 2559376 : E = gel(F, 2); /* exponents */
835 2559376 : L = gel(bid,4);
836 2559376 : pe = znstar_get_pe(bid);
837 2559376 : fao = gel(L,2);
838 2559376 : gen = znstar_get_conreygen(bid); /* local generators of (Z/p^k)^* */
839 2559376 : lo = gel(L,6); /* 1/log_p((g_i)^(p_i-1)) */
840 :
841 2559376 : l = lg(gen); i = 1;
842 2559376 : y = cgetg(l, t_COL);
843 2559376 : if (!mod2(N) && !mod2(x)) pari_err_COPRIME("znconreylog", x, N);
844 2559362 : if (absequaliu(gel(P,1), 2) && E[1] >= 2)
845 : {
846 1259759 : if (E[1] == 2)
847 310183 : gel(y,i++) = mod4(x) == 1? gen_0: gen_1;
848 : else
849 : {
850 949576 : GEN a, x2, q2 = gel(pe,1);
851 949576 : x2 = modii(x, q2);
852 949576 : if (mod4(x) == 1) /* 1 or 5 mod 8*/
853 592362 : gel(y,i++) = gen_0;
854 : else /* 3 or 7 */
855 357214 : { gel(y,i++) = gen_1; x2 = subii(q2, x2); }
856 : /* x2 = 5^x mod q */
857 949576 : a = Zideallog_2k(x2, gel(gen,i), E[1], q2);
858 949576 : if (!a) pari_err_COPRIME("znconreylog", x, N);
859 949576 : gel(y, i++) = a;
860 : }
861 : }
862 4559405 : while (i < l)
863 : {
864 2000050 : GEN p = gel(P,i), q = gel(pe,i), xpe = modii(x, q);
865 2000050 : GEN a = Zideallog_pk(xpe, gel(gen,i), p, E[i], q, gel(fao,i), gel(lo,i));
866 2000050 : if (!a) pari_err_COPRIME("znconreylog", x, N);
867 2000043 : gel(y, i++) = a;
868 : }
869 2559355 : return gerepilecopy(av, y);
870 : }
871 : GEN
872 23024 : Zideallog(GEN bid, GEN x)
873 : {
874 23024 : pari_sp av = avma;
875 23024 : GEN y = znconreylog(bid, x), U = znstar_get_U(bid);
876 22996 : return gerepileupto(av, ZM_ZC_mul(U, y));
877 : }
878 : GEN
879 294 : znlog0(GEN h, GEN g, GEN o)
880 : {
881 294 : if (typ(g) == t_VEC)
882 : {
883 : GEN N;
884 56 : if (o) pari_err_TYPE("znlog [with znstar]", o);
885 56 : if (!checkznstar_i(g)) pari_err_TYPE("znlog", g);
886 56 : N = znstar_get_N(g);
887 56 : h = Rg_to_Fp(h,N);
888 49 : return Zideallog(g, h);
889 : }
890 238 : return znlog(h, g, o);
891 : }
892 :
893 : GEN
894 260771 : znconreyexp(GEN bid, GEN x)
895 : {
896 260771 : pari_sp av = avma;
897 : long i, l;
898 : GEN N, pe, gen, cycg, v, vmod;
899 : int e2;
900 260771 : if (!checkznstar_i(bid)) pari_err_TYPE("znconreyexp", bid);
901 260771 : cycg = znstar_get_conreycyc(bid);
902 260771 : switch(typ(x))
903 : {
904 21 : case t_VEC:
905 21 : x = znconreylog(bid, x);
906 21 : break;
907 260750 : case t_COL:
908 260750 : if (RgV_is_ZV(x) && lg(x) == lg(cycg)) break;
909 7 : default: pari_err_TYPE("znconreyexp",x);
910 : }
911 260764 : pe = znstar_get_pe(bid);
912 260764 : gen = znstar_get_conreygen(bid); /* local generators of (Z/p^k)^* */
913 260764 : cycg = znstar_get_conreycyc(bid);
914 260764 : l = lg(x); v = cgetg(l, t_VEC);
915 260764 : N = znstar_get_N(bid);
916 260764 : e2 = !mod8(N); /* 2 generators at p = 2 */
917 805679 : for (i = 1; i < l; i++)
918 : {
919 : GEN q, g, m;
920 544915 : if (i == 1 && e2) { gel(v,1) = NULL; continue; }
921 512631 : q = gel(pe,i);
922 512631 : g = gel(gen,i);
923 512631 : m = modii(gel(x,i), gel(cycg,i));
924 512631 : m = Fp_pow(g, m, q);
925 512631 : if (i == 2 && e2 && signe(gel(x,1))) m = Fp_neg(m, q);
926 512631 : gel(v,i) = mkintmod(m, q);
927 : }
928 260764 : if (e2) v = vecsplice(v, 1);
929 260764 : v = chinese1_coprime_Z(v);
930 260764 : vmod = gel(v,1);
931 260764 : v = gel(v,2);
932 260764 : if (mpodd(v) || mpodd(N)) return gerepilecopy(av, v);
933 : /* handle N = 2 mod 4 */
934 231 : return gerepileuptoint(av, addii(v, vmod));
935 : }
936 :
937 : /* Return Dirichlet character \chi_q(m,.), where bid = znstar(q);
938 : * m is either a t_INT, or a t_COL [Conrey logarithm] */
939 : GEN
940 46767 : znconreychar(GEN bid, GEN m)
941 : {
942 46767 : pari_sp av = avma;
943 : GEN c, d, nchi;
944 :
945 46767 : if (!checkznstar_i(bid)) pari_err_TYPE("znconreychar", bid);
946 46760 : switch(typ(m))
947 : {
948 0 : case t_INTMOD:
949 0 : if (!equalii(gel(m,1), znstar_get_N(bid)))
950 0 : pari_err_TYPE("znconreychar",m);
951 0 : m = gel(m,2); /* fall through */
952 46760 : case t_INT:
953 : case t_COL:
954 46760 : nchi = znconrey_normalized(bid,m); /* images of primroot gens */
955 46753 : break;
956 0 : default:
957 0 : pari_err_TYPE("znconreychar",m);
958 : return NULL;/*LCOV_EXCL_LINE*/
959 : }
960 46753 : d = gel(nchi,1);
961 46753 : c = ZV_ZM_mul(gel(nchi,2), znstar_get_Ui(bid)); /* images of bid gens */
962 46753 : return gerepilecopy(av, char_denormalize(znstar_get_cyc(bid),d,c));
963 : }
964 :
965 : /* chi a t_INT or Conrey log describing a character. Return conductor, as an
966 : * integer if primitive; as a t_VEC [N,factor(N)] if not. Set *pm=m to the
967 : * attached primitive character: chi(g_i) = m[i]/ord(g_i)
968 : * Caller should use znconreylog_normalize(BID, m), once BID(conductor) is
969 : * computed (wasteful to do it here since BID is shared by many characters) */
970 : GEN
971 735427 : znconreyconductor(GEN bid, GEN chi, GEN *pm)
972 : {
973 735427 : pari_sp av = avma;
974 : GEN q, m, F, P, E;
975 : long i, j, l;
976 735427 : int e2, primitive = 1;
977 :
978 735427 : if (!checkznstar_i(bid)) pari_err_TYPE("znconreyconductor", bid);
979 735427 : if (typ(chi) == t_COL)
980 : {
981 704886 : if (!znconrey_check(znstar_get_conreycyc(bid), chi))
982 0 : pari_err_TYPE("znconreyconductor",chi);
983 : }
984 : else
985 30541 : chi = znconreylog(bid, chi);
986 735420 : l = lg(chi);
987 735420 : F = znstar_get_faN(bid);
988 735420 : P = gel(F,1);
989 735420 : E = gel(F,2);
990 735420 : if (l == 1)
991 : {
992 104720 : set_avma(av);
993 104720 : if (pm) *pm = cgetg(1,t_COL);
994 104720 : if (lg(P) == 1) return gen_1;
995 14 : retmkvec2(gen_1, trivial_fact());
996 : }
997 630700 : P = leafcopy(P);
998 630700 : E = leafcopy(E);
999 630700 : m = cgetg(l, t_COL);
1000 630700 : e2 = (E[1] >= 3 && absequaliu(gel(P,1),2));
1001 630700 : i = j = 1;
1002 630700 : if (e2)
1003 : { /* two generators at p=2 */
1004 286153 : GEN a1 = gel(chi,1), a = gel(chi,2);
1005 286153 : i = 3;
1006 286153 : if (!signe(a))
1007 : {
1008 98014 : e2 = primitive = 0;
1009 98014 : if (signe(a1))
1010 : { /* lose one generator */
1011 45507 : E[1] = 2;
1012 45507 : gel(m,1) = a1;
1013 45507 : j = 2;
1014 : }
1015 : /* else lose both */
1016 : }
1017 : else
1018 : {
1019 188139 : long v = Z_pvalrem(a, gen_2, &a);
1020 188139 : if (v) { E[1] -= v; E[2] = E[1]; primitive = 0; }
1021 188139 : gel(m,1) = a1;
1022 188139 : gel(m,2) = a;
1023 188139 : j = 3;
1024 : }
1025 : }
1026 630700 : l = lg(P);
1027 1828568 : for (; i < l; i++)
1028 : {
1029 1197868 : GEN p = gel(P,i), a = gel(chi,i);
1030 : /* image of g_i in Q/Z is a/cycg[i], cycg[i] = order(g_i) */
1031 1197868 : if (!signe(a)) primitive = 0;
1032 : else
1033 : {
1034 937342 : long v = Z_pvalrem(a, p, &a);
1035 937342 : E[j] = E[i]; if (v) { E[j] -= v; primitive = 0; }
1036 937342 : gel(P,j) = gel(P,i);
1037 937342 : gel(m,j) = a; j++;
1038 : }
1039 : }
1040 630700 : setlg(m,j);
1041 630700 : setlg(P,j);
1042 630700 : setlg(E,j);
1043 630700 : if (pm) *pm = m; /* attached primitive character */
1044 630700 : if (primitive)
1045 : {
1046 294721 : q = znstar_get_N(bid);
1047 294721 : if (mod4(q) == 2) primitive = 0;
1048 : }
1049 630700 : if (!primitive)
1050 : {
1051 336756 : if (e2)
1052 : { /* remove duplicate p=2 row from factorization */
1053 114744 : P = vecsplice(P,1);
1054 114744 : E = vecsplice(E,1);
1055 : }
1056 336756 : E = zc_to_ZC(E);
1057 336756 : q = mkvec2(factorback2(P,E), mkmat2(P,E));
1058 : }
1059 630700 : return gc_all(av, pm? 2: 1, &q, pm);
1060 : }
1061 :
1062 : GEN
1063 8414 : zncharinduce(GEN G, GEN chi, GEN N)
1064 : {
1065 8414 : pari_sp av = avma;
1066 : GEN q, faq, P, E, Pq, Eq, CHI;
1067 : long i, j, l;
1068 : int e2;
1069 :
1070 8414 : if (!checkznstar_i(G)) pari_err_TYPE("zncharinduce", G);
1071 8414 : if (!zncharcheck(G, chi)) pari_err_TYPE("zncharinduce", chi);
1072 8414 : q = znstar_get_N(G);
1073 8414 : if (typ(chi) != t_COL) chi = znconreylog(G, chi);
1074 8414 : if (checkznstar_i(N))
1075 : {
1076 8225 : GEN faN = znstar_get_faN(N);
1077 8225 : P = gel(faN,1); l = lg(P);
1078 8225 : E = gel(faN,2);
1079 8225 : N = znstar_get_N(N);
1080 8225 : if (l > 2 && equalii(gel(P,1),gel(P,2)))
1081 : { /* remove duplicate 2 */
1082 2583 : l--;
1083 2583 : P = vecsplice(P,1);
1084 2583 : E = vecsplice(E,1);
1085 : }
1086 : }
1087 : else
1088 : {
1089 189 : GEN faN = check_arith_pos(N, "zncharinduce");
1090 189 : if (!faN) faN = Z_factor(N);
1091 : else
1092 0 : N = (typ(N) == t_VEC)? gel(N,1): factorback(faN);
1093 189 : P = gel(faN,1);
1094 189 : E = gel(faN,2);
1095 : }
1096 8414 : if (!dvdii(N,q)) pari_err_DOMAIN("zncharinduce", "N % q", "!=", gen_0, N);
1097 8407 : if (mod4(N) == 2)
1098 : { /* remove 2 */
1099 77 : if (lg(P) > 1 && absequaliu(gel(P,1), 2))
1100 : {
1101 35 : P = vecsplice(P,1);
1102 35 : E = vecsplice(E,1);
1103 : }
1104 77 : N = shifti(N,-1);
1105 : }
1106 8407 : l = lg(P);
1107 : /* q = N or q = 2N, N odd */
1108 8407 : if (cmpii(N,q) <= 0) return gerepilecopy(av, chi);
1109 : /* N > 1 => l > 1*/
1110 8281 : if (typ(E) != t_VECSMALL) E = ZV_to_zv(E);
1111 8281 : e2 = (E[1] >= 3 && absequaliu(gel(P,1),2)); /* 2 generators at 2 mod N */
1112 8281 : if (ZV_equal0(chi))
1113 : {
1114 5383 : set_avma(av);
1115 5383 : return equali1(N)? cgetg(1, t_COL): zerocol(l+e2 - 1);
1116 : }
1117 :
1118 2898 : faq = znstar_get_faN(G);
1119 2898 : Pq = gel(faq,1);
1120 2898 : Eq = gel(faq,2);
1121 2898 : CHI = cgetg(l+e2, t_COL);
1122 2898 : i = j = 1;
1123 2898 : if (e2)
1124 : {
1125 1183 : i = 2; j = 3;
1126 1183 : if (absequaliu(gel(Pq,1), 2))
1127 : {
1128 1015 : if (Eq[1] >= 3)
1129 : { /* 2 generators at 2 mod q */
1130 560 : gel(CHI,1) = gel(chi,1);
1131 560 : gel(CHI,2) = shifti(gel(chi,2), E[1]-Eq[1]);
1132 : }
1133 455 : else if (Eq[1] == 2)
1134 : { /* 1 generator at 2 mod q */
1135 455 : gel(CHI,1) = gel(chi,1);
1136 455 : gel(CHI,2) = gen_0;
1137 : }
1138 : else
1139 0 : gel(CHI,1) = gel(CHI,2) = gen_0;
1140 : }
1141 : else
1142 168 : gel(CHI,1) = gel(CHI,2) = gen_0;
1143 : }
1144 7315 : for (; i < l; i++,j++)
1145 : {
1146 4417 : GEN p = gel(P,i);
1147 4417 : long k = ZV_search(Pq, p);
1148 4417 : gel(CHI,j) = k? mulii(gel(chi,k), powiu(p, E[i]-Eq[k])): gen_0;
1149 : }
1150 2898 : return gerepilecopy(av, CHI);
1151 : }
1152 :
1153 : /* m a Conrey log [on the canonical primitive roots], cycg the primitive
1154 : * roots orders */
1155 : GEN
1156 2593017 : znconreylog_normalize(GEN G, GEN m)
1157 : {
1158 2593017 : GEN cycg = znstar_get_conreycyc(G);
1159 : long i, l;
1160 2593017 : GEN d, M = cgetg_copy(m, &l);
1161 2593017 : if (typ(cycg) != t_VEC || lg(cycg) != l)
1162 0 : pari_err_TYPE("znconreylog_normalize",mkvec2(m,cycg));
1163 6879509 : for (i = 1; i < l; i++) gel(M,i) = gdiv(gel(m,i), gel(cycg,i));
1164 : /* m[i]: image of primroot generators g_i in Q/Z */
1165 2593017 : M = Q_remove_denom(M, &d);
1166 2593017 : return mkvec2(d? d: gen_1, M);
1167 : }
1168 :
1169 : /* return normalized character on Conrey generators attached to chi: Conrey
1170 : * label (t_INT), char on (SNF) G.gen* (t_VEC), or Conrey log (t_COL) */
1171 : GEN
1172 2586332 : znconrey_normalized(GEN G, GEN chi)
1173 : {
1174 2586332 : switch(typ(chi))
1175 : {
1176 420 : case t_INT: /* Conrey label */
1177 420 : return znconreylog_normalize(G, znconreylog(G, chi));
1178 2575776 : case t_COL: /* Conrey log */
1179 2575776 : if (!RgV_is_ZV(chi)) break;
1180 2575776 : return znconreylog_normalize(G, chi);
1181 10136 : case t_VEC: /* char on G.gen */
1182 10136 : if (!RgV_is_ZV(chi)) break;
1183 10136 : return znconreyfromchar_normalized(G, chi);
1184 : }
1185 0 : pari_err_TYPE("znchareval",chi);
1186 : return NULL;/* LCOV_EXCL_LINE */
1187 : }
1188 :
1189 : /* return 1 iff chi(-1) = -1, and 0 otherwise */
1190 : long
1191 197442 : zncharisodd(GEN G, GEN chi)
1192 : {
1193 : long i, l, s;
1194 : GEN N;
1195 197442 : if (!checkznstar_i(G)) pari_err_TYPE("zncharisodd", G);
1196 197442 : if (!zncharcheck(G, chi)) pari_err_TYPE("zncharisodd", chi);
1197 197442 : if (typ(chi) != t_COL) chi = znconreylog(G, chi);
1198 197442 : N = znstar_get_N(G);
1199 197442 : l = lg(chi);
1200 197442 : s = 0;
1201 197442 : if (!mod8(N))
1202 : {
1203 91371 : s = mpodd(gel(chi,1));
1204 91371 : i = 3;
1205 : }
1206 : else
1207 106071 : i = 1;
1208 537635 : for (; i < l; i++) s += mpodd(gel(chi,i));
1209 197442 : return odd(s);
1210 : }
1211 :
1212 : GEN
1213 847 : znchartokronecker(GEN G, GEN chi, long flag)
1214 : {
1215 847 : pari_sp av = avma;
1216 : long s;
1217 : GEN F, o;
1218 :
1219 847 : if (flag && flag != 1) pari_err_FLAG("znchartokronecker");
1220 847 : s = zncharisodd(G, chi)? -1: 1;
1221 847 : if (typ(chi) != t_COL) chi = znconreylog(G, chi);
1222 847 : o = zncharorder(G, chi);
1223 847 : if (abscmpiu(o,2) > 0) { set_avma(av); return gen_0; }
1224 581 : F = znconreyconductor(G, chi, NULL);
1225 581 : if (typ(F) == t_INT)
1226 : {
1227 469 : if (s < 0) F = negi(F);
1228 469 : return gerepileuptoint(av, F);
1229 : }
1230 112 : F = gel(F,1);
1231 112 : F = (s < 0)? negi(F): icopy(F);
1232 112 : if (!flag)
1233 : {
1234 49 : GEN MF = znstar_get_faN(G), P = gel(MF,1);
1235 49 : long i, l = lg(P);
1236 140 : for (i = 1; i < l; i++)
1237 : {
1238 91 : GEN p = gel(P,i);
1239 91 : if (!dvdii(F,p)) F = mulii(F,sqri(p));
1240 : }
1241 : }
1242 112 : return gerepileuptoint(av, F);
1243 : }
1244 :
1245 : /* (D/.) as a character mod N; assume |D| divides N and D = 0,1 mod 4*/
1246 : GEN
1247 303331 : znchar_quad(GEN G, GEN D)
1248 : {
1249 303331 : GEN cyc = znstar_get_conreycyc(G);
1250 303331 : GEN gen = znstar_get_conreygen(G);
1251 303331 : long i, l = lg(cyc);
1252 303331 : GEN chi = cgetg(l, t_COL);
1253 1351686 : for (i = 1; i < l; i++)
1254 : {
1255 1048355 : long k = kronecker(D, gel(gen,i));
1256 1048355 : gel(chi,i) = (k==1)? gen_0: shifti(gel(cyc,i), -1);
1257 : }
1258 303331 : return chi;
1259 : }
1260 :
1261 : GEN
1262 3297 : znchar(GEN D)
1263 : {
1264 3297 : pari_sp av = avma;
1265 : GEN G, chi;
1266 3297 : switch(typ(D))
1267 : {
1268 2513 : case t_INT:
1269 2513 : if (!signe(D) || Mod4(D) > 1) pari_err_TYPE("znchar", D);
1270 2492 : G = znstar0(D,1);
1271 2492 : chi = mkvec2(G, znchar_quad(G,D));
1272 2492 : break;
1273 714 : case t_INTMOD:
1274 714 : G = znstar0(gel(D,1), 1);
1275 714 : chi = mkvec2(G, znconreylog(G, gel(D,2)));
1276 714 : break;
1277 56 : case t_VEC:
1278 56 : if (checkMF_i(D)) { chi = vecslice(MF_get_CHI(D),1,2); break; }
1279 49 : else if (checkmf_i(D)) { chi = vecslice(mf_get_CHI(D),1,2); break; }
1280 42 : if (lg(D) != 3) pari_err_TYPE("znchar", D);
1281 35 : G = gel(D,1);
1282 35 : if (!checkznstar_i(G)) pari_err_TYPE("znchar", D);
1283 28 : chi = gel(D,2);
1284 28 : if (typ(chi) == t_VEC && lg(chi) == 3 && is_vec_t(typ(gel(chi,2))))
1285 : { /* normalized character */
1286 7 : GEN n = gel(chi,1), chic = gel(chi,2);
1287 7 : GEN cyc = typ(chic)==t_VEC? znstar_get_cyc(G): znstar_get_conreycyc(G);
1288 7 : if (!char_check(cyc, chic)) pari_err_TYPE("znchar",D);
1289 7 : chi = char_denormalize(cyc, n, chic);
1290 : }
1291 28 : if (!zncharcheck(G, chi)) pari_err_TYPE("znchar", D);
1292 21 : chi = mkvec2(G,chi); break;
1293 14 : default:
1294 14 : pari_err_TYPE("znchar", D);
1295 : return NULL; /*LCOV_EXCL_LINE*/
1296 : }
1297 3241 : return gerepilecopy(av, chi);
1298 : }
1299 :
1300 : /* G a znstar, not stack clean */
1301 : GEN
1302 2522226 : znchareval(GEN G, GEN chi, GEN n, GEN z)
1303 : {
1304 2522226 : GEN nchi, N = znstar_get_N(G);
1305 : /* avoid division by 0 */
1306 2522226 : if (typ(n) == t_FRAC && !equali1(gcdii(gel(n,2), N))) return not_coprime(z);
1307 2522219 : n = Rg_to_Fp(n, N);
1308 2522219 : if (!equali1(gcdii(n, N))) return not_coprime(z);
1309 : /* nchi: normalized character on Conrey generators */
1310 2520385 : nchi = znconrey_normalized(G, chi);
1311 2520385 : return chareval_i(nchi, znconreylog(G,n), z);
1312 : }
1313 :
1314 : /* G is a znstar, chi a Dirichlet character */
1315 : GEN
1316 5810 : zncharconj(GEN G, GEN chi)
1317 : {
1318 5810 : switch(typ(chi))
1319 : {
1320 7 : case t_INT: chi = znconreylog(G, chi); /* fall through */
1321 665 : case t_COL: return charconj(znstar_get_conreycyc(G), chi);
1322 5145 : case t_VEC: return charconj(znstar_get_cyc(G), chi);
1323 : }
1324 0 : pari_err_TYPE("zncharconj",chi);
1325 : return NULL; /*LCOV_EXCL_LINE*/
1326 : }
1327 :
1328 : /* G is a znstar, chi a Dirichlet character */
1329 : GEN
1330 388458 : zncharorder(GEN G, GEN chi)
1331 : {
1332 388458 : switch(typ(chi))
1333 : {
1334 21 : case t_INT: chi = znconreylog(G, chi); /*fall through*/
1335 382998 : case t_COL: return charorder(znstar_get_conreycyc(G), chi);
1336 5460 : case t_VEC: return charorder(znstar_get_cyc(G), chi);
1337 0 : default: pari_err_TYPE("zncharorder",chi);
1338 : return NULL; /* LCOV_EXCL_LINE */
1339 : }
1340 : }
1341 :
1342 : /* G is a znstar, chi a Dirichlet character */
1343 : GEN
1344 21 : zncharker(GEN G, GEN chi)
1345 : {
1346 21 : if (typ(chi) != t_VEC) chi = znconreychar(G, chi);
1347 21 : return charker(znstar_get_cyc(G), chi);
1348 : }
1349 :
1350 : /* G is a znstar, 'a' is a Dirichlet character */
1351 : GEN
1352 210 : zncharpow(GEN G, GEN a, GEN n)
1353 : {
1354 210 : switch(typ(a))
1355 : {
1356 21 : case t_INT: return Fp_pow(a, n, znstar_get_N(G));
1357 21 : case t_VEC: return charpow(znstar_get_cyc(G), a, n);
1358 168 : case t_COL: return charpow(znstar_get_conreycyc(G), a, n);
1359 0 : default: pari_err_TYPE("znchapow",a);
1360 : return NULL; /* LCOV_EXCL_LINE */
1361 : }
1362 : }
1363 : /* G is a znstar, 'a' and 'b' are Dirichlet character */
1364 : GEN
1365 302015 : zncharmul(GEN G, GEN a, GEN b)
1366 : {
1367 302015 : long ta = typ(a), tb = typ(b);
1368 302015 : if (ta == tb) switch(ta)
1369 : {
1370 7 : case t_INT: return Fp_mul(a, b, znstar_get_N(G));
1371 7 : case t_VEC: return charmul(znstar_get_cyc(G), a, b);
1372 301980 : case t_COL: return charmul(znstar_get_conreycyc(G), a, b);
1373 0 : default: pari_err_TYPE("zncharmul",a);
1374 : return NULL; /* LCOV_EXCL_LINE */
1375 : }
1376 21 : if (ta != t_COL) a = znconreylog(G, a);
1377 21 : if (tb != t_COL) b = znconreylog(G, b);
1378 21 : return charmul(znstar_get_conreycyc(G), a, b);
1379 : }
1380 :
1381 : /* G is a znstar, 'a' and 'b' are Dirichlet character */
1382 : GEN
1383 5383 : znchardiv(GEN G, GEN a, GEN b)
1384 : {
1385 5383 : long ta = typ(a), tb = typ(b);
1386 5383 : if (ta == tb) switch(ta)
1387 : {
1388 7 : case t_INT: return Fp_div(a, b, znstar_get_N(G));
1389 7 : case t_VEC: return chardiv(znstar_get_cyc(G), a, b);
1390 5348 : case t_COL: return chardiv(znstar_get_conreycyc(G), a, b);
1391 0 : default: pari_err_TYPE("znchardiv",a);
1392 : return NULL; /* LCOV_EXCL_LINE */
1393 : }
1394 21 : if (ta != t_COL) a = znconreylog(G, a);
1395 21 : if (tb != t_COL) b = znconreylog(G, b);
1396 21 : return chardiv(znstar_get_conreycyc(G), a, b);
1397 : }
1398 :
1399 : /* CHI mod N = \prod_p p^e; let CHI = \prod CHI_p, CHI_p mod p^e
1400 : * return \prod_{p | (Q,N)} CHI_p. E.g if Q = p, return chi_p */
1401 : GEN
1402 791 : znchardecompose(GEN G, GEN chi, GEN Q)
1403 : {
1404 : GEN c, P, E, F;
1405 : long l, lP, i;
1406 :
1407 791 : if (!checkznstar_i(G)) pari_err_TYPE("znchardecompose", G);
1408 791 : if (typ(Q) != t_INT) pari_err_TYPE("znchardecompose", Q);
1409 791 : if (typ(chi) == t_COL)
1410 560 : { if (!zncharcheck(G, chi)) pari_err_TYPE("znchardecompose", chi); }
1411 : else
1412 231 : chi = znconreylog(G, chi);
1413 791 : l = lg(chi); if (l == 1) return cgetg(1, t_VEC);
1414 784 : F = znstar_get_faN(G);
1415 784 : c = zerocol(l-1);
1416 784 : P = gel(F,1); /* prime divisors of N */
1417 784 : lP = lg(P);
1418 784 : E = gel(F,2); /* exponents */
1419 2471 : for (i = 1; i < lP; i++)
1420 : {
1421 1687 : GEN p = gel(P,i);
1422 1687 : if (i == 1 && equaliu(p,2) && E[1] >= 3)
1423 : {
1424 567 : if (!mpodd(Q))
1425 : {
1426 203 : gel(c,1) = icopy(gel(chi,1));
1427 203 : gel(c,2) = icopy(gel(chi,2));
1428 : }
1429 567 : i = 2; /* skip P[2] = P[1] = 2 */
1430 : }
1431 : else
1432 1120 : if (dvdii(Q, p)) gel(c,i) = icopy(gel(chi,i));
1433 : }
1434 784 : return c;
1435 : }
1436 :
1437 : GEN
1438 24325 : zncharconductor(GEN G, GEN chi)
1439 : {
1440 24325 : pari_sp av = avma;
1441 24325 : GEN F = znconreyconductor(G, chi, NULL);
1442 24325 : if (typ(F) == t_INT) return F;
1443 12873 : return gerepilecopy(av, gel(F,1));
1444 : }
1445 : GEN
1446 5558 : znchartoprimitive(GEN G, GEN chi)
1447 : {
1448 5558 : pari_sp av = avma;
1449 5558 : GEN chi0, F = znconreyconductor(G, chi, &chi0);
1450 5558 : if (typ(F) == t_INT)
1451 5299 : chi = mkvec2(G,chi);
1452 : else
1453 259 : chi = mkvec2(znstar0(F,1), chi0);
1454 5558 : return gerepilecopy(av, chi);
1455 : }
|