forked from kedlaya/root-unitary
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathprescribed_roots.sage
executable file
·227 lines (195 loc) · 7.3 KB
/
prescribed_roots.sage
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
"""
Finding polynomials with roots in prescribed regions
AUTHOR:
-- Kiran S. Kedlaya (2007-05-28): initial version
-- Kiran S. Kedlaya (2015-08-29): updated version; switch from NTL to FLINT
EXAMPLES:
sage: polRing.<x> = PolynomialRing(Rationals())
sage: P0 = 3*x^21 + 5*x^20 + 6*x^19 + 7*x^18 + 5*x^17 + 4*x^16 + 2*x^15 - x^14 - 3*x^13 - 5*x^12 - 5*x^11 - 5*x^10 - 5*x^9 - 3*x^8 - x^7 + 2*x^6 + 4*x^5 + 5*x^4 + 7*x^3 + 6*x^2 + 5*x + 3
sage: ans, count = roots_on_unit_circle(P0, 3^2, 1)
sage: print "Number of terminal nodes:", count
Number of terminal nodes: 1404
sage: print ans
[3*x^21 + 5*x^20 + 6*x^19 + 7*x^18 + 5*x^17 + 4*x^16 + 2*x^15 - x^14
- 3*x^13 - 5*x^12 - 5*x^11 - 5*x^10 - 5*x^9 - 3*x^8 - x^7 + 2*x^6
+ 4*x^5 + 5*x^4 + 7*x^3 + 6*x^2 + 5*x + 3]
NOTES:
"""
load("prescribed_roots_pyx.spyx")
## Auxiliary function for detecting roots of unity
def no_roots_of_unity(pol):
"""
Return True if the given polynomial is irreducible (or a power of an
irreducible) and not cyclotomic, and False if it is divisible by a
cyclotomic polynomial. Other inputs give undetermined results.
INPUT:
pol -- polynomial with rational coefficients
OUTPUT:
True if pol is irreducible and not cyclotomic.
False if pol has a cyclotomic factor.
EXAMPLES:
sage: pol.<x> = PolynomialRing(Rationals())
sage: no_roots_of_unity(x^5-1)
False
sage: no_roots_of_unity(x^5-2)
True
"""
polRing = pol.parent()
x = polRing.gen()
pol1 = pol
pol2 = pol(-x)
while pol2 == pol1: # Force pol1 not to be even
l = pol1.list()
l1 = [l[i] for i in range(0, len(l), 2)]
pol1 = polRing(l1)
pol2 = pol1(-x)
if not pol1.gcd(pol2).is_constant(): return(False) # zeta_{*}, v_2(*) >= 2
l = (pol1*pol2).list()
l1 = [l[i] for i in range(0, len(l), 2)]
pol3 = polRing(l1)
if not pol1.gcd(pol3).is_constant(): return(False) # zeta_{*}, v_2(*) = 0
pol4 = pol3(-x)
if not pol1.gcd(pol4).is_constant(): return(False) # zeta_{*}, v_2(*) = 1
return(True)
def ej_test(pol): # Elsenhans-Jahnel condition based on Artin-Tate formula
polRing = pol.parent()
x = polRing.gen()
pol1 = pol[0].sign() * pol // (1-x)^pol.ord(1-x)
return(pol1(-1).is_square())
def asymmetrize(P):
"""
Convert a self-inversive polynomial to asymmetric form.
INPUT:
P -- polynomial with rational coefficients, which must be
self-inversive
OUTPUT:
Q, R -- two polynomials such that R is a monic divisor of (x+1)(x-1),
and P(x) = Q(x + 1/x) x^{\deg(Q)} R(x).
EXAMPLES:
sage: pol.<x> = PolynomialRing(Rationals())
sage: asymmetrize(x^5-1)
(x^2 + x - 1, x - 1)
"""
polRing = P.parent()
x = polRing.gen()
if P[0] == 0:
raise ValueError, "Polynomial " + str(P) + " not self-inversive"
d = P.degree()
sg = (P[d]/P[0]).sign()
q = abs(P[d]/P[0])^(2/d)
if not q in Rationals():
raise ValueError, "Polynomial " + str(P) + " not self-inversive"
for i in range(d+1):
if P[i] != sg*P[d-i]/q^(d/2-i):
raise ValueError, "Polynomial " + str(P) + " not self-inversive"
cofactor = polRing(1)
Q = P
if sg == -1:
cofactor *= 1-q*x
Q //= 1-q*x
if Q.degree() %2 == 1:
cofactor *= 1+q*x
Q //= 1+q*x
coeffs = []
m = Q.degree() // 2
for i in reversed(range(m+1)):
coeffs.insert(0, Q.constant_coefficient())
Q = (Q % (1 + q*x^2)^i) // x
return polRing(coeffs), cofactor, q
def symmetrize(Q, R=1, q=1):
"""
Convert a self-inversive polynomial from asymmetric form.
INPUT:
Q, R -- polynomials with rational coefficients
OUTPUT:
P -- the polynomial P(x) = Q(x + 1/x) x^{\deg(Q)} R(x).
EXAMPLES:
sage: pol.<x> = PolynomialRing(Rationals())
sage: symmetrize(x^2+x-1, x-1)
x^5 - 1
"""
polRing = Q.parent()
x = polRing.gen()
return polRing(x^(Q.degree()) * Q(q*x + 1/x)) * R
def roots_on_unit_circle(P0, modulus=1, n=1,
answer_count=None,
verbosity=None, node_count=None, filter=None,
num_threads=None, output=None):
"""
Find polynomials with roots on the unit circle under extra restrictions.
INPUT:
P0 -- polynomial with rational coefficients, which must be
self-inversive
m -- positive integer or list of positive integers
n -- positive integer
answer_count -- positive integer or None
node_count -- positive integer or None; if not None, an exception will
be raised if this many nodes of the tree are encountered.
filter -- function or None; if not None, only polynomials for which
this function evaluates to True will be returned.
OUTPUT:
list -- a list of all polynomials P with roots on the unit circle
such that P is congruent to P0 modulo m and shares its highest
n coefficients with P0. If answer_count is not None, return at
most answer_count polynomials, otherwise return all of them.
integer -- the number of terminal nodes in the tree enumerated
in order to compute the list.
EXAMPLES:
sage: pol.<x> = PolynomialRing(Rationals())
sage: roots_on_unit_circle(x^5 - 1, 2, 1)
([x^5 - 1, x^5 - 2*x^4 + 2*x^3 - 2*x^2 + 2*x - 1], 4)
sage: roots_on_unit_circle(x^5 - 1, 4, 1)
([x^5 - 1], 2)
"""
polRing = P0.parent()
x = polRing.gen()
Q0, cofactor, q = asymmetrize(P0)
num_cofactor = [1, 1+q*x, 1-q*x, 1-q*x^2].index(cofactor)
sign = cmp(Q0.leading_coefficient(), 0)
Q0 *= sign
d = Q0.degree()
lead = Q0.leading_coefficient()
count = 0
try:
modlist = list(modulus)
except TypeError:
modlist = [modulus]
modlist = [0]*n + modlist
if len(modlist) < d+1:
modlist += [modlist[-1]] * (d+1 - len(modlist))
process = process_queue(d, n, lead, sign, q, num_cofactor,
modlist, node_count, verbosity, Q0)
ans = []
anslen = 0
if (num_threads): # parallel version
ans1 = process.parallel_exhaust(num_threads, output)
if output != None:
return process.count
for i in ans1:
Q2 = polRing(i)
if filter == None or filter(Q2):
ans.append(Q2)
anslen += 1
if answer_count != None and anslen >= answer_count:
break
process.clear()
return(ans, process.count)
try:
while True:
t = process.exhaust_next_answer()
if t>0:
Q2 = polRing(process.Qsym_array.tolist())
if filter == None or filter(Q2):
if output != None: output.write(str(list(Q2)))
else: ans.append(Q2)
anslen += 1
if answer_count != None and anslen >= answer_count:
break
elif t==0:
break
else:
raise RuntimeError, "Node count (" + str(self.node_count) + ") exceeded"
finally:
process.clear()
if output != None: return(process.count)
return(ans, process.count)