-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathdirichlet_conrey.pyx
1604 lines (1368 loc) · 55.8 KB
/
dirichlet_conrey.pyx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
#
# A new implementation of Dirichlet characters based on the numbering scheme
# devised by Brian Conrey.
#
# Maybe the following lines used to be necessary for something with some old
# version of sage/cython? I don't know. I just did a copy-and-paste of some
# example a long time ago. Anyway, they don't seem necessary now, and in fact
# break compilation on new versions of Sage. But maybe with a really old
# version of Sage/cython it would be necessary to uncomment some of the
# following lines and get rid of the from libc.stdlib... line.
#
# include "interrupt.pxi" # ctrl-c interrupt block support
# include "stdsage.pxi" # ctrl-c interrupt block support
# include "cdefs.pxi"
from libc.stdlib cimport malloc, free
from sage.all import (factor,
primitive_root,
IntegerModRing,
euler_phi,
gcd,
lcm,
is_prime,
DirichletGroup,
vector,
Integer,
power_mod,
prod,
crt,
mod,
inverse_mod,
multiplicative_order,
pi,
RR,
CC,
ZZ,
diagonal_matrix,
Mod,
round,
imag)
from sage.modular.dirichlet import DirichletCharacter
from sage.structure.richcmp import richcmp
import cmath
cdef long ipow(long a, long b):
'''
yes, this is stupid. but b will be small.
'''
cdef long x = 1
for i in range(b):
x = x*a
return x
def correct_primitive_root(q):
if q < 4294967296:
if q == 40487:
return 10
else:
return primitive_root(q)
else:
a = primitive_root(q)
R2 = IntegerModRing(q**2)
if R2(a).is_primitive_root():
return a
else:
print(r"""You have found a nice example of a case where the smallest primitive root
mod p is not a primitive root mod p^2. Please email [email protected] and/or
[email protected].""")
a += 1
R1 = IntegerModRing(q)
while not (R1(a).is_primitive_root() and R2(a).is_primitive_root()):
a += 1
return a
# the next example is: 6692367337
cdef long pos_mod(long m, long n):
"""
Return the positive remainder of n modulo m.
Note that the default behaviour of 'n % m' in C is to return a remainder
with the same sign as the n.
The default behaviour in cython is now to return positive numbers.
I guess it was different at some point.
"""
#if n < 0:
# return (n % m) + m
return n % m
cdef complex twopii = 3.1415926535897932384626433833 * 2.0 * 1.0j
cdef float PI = float(pi)
cdef class DirichletGroup_conrey:
#
# Note: perhaps the discrete log tables should be stored
# separately for each prime. This will make computation a
# little slower, but it would be possible to work efficiently
# when the modulus is huge but divisible only by small primes
# to small powers.
#
# Random question for self: Given a discrete log table mod p,
# is it easy to solve the discrete log problem mod p^a? If so,
# it would be possible to remove the last three words from the
# above paragraph.
#
# Pascal > The answer to the above question is YES. Just
# > do a p-adic decomposition (keyword Pohlig-Hellman)
# > of the result.
# > In general, there sould be a mix of precomputed
# > logs for small primes (to any power) and discrete
# > logs computations for large primes.
# > This is a TODO.
cdef long q # the modulus
cdef long q_even # for computation we will strip out the even
cdef long q_odd # factors from the modulus. q == q_even * q_odd
cdef long k # the number of factors of q_odd
cdef long * primes # a list of the odd prime factors of the modulus
cdef long * exponents # a list of the exponents of the odd prime factors in the factorization
cdef long * generators # a primitive root for each odd prime factor
cdef long precomp # shall we precompute logs ?
cdef long * A # exponent vectors:
# for each m coprime to q_odd we store an array
# with the property that
#
# m == g[j]**A[m][j] mod p[j]**e[j]
#
# (where "A[m][k] == A[m * self.k + j]")
#
# where g[j] is a primitive root mod p[j]**e[j],
# and p[j] is the j-th prime factor of q_odd.
#
# This array is the obstacle that will prevent this
# implementation from working reasonably for very
# large modulus. We will need something else which
# does not use any precomputation for that case.
cdef long * B # exponents for q_even:
# for each odd m, 0 <= m < q_even, we will compute B
# so that
#
# m == B[m-1] * 5**B[m] mod q_even,
#
# where B[m-1] = +- 1 and 0 <= B[m] < q_even/4
cdef long * PHI # PHI[j] = phi(q_odd)/phi(p[j]**e[j]). This will make it easier
# to compute the characters.
cdef long phi_q_odd # phi(q_odd)
cdef long phi_q # phi(q)
cdef complex * zeta_powers_odd # an array holding powers of a root of unity.
# this should be the only part of the code that
# needs to change in order to work over a cyclotomic field
cdef complex * zeta_powers_even # for the even part of the character
cdef _standard_dirichlet_group
cdef _invariants
cdef _gens
def __cinit__(self, modulus, basering=None):
if modulus <= 0:
raise ArithmeticError("The modulus of a Dirichlet group must be a positive integer.")
try:
self.q = modulus
except OverflowError:
raise NotImplementedError("Currently this implementation does not allow a modulus that large.")
# self.precomp = (self.q < 100000) # should test to find right value
self.precomp = 1
self.q_even = 1
self.q_odd = self.q
while self.q_odd % 2 == 0:
self.q_odd = self.q_odd // 2
self.q_even = self.q_even * 2
if self.q_odd > 1:
X = factor(self.q_odd)
self.k = len(X)
self.primes = <long*>malloc(self.k * sizeof(long))
self.exponents = <long*>malloc(self.k * sizeof(long))
for n in range(self.k):
self.primes[n] = X[n][0]
self.exponents[n] = X[n][1]
self.generators = <long*>malloc(self.k * sizeof(long))
self.PHI = <long*>malloc(self.k * sizeof(long))
if self.precomp:
self.A = <long*>malloc(self.q_odd * self.k * sizeof(long))
self.zeta_powers_odd = <complex*>malloc(self.q * sizeof(complex))
if self.precomp and self.q_even > 4:
# We are only going to use zeta_powers_even if q_even is large enough.
# When q_even == 2, it would just be {1}, and when q_even == 4, it
# would just be {1,-1}.
#
# This way, it is always the case that, if zeta_powers_even has been
# initialized, it will be of size q_even/4
self.B = <long*>malloc(self.q_even * sizeof(long))
self.zeta_powers_even = <complex*>malloc(self.q_even // 4 * sizeof(complex))
def __init__(self, modulus, basering=None):
# Once we've hit this stage, all of our arrays are allocated,
# and both self.prime and self.exponents contain the right things.
#
# We now set up the rest of the precomputed arrays.
self.phi_q_odd = euler_phi(self.q_odd)
if self.q_even > 1:
self.phi_q = self.phi_q_odd * self.q_even // 2
else:
self.phi_q = self.phi_q_odd
cdef long g
cdef long a
for j in range(self.k):
x = ipow(self.primes[j], self.exponents[j])
#x = self.primes[j]**self.exponents[j]
g = correct_primitive_root(x)
self.generators[j] = g
phi = ipow(self.primes[j], self.exponents[j] - 1) * (self.primes[j] - 1)
#phi = self.primes[j]**(self.exponents[j] - 1) * (self.primes[j] - 1)
self.PHI[j] = self.phi_q_odd // phi
if self.precomp:
a = 1
for l in range(phi):
for m in range(a, self.q_odd, x):
self.A[m * self.k + j] = l
a = (a * g) % x
#
# Store a flag in A for each m that is not coprime to q_odd.
# (This will save on expensive gcd computations later.)
#
if self.precomp:
if self.q_odd > 1:
for m in range(self.q_odd):
if gcd(m, self.q_odd) > 1:
self.A[m * self.k] = -1
#
# Compute a table of powers of the root of unity. This will
# save on expensive calls to exp() later. It does increase
# memory usage by an appreciable amount, though, so we might
# want to add an option to not do this.
#
# We will _need_ to not do this later when allowing very
# large moduli.
#
if self.precomp and self.q_odd > 1:
for n in range(self.phi_q_odd):
self.zeta_powers_odd[n] = cmath.exp(twopii * n / <double>self.phi_q_odd)
cdef long pow_five = 1
if self.precomp and self.q_even > 4:
for n in range(self.q_even // 4):
self.zeta_powers_even[n] = cmath.exp(twopii * n * 4 / <double>self.q_even)
for e in range(self.q_even // 4):
self.B[pow_five] = e
self.B[pow_five - 1] = 1
self.B[self.q_even - pow_five] = e
self.B[self.q_even - pow_five - 1] = -1
pow_five = pow_five * 5
pow_five = pow_five % self.q_even
cpdef long _chi_odd_exponent(self, long m, long n):
r"""
BE CAREFUL CALLING THIS. It implicitly assumes:
- 1 <= m < self.q_odd
- 1 <= n < self.q_odd
- gcd(m, self.q_odd) == 1
- gcd(n, self.q_odd) == 1
- (anything else?)
"""
cdef long x = 0
if self.precomp:
for j in range(self.k):
x += self.A[m * self.k + j]*self.A[n * self.k + j]*self.PHI[j]
x = x % self.phi_q_odd
return x
else:
for j in range(self.k):
pj = self.primes[j]**self.exponents[j]
gj = Mod(self.generators[j], pj)
logm = Mod(m, pj).log(gj)
logn = Mod(n, pj).log(gj)
x += self.PHI[j] * logm * logn % self.phi_q_odd
return x
cpdef long _chi_even_exponent(self, long m, long n):
r"""
BE CAREFUL CALLING THIS. It implicitly assumes that:
- 0 < m < self.q_even
- 0 < n < self.q_even
- self.q_even > 4
- m and n are odd
"""
cdef long exponent = 0
if self.precomp:
exponent = self.B[m]*self.B[n]
if self.B[m-1] == -1 and self.B[n-1] == -1:
exponent += self.q_even // 8
return exponent % (self.q_even // 4)
else:
if self.q_even > 2:
if m % 4 == 3 and n % 4 == 3:
exponent = self.q_even // 8
if self.q_even > 4:
g2 = Mod(5, self.q_even)
if(m % 4 == 3):
m = -m
if(n % 4 == 3):
n = -n
logm = Mod(m, self.q_even).log(g2)
logn = Mod(n, self.q_even).log(g2)
exponent += logn*logn*self.q_even // 4
return exponent % (self.q_even // 4)
else:
return 0
cpdef complex chi(self, long m, long n):
if not self.precomp:
raise NotImplementedError
cdef complex odd_part = 1
cdef complex even_part = 1
if self.q_even > 1:
if m % 2 == 0 or n % 2 == 0:
return 0
elif self.q_even == 2:
even_part = 1
elif self.q_even == 4:
if m % 4 == 3 and n % 4 == 3:
even_part = -1
else:
even_part = 1
else:
even_part = self.zeta_powers_even[self._chi_even_exponent(m % self.q_even, n % self.q_even)]
if self.q_odd > 1:
m = m % self.q_odd
n = n % self.q_odd
if self.A[m * self.k] == -1 or self.A[n * self.k] == -1:
odd_part = 0
else:
odd_part = self.zeta_powers_odd[self._chi_odd_exponent(m, n)]
return even_part * odd_part
def __iter__(self):
cdef long n = 1
if self.precomp:
while n <= self.q:
if self.q_odd == 1 or self.A[(n % self.q_odd) * self.k] != -1:
if self.q_even == 1 or n % 2 == 1:
yield self._getitem_(n)
n = n + 1
else:
while n <= self.q:
if gcd(n, self.q) == 1:
yield self._getitem_(n)
n = n + 1
def primitive_characters(self):
for chi in self:
if chi.is_primitive():
yield chi
def __dealloc__(self):
if self.primes != NULL:
free(self.primes)
if self.exponents != NULL:
free(self.exponents)
if self.generators != NULL:
free(self.generators)
if self.PHI != NULL:
free(self.PHI)
if self.A != NULL:
free(self.A)
if self.zeta_powers_odd != NULL:
free(self.zeta_powers_odd)
if self.zeta_powers_even != NULL:
free(self.zeta_powers_even)
if self.B != NULL:
free(self.B)
def __getitem__(self, n):
return self._getitem_(n)
cdef DirichletCharacter_conrey _getitem_(self, long n):
return DirichletCharacter_conrey(self, n)
def __repr__(self):
return "Group of dirichlet characters with modulus %d" % self.q
def standard_dirichlet_group(self):
"""
Return the "standard" Sage Dirichlet group with the same modulus,
when characters taking values in a cyclotomic field. This is only
computed when it is asked for, but it is cached after being
computed.
Maybe this function needs a better name.
"""
if self._standard_dirichlet_group is None:
self._standard_dirichlet_group = DirichletGroup(self.q)
return self._standard_dirichlet_group
cpdef modulus(self):
return self.q
cpdef order(self):
return self.phi_q
cpdef _gen_invariants(self):
"""
compute elementary divisors structure of the group
first make a list w of cyclic components over primes
and lift local generators
gj == g[j] mod pj
gj == 1 mod qj, qj = q//pj
uj qj + vj pj = 1
-> gj = uj qj g[j] + vj pj = 1 + uj qj (g[j]-1)
"""
w, g = [], []
for j in range(self.k):
p, e = self.primes[j], self.exponents[j]
pj = p**(e - 1)
w.append(pj*(p-1)) # phi(p**e)
pj *= p
qj = self.q // pj
assert qj*pj == self.q
gj = self.generators[j]
uj = inverse_mod(qj, pj)
gj = 1 + qj*uj*(gj-1) % self.q
g.append(gj)
if self.q_even >= 4:
w.append(2)
p2, q2 = self.q_even, self.q_odd
g2 = -1
u2 = inverse_mod(q2, p2)
g2 = 1 + q2*u2*(g2-1) % self.q
g.append(g2)
if self.q_even > 4:
w.append(self.q_even // 4)
g2 = 5
g2 = 1 + q2*u2*(g2-1) % self.q
g.append(g2)
"""
then compute the Smith normal form
and perform base change
"""
k = len(w)
W = diagonal_matrix(ZZ, w)
d, u, v = W.smith_form()
gens, inv = [], []
u = u.inverse()
for j in range(k-1, -1, -1):
if d[j, j] > 1:
inv.append(d[j, j])
s = 1
for l in range(k):
s = s * g[l]**u[l, j] % self.q
gens.append(s)
self._gens = gens
self._invariants = inv
def gens(self):
if self._gens is None:
self._gen_invariants()
return tuple(self._gens)
def invariants(self):
if self._invariants is None:
self._gen_invariants()
return tuple(self._invariants)
cpdef zeta_order(self):
r"""
Every character in this Dirichlet group takes values in some
cyclotomic field `\Q(\zeta_n)`. This function returns the smallest
such `n`.
Warning: We do not actually use this function internally for our table
of roots of unity right now, and so, internally, we may use a larger
zeta order.
EXAMPLES::
sage: from dirichlet_conrey import *
sage: G = DirichletGroup_conrey(37)
sage: G.zeta_order()
36
TESTS::
sage: from dirichlet_conrey import *
sage: L = [ZZ.random_element(1, 100) for n in range(10)]
sage: for n in L:
....: if DirichletGroup_conrey(n).zeta_order() != DirichletGroup(n).zeta_order():
....: print("wrong answer for n =", n)
"""
if self.q > 2:
return Integer(self.invariants()[0])
else:
return Integer(1)
cpdef from_sage_character(self, chi):
r"""
Given a 'sage character' (that is, a Dirichlet character in the
standard sage format), return the character that correponds to it.
EXAMPLES::
sage: from dirichlet_conrey import *
sage: q = ZZ.random_element(2, 500)
sage: q = q - 1 + q%2
sage: G = DirichletGroup_conrey(q)
sage: G[1] == G.from_sage_character(G[1].sage_character())
True
sage: G[2] == G.from_sage_character(G[2].sage_character())
True
sage: x = ZZ.random_element(1,q)
sage: while gcd(x,q) > 1:
....: x = ZZ.random_element(1,q)
sage: if G[x] == G.from_sage_character(G[x].sage_character()):
....: print(True)
....: else:
....: print(q, x)
True
sage: q = 2 * q
sage: G = DirichletGroup_conrey(q)
sage: G[1] == G.from_sage_character(G[1].sage_character())
True
sage: x = ZZ.random_element(1,q)
sage: while gcd(x,q) > 1:
....: x = ZZ.random_element(1,q)
sage: G[x] == G.from_sage_character(G[x].sage_character())
True
sage: q = 2 * q
sage: G = DirichletGroup_conrey(q)
sage: G[1] == G.from_sage_character(G[1].sage_character())
True
sage: x = ZZ.random_element(1,q)
sage: while gcd(x,q) > 1:
....: x = ZZ.random_element(1,q)
sage: G[x] == G.from_sage_character(G[x].sage_character())
True
sage: q = 2 * q
sage: G = DirichletGroup_conrey(q)
sage: G[1] == G.from_sage_character(G[1].sage_character())
True
sage: x = ZZ.random_element(1,q)
sage: while gcd(x,q) > 1:
....: x = ZZ.random_element(1,q)
sage: G[x] == G.from_sage_character(G[x].sage_character())
True
sage: q = 2 * q
sage: G = DirichletGroup_conrey(q)
sage: G[1] == G.from_sage_character(G[1].sage_character())
True
sage: x = ZZ.random_element(1,q)
sage: while gcd(x,q) > 1:
....: x = ZZ.random_element(1,q)
sage: if G[x] == G.from_sage_character(G[x].sage_character()):
....: print(True)
....: else:
....: print(q, x)
True
sage: # Testing that a bug reported by Fredrik Stromberg is fixed...
sage: eps1 = DirichletGroup(5)([-1])
sage: eps2 = DirichletGroup(5,QQ)([-1])
sage: DirichletGroup_conrey(5).from_sage_character(eps1)==DirichletGroup_conrey(5).from_sage_character(eps2)
True
"""
#
# At odd prime powers, it is relatively easy to construct a character
# in our format from the sage character. For even prime powers it is
# a little bit trickier, but not much worse.
#
# So to construct our character, we will decompose the sage character
# into a product of characters to prime power modulus, do the translation
# there, and then glue everything back together.
#
if chi.modulus() != self.q:
raise ArithmeticError("The character we are translating from must have the same modulus as this Dirichlet group.")
# The conversion code below assumes that the character
# has been constructed as a member of the full group of characters
# mod q. Thus, it would break with a chacter constructed
# as DirichletGroup(5, QQ)([-1]), for example, if we didn't first
# "promote" the character to a member of the full group.
chi = self.standard_dirichlet_group()(chi)
decomposition = chi.decomposition()
n_even = 1
# We deal with the even part first. First of all, we strip off the
# even modulus from the decomposition of the character.
if self.q_even > 1:
psi, decomposition = decomposition[0], decomposition[1:]
# Now the even part splits into 3 cases depending on how
# many times 2 divides q. (i.e. 1 time, 2 times, or more than 2
# times.)
# if 2 divides q just once, the relevant character mod 2 is
# always the trivial one
if self.q_even == 2:
n_even = 1
# if 2 divides q twice, then we just need to check if the
# part at two is trivial or not.
elif self.q_even == 4:
if psi.is_trivial():
n_even = 1
else:
n_even = 3
# When 8 divides q, we want to somehow find the exponents
# of the value of the character on 5 and -5, and use these
# to construct the character that we are interested in.
#
# This is going to be ugly...
elif self.q_even > 4:
psi5_exponent = round(imag(CC(psi(5)).log() * (self.q_even//4)/(2*pi)))
n_even = power_mod(5, psi5_exponent, ZZ(self.q_even))
if psi(5) != psi(-5):
n_even = self.q_even - n_even
# I think this might work...
exponent_vector = [ZZ(psi.element()[0]) for psi in decomposition]
odd_prime_powers = [ZZ(self.primes[j])**ZZ(self.exponents[j])
for j in range(self.k)]
generators = [ZZ(self.generators[j]) for j in range(self.k)]
n_odd = crt([power_mod(g, a, pp)
for (g, a, pp) in zip(generators, exponent_vector,
odd_prime_powers)],
odd_prime_powers)
if n_odd == 0:
n_odd = 1
n = crt(n_odd, n_even, ZZ(self.q_odd), ZZ(self.q_even))
return self[n]
cpdef _galois_orbits(self):
'''
Return the Galois orbits of this Dirichlet group as a list of lists
of integers mod q.
'''
N = self.zeta_order()
q = self.modulus()
R = IntegerModRing(q)
S = set(R)
for x in R:
if gcd(x, q) != 1:
S.remove(x)
orbits = []
while S:
n = S.pop()
orbit = set([n])
for a in N.coprime_integers(N):
k = n**a
if k in S:
S.remove(k)
orbit.add(k)
orbits.append(list(orbit))
trace_size = min(10, q)
traces_unique = False
while not traces_unique and trace_size <= q:
traces_lists = []
for orbit in orbits:
order = multiplicative_order(mod(orbit[0], q))
traces = [int(round(sum([self.chi(chi, n) for chi in orbit]).real))*(N/len(orbit)) for n in range(1, trace_size)]
traces_lists.append((order, traces, orbit))
traces_lists.sort()
traces_unique = True
for (order1, traces1, orbits1), (order2, traces2, orbits2) in zip(traces_lists, traces_lists[1:]):
if order1 == order2 and traces1 == traces2:
traces_unique = False
if trace_size == q:
raise NotImplementedError("We did not expect this case to ever occur. ohno.")
trace_size = min(2*trace_size, q)
break
return [o for (order, t, o) in traces_lists]
cpdef galois_orbits(self):
'''
Return the Galois orbits of this Dirichlet group as a list of lists
of Dirichlet characters.
TESTS::
sage: from dirichlet_conrey import *
sage: for q in [14, 17, 109, 64]:
....: G = DirichletGroup_conrey(q)
....: orbits = G.galois_orbits()
....: G2 = DirichletGroup(q)
....: orbits2 = G2.galois_orbits()
....: S1 = sorted([ sorted([chi.number() for chi in o]) for o in orbits])
....: S2 = sorted([ sorted([G.from_sage_character(chi).number() for chi in o]) for o in orbits2])
....: if S1 != S2:
....: print(q)
'''
orbits = self._galois_orbits()
orbits_as_characters = []
for o in orbits:
orbit = []
for n in o:
orbit.append(self[n])
orbits_as_characters.append(orbit)
return orbits_as_characters
def test_conversion(q):
G = DirichletGroup_conrey(q)
for chi in G:
if G.from_sage_character(chi.sage_character()) != chi:
print("Failure for q = {}, chi number {}".format(q, chi.number()))
break
cdef class DirichletCharacter_conrey:
cdef long _n # we will store the number used to create this character,
cdef _number # e.g., -1, but for all computations we use _n, which is number % q.
cdef long q # the modulus of this character
cdef DirichletGroup_conrey _parent
def __init__(self, DirichletGroup_conrey parent, long n):
"""
The nth character for the Dirichlet Group parent.
"""
if gcd(parent.q, n) != 1:
raise ArithmeticError("n must be coprime to the modulus")
self._parent = parent
if parent.q == 1:
self._number = 1
else:
self._number = n % parent.q
self._n = n % parent.q
self.q = parent.q
def __call__(self, long m):
return self.value(m)
def __richcmp__(self, DirichletCharacter_conrey other, op):
r"""
Compare self to other. Return equality if and only if the moduli and
the character number are the same. When different, characters are first
ordered by modulus and then by number.
EXAMPLES::
sage: from dirichlet_conrey import *
sage: G = DirichletGroup_conrey(17)
sage: G2 = DirichletGroup_conrey(20)
sage: chi1 = G[3]
sage: chi2 = G[20]
sage: chi3 = G[21]
sage: chi1 == chi2
True
sage: chi1 < chi3
True
sage: chi2 > chi3
False
sage: chi1 < G2[1]
True
"""
if(self._parent.q != other._parent.q):
return richcmp(self._parent.q, other._parent.q, op)
else:
return richcmp(self._n, other._n, op)
def conductor(self):
r"""
Return the conductor of this character.
TESTS::
sage: from dirichlet_conrey import *
sage: G = DirichletGroup_conrey(17)
sage: G[1].conductor()
1
sage: G[18].conductor()
1
sage: G[2].conductor()
17
This is tested more thoroughly in primitive_character().
"""
odd_parts = (self._primitive_part_at_known_p(k)
for k in range(self._parent.k))
odd_conductor = prod([c for (_, c) in odd_parts])
_, even_conductor = self._primitive_part_at_two()
return odd_conductor * even_conductor
cpdef decomposition(self):
r"""
Return the factorization of this character into characters of prime
power modulus.
EXAMPLES::
sage: from dirichlet_conrey import *
sage: G = DirichletGroup_conrey(8 * 3 * 25)
sage: (chi1, chi2, chi3) = G[7].decomposition()
sage: chi1.modulus()
8
sage: chi2.modulus()
3
sage: chi3.modulus()
25
sage: chi1 * chi2 * chi3 == G[7]
True
TESTS::
sage: chi1.sage_character().extend(600).maximize_base_ring() * chi2.sage_character().extend(600).maximize_base_ring() * chi3.sage_character().extend(600).maximize_base_ring() == G[7].sage_character()
True
"""
cdef long n = self._n
cdef long q
L = []
if self._parent.q_even > 1:
L.append(DirichletGroup_conrey(self._parent.q_even)[n % self._parent.q_even])
for m in range(self._parent.k):
q = self._parent.primes[m]**self._parent.exponents[m]
L.append(DirichletGroup_conrey(q)._getitem_(n % q))
return L
cpdef long exponent(self, long m):
r"""
Return the number a such that chi(m) = e(a/phi(q)).
"""
if gcd(m, self._parent.q) != 1:
return -1
cdef long exponent
cdef long q_even = self._parent.q_even
cdef long q_odd = self._parent.q_odd
if q_odd > 1:
odd_exponent = self._parent._chi_odd_exponent(self._n % q_odd, pos_mod(q_odd,m))
else:
odd_exponent = 0
if q_even > 4:
even_exponent = self._parent._chi_even_exponent(self._n % q_even, pos_mod(q_even,m))
even_exponent *= 2 # the function just above computes the exponent of
# e(1/ (q_even/4) ), but we want the exponent of
# e(1/phi(q_even)) = e(1/(q_even/2))
elif q_even == 4:
if (self._n % q_even) == 3 == pos_mod(q_even,m):
even_exponent = 1
else:
even_exponent = 0
else:
even_exponent = 0
if q_even == 1: # special case because phi(1) != 1/2.
exponent = odd_exponent
else:
exponent = odd_exponent * q_even // 2 + even_exponent * self._parent.phi_q_odd
# we now have the value of chi(m) as e(exponent/phi(q))
# it could be equal to phi(q), though, and in that case we
# want it to be zero...
if exponent == self._parent.phi_q:
exponent -= self._parent.phi_q
return exponent
cpdef extend(self, M):
r"""
Return the extension of this character to a character of modulus M,
where M is a multiple of the modulus of this character.
EXAMPLES::
sage: from dirichlet_conrey import *
sage: G = DirichletGroup_conrey(17)
sage: chi = G[5].extend(17 * 3); chi
Dirichlet character with index 22 modulo 51
sage: chi.sage_character() == G[5].sage_character().extend(17 * 3).maximize_base_ring()
True
sage: chi2 = DirichletGroup_conrey(17 * 3)[7]
sage: chi2.extend(17 * 9).sage_character() == chi2.sage_character().extend(17 * 9).maximize_base_ring()
True
"""
if M % self.modulus() != 0:
raise ArithmeticError("M must be a multiple of the modulus")
# This is a little tricky with the definition of characters that we
# are using, and it is easiest to do this a prime at a time. For an
# odd prime p, to extend a character mod p^a to a character mod p^e
# we just raise the index to the p^(e-a)th power. We treat independent
# primes separately and glue things together using the chinese
# remainder theorem.
cdef DirichletGroup_conrey G = DirichletGroup_conrey(M)
cdef long x, y
y = 0
indices = []
moduli = []
for x in range(self._parent.k):
while G.primes[y] != self._parent.primes[x]:
indices.append(1)
moduli.append(G.primes[y]**G.exponents[y])
y = y + 1
if G.exponents[y] == self._parent.exponents[x]:
q = G.primes[y]**G.exponents[y]
indices.append(self._n % q)
moduli.append(q)
else:
p = self._parent.primes[y]
q1 = p**self._parent.exponents[y]
q2 = p**G.exponents[y]
indices.append(power_mod(self._n % q1, q2//q1, q2))
moduli.append(q2)
# that takes care of the even part of the modulus. still need to deal
# with the odd part.
cdef q_even1 = self._parent.q_even
cdef q_even2 = G.q_even
if q_even1 == q_even2:
indices.append(self._n % q_even1)
else:
raise NotImplementedError
# # this stuff isn't really written yet...
# if q_even1 <= 2:
# indices.append(1)
# if q_even1 == 4:
# if self._n % 4 == 1:
# indices.append(1)
# else:
# indices.append(q_even2/2 - 1)
# if q_even1 == 8:
# pass
moduli.append(q_even2)
return G[crt(indices, moduli)]
def galois_orbit(self):
r"""
Return the galois conjugates of this character.
This is going to be rather inefficient for now.
TESTS::
sage: from dirichlet_conrey import *
sage: G = DirichletGroup_conrey(56)
sage: chi = G[3]
sage: set([psi.sage_character() for psi in chi.galois_orbit()]) == set(chi.sage_character().galois_orbit())
True
"""
L = []
N = self._parent.zeta_order()
q = self._parent.modulus()
for a in N.coprime_integers(N):
L.append(power_mod(self._n, a, q))
return [self._parent[n] for n in set(L)]
cpdef complex gauss_sum(self, long a=1):