-
Notifications
You must be signed in to change notification settings - Fork 6
/
subgroupcheck.sage
executable file
·75 lines (55 loc) · 1.84 KB
/
subgroupcheck.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
#!/usr/bin/env sage
# Find the smallest element > 1 of { \omega^j : j \in [0, 2^32) }, over the Pasta Fp and Fq.
#
# This is a bit clunky at the moment since the threads work independently on subsets
# of the space, so it requires you to scan the output by eye to get the actual smallest
# element for each field.
import sys
from multiprocessing import Pool, cpu_count
from traceback import print_exc
if sys.version_info[0] == 2:
range = xrange
p = 0x40000000000000000000000000000000224698fc094cf91b992d30ed00000001
q = 0x40000000000000000000000000000000224698fc0994a8dd8c46eb2100000001
def check(ps):
workers = cpu_count()//len(ps)
pool = Pool(processes=workers*len(ps))
try:
for (which, p) in ps.items():
print("Checking %s = %r" % (which, p))
t = p >> 32
omega = GF(p).multiplicative_generator()^t
assert omega.multiplicative_order() == 1<<32
for wid in range(1, workers+1):
pool.apply_async(worker, (which, p, omega, wid, workers))
while True:
sleep(1000)
except (KeyboardInterrupt, SystemExit):
pass
finally:
pool.terminate()
def worker(*args):
try:
real_worker(*args)
except (KeyboardInterrupt, SystemExit):
pass
except:
print_exc()
def real_worker(which, p, omega, wid, workers):
print("Worker %d for %s" % (wid, which))
lowest = 1<<240
dot = 0
x = omega^wid
m = omega^workers
for i in range(wid, 1<<32, workers):
if dot == 65536:
sys.stdout.write('.')
sys.stdout.flush()
dot = 0
dot += 1
if int(x) < lowest:
lowest = int(x)
print("\n%s: i = %r, %r (%d bits)" % (which, i, lowest, len(format(lowest, 'b'))))
x *= m
return lowest
check({"p": p, "q": q})