import random
from Queue import Queue
class Pow:
def __init__(self, base, exp):
self.base = base
self.exp = exp
class Pair:
def __init__(self, x, y):
self.x = x
self.y = y
def gcd(a, b):
while b: a, b = b, a % b
return a
def rabin_miller(p):
if p < 2: return False
if p != 2 and p % 2 == 0: return False
s = p - 1
while s % 2 == 0: s >>= 1
for i in xrange(10):
a = random.randrange(p - 1) + 1
temp = s
mod = pow(a, temp, p)
while temp != p-1 and mod != 1 and mod != p-1:
mod = (mod * mod) % p
temp = temp * 2
if mod != p-1 and temp % 2 == 0:
return False
return True
def brent(n):
if n % 2 == 0: return 2;
x, c, m = random.randrange(0,n), random.randrange(1,n), random.randrange(1,n)
y, r, q = x, 1, 1
g, ys = 0, 0
while True:
x = y
for i in range(r):
y, k = (y*y+c)%n, 0
while True:
ys = y
for i in range(min(m,r-k)):
y, q = (y*y+c)%n, q*abs(x-y)%n
g, k = gcd(q,n), k+m
if k >= r or g > 1: break
r = 2 * r
if g > 1: break
if g == n:
while True :
ys, g = (x*x+c)%n, gcd(abs(x-ys),n)
if g > 1: break
return g
def pollard(n):
if n % 2 == 0:
return 2;
x = random.randrange(2,1000000)
c = random.randrange(2,1000000)
y = x
d = 1
while d == 1:
x = (x * x + c) % n
y = (y * y + c) % n
y = (y * y + c) % n
d = gcd(x - y, n)
if d == n:
break
return d
# Prime Factorization by Pollard Rho alrorithm
# http://c...content-available-to-author-only...e.com/recipes/577037-pollard-rho-prime-factorization/
def factor(n):
q1 = Queue()
q2 = []
q1.put(n)
while not q1.empty():
l = q1.get()
if rabin_miller(l):
q2.append(l)
continue
d = pollard(l)
if d == l: q1.put(l)
else:
q1.put(d)
q1.put(l / d)
return q2
# Square root of n in integer.
def sqrt(n):
if n <= 0: return 0
s = 1
t = n;
while s < t:
s <<= 1
t >>= 1
while True:
t = s
s = (n / s + s) >> 1
if s >= t: break
return t
# Return (a|n).
# (a|n) = 0 if a = 0 (mod p)
# = +1 if x^2 = a (mod p) for some x
# = -1 if there is no such x
def jacobi(a, n):
t = 1
while a != 0:
while a % 2 == 0:
a = a / 2
if n % 8 == 3 or n % 8 == 5: t = -t
(a, n) = (n, a)
if a % 4 == 3 and n % 4 == 3: t = -t
a = a % n
if n == 1: return t
return 0
# Return a^n (mod p)
def pow_mod(a, n, p):
if n == 0: return 1
if n % 2 == 0: return pow_mod(a * a % p, n / 2, p)
return a * pow_mod( a, n - 1, p ) % p
# Find x who satisfies x*x=n (mod prime p) by Shanks-Tonelli algorithm.
# http://w...content-available-to-author-only...x.com/wiki/Shanks-Tonelli_algorithm
def mod_sqrt_by_shanks_tonelli(prime, arg):
y = 2
result = 0
r = 0
q = prime - 1
if jacobi(arg, prime) == -1: return -1
while jacobi(y, prime) != -1: y += 1
result = prime - 1
while q % 2 == 0:
r += 1
q /= 2
result >>= r
y = pow_mod(y, result, prime)
result >>= 1
b = pow_mod(arg, result, prime)
result = arg * b
result %= prime
b *= result
b %= prime
while b != 1:
t = b * b
t %= prime
m = 1
while t != 1:
t *= t
t %= prime
m += 1
t = 0
t |= (1 << (r-m-1))
t %= prime
y %= prime
t = pow_mod(y, t, prime)
y = t * t
r = m
result *= t
result %= prime
b *= y % prime
b %= prime
return result
# Express prime p as a sum of squared two integers, based on Serret's algorithm.
# See p.122 of "The higher arithmetic: an introduction to the theory of numbers".
def square_sum_of_prime_by_serret(p):
if p == 2: return Pair(1, 1)
if p % 4 != 1: return None
q = Pair(p, mod_sqrt_by_shanks_tonelli(p, p - 1))
if 2 * q.y >= p: q.y = p - q.y
v = []
while q.y > 0:
v.append(q.x / q.y)
q.x, q.y = q.y, q.x - (q.x / q.y) * q.y
q.x = v[len(v) / 2]
q.y = 1
for i in range(len(v)/2+1, len(v)):
q.x, q.y = v[i] * q.x + q.y, q.x
if 2 * q.x >= p:
q.x = p - q.x
q.y = sqrt(p - q.x * q.x)
if q.y < q.x:
q.x, q.y = q.y, q.x
return q
def expand(a, b, c, d, u):
q = Pair(0, a * d + b * c)
if a * c >= b * d:
q.x = a * c - b * d
else:
q.x = b * d - a * c
if q.y < q.x:
q.x, q.y = q.y, q.x
for p in u:
if p.x == q.x: return False
u.append(q)
return u
# Find all pairs x>0 and y>0 which satisfy x^2+y^2=n (x<y).
# n is the form of exponent expression.
def square_sum_of_integer(v):
n = 0
u, v1, v2 = [], [], []
for p in v:
if p.base == 2 or p.base % 4 == 1:
for i in range(p.exp): v1.append(p.base)
n += p.exp
else:
if p.exp % 2 != 0: return False
for i in range(p.exp / 2): v2.append(p.base)
if len(v1) == 0:
q = Pair(1, 0)
for p in v:
for i in range(p.exp): q.x *= p.base
u.append(q)
return false
w = []
r = square_sum_of_prime_by_serret(v1[0])
if r == None:
r.x = r.y = v1[0]
u.append(r)
for i in range(1, len(v1)):
r = square_sum_of_prime_by_serret(v1[i])
if r != None:
for q in u:
w = expand(q.x, q.y, r.x, r.y, w)
w = expand(q.x, q.y, r.y, r.x, w)
u, w = w, u
w = []
for i in range(1, len(v2)):
for q in u:
q.x *= v2[i]
q.y *= v2[i]
return u
def solve(n):
r = 0
t = 10**(n/2)
v = []
u = []
m = t*t + 1
p = factor(m)
p.sort()
q = Pow(p[0], 1)
for i in range(1, len(p)):
if p[i] != p[i-1]:
v.append(q)
q = Pow(p[i], 1)
else: q.exp += 1
v.append(q)
u = square_sum_of_integer(v)
for s in u:
if s.x % 2 != 0: s.x, s.y = s.y, s.x
x = t / 2 + s.x / 2
y = (1 + s.y) / 2
if x >= t / 10 and x < t and y < t:
# print " " + str(t * x + y)
r += t * x + y
x = t / 2 - s.x / 2
if x >= t / 10 and x < t and y < t:
# print " " + str(t * x + y)
r += t * x + y
print str(n) + ': ' + str(r)
for n in range(2, 38, 2):
solve(n)
solve(68)
import random
from Queue import Queue

class Pow:
    def __init__(self, base, exp):
        self.base = base
        self.exp = exp

class Pair:
    def __init__(self, x, y):
        self.x = x
        self.y = y

def gcd(a, b):
    while b: a, b = b, a % b
    return a

def rabin_miller(p):
	if p < 2: return False
	if p != 2 and p % 2 == 0: return False
	s = p - 1
	while s % 2 == 0: s >>= 1
	for i in xrange(10):
		a = random.randrange(p - 1) + 1
		temp = s
		mod = pow(a, temp, p)
		while temp != p-1 and mod != 1 and mod != p-1:
			mod = (mod * mod) % p
			temp = temp * 2
		if mod != p-1 and temp % 2 == 0:
			return False
	return True

def brent(n):
    if n % 2 == 0: return 2;
    x, c, m = random.randrange(0,n), random.randrange(1,n), random.randrange(1,n)
    y, r, q = x, 1, 1
    g, ys = 0, 0
    while True:
        x = y
        for i in range(r):
            y, k = (y*y+c)%n, 0
        while True:
            ys = y
            for i in range(min(m,r-k)):
                y, q = (y*y+c)%n, q*abs(x-y)%n
            g, k = gcd(q,n), k+m
            if k >= r or g > 1: break
        r = 2 * r
        if g > 1: break
    if g == n:
        while True :
            ys, g = (x*x+c)%n, gcd(abs(x-ys),n)
            if g > 1: break
    return g

def pollard(n):
    if n % 2 == 0:
        return 2;
    x = random.randrange(2,1000000)
    c = random.randrange(2,1000000)
    y = x
    d = 1
    while d == 1:
        x = (x * x + c) % n
        y = (y * y + c) % n
        y = (y * y + c) % n
        d = gcd(x - y, n)
        if d == n:
            break
    return d

#    Prime Factorization by Pollard Rho alrorithm
#    http://c...content-available-to-author-only...e.com/recipes/577037-pollard-rho-prime-factorization/
def factor(n):
    q1 = Queue()
    q2 = []
    q1.put(n)
    while not q1.empty():
        l = q1.get()
        if rabin_miller(l):
            q2.append(l)
            continue
        d = pollard(l)
        if d == l: q1.put(l)
        else:
            q1.put(d)
            q1.put(l / d)
    return q2

#   Square root of n in integer.
def sqrt(n):
    if n <= 0: return 0
    s = 1
    t = n;
    while s < t:
        s <<= 1
        t >>= 1
    while True:
        t = s
        s = (n / s + s) >> 1
        if s >= t: break
    return t

#    Return (a|n).
#    (a|n) = 0  if a = 0 (mod p)
#          = +1 if x^2 = a (mod p) for some x
#          = -1 if there is no such x
def jacobi(a, n):
    t = 1
    while a != 0:
        while a % 2 == 0:
            a = a / 2
            if n % 8 == 3 or n % 8 == 5: t = -t
        (a, n) = (n, a)
        if a % 4 == 3 and n % 4 == 3: t = -t
        a = a % n
    if n == 1: return t
    return 0

#    Return a^n (mod p)
def pow_mod(a, n, p): 
    if n == 0: return 1
    if n % 2 == 0: return pow_mod(a * a % p, n / 2, p)
    return a * pow_mod( a, n - 1, p ) % p

#    Find x who satisfies x*x=n (mod prime p) by Shanks-Tonelli algorithm.
#    http://w...content-available-to-author-only...x.com/wiki/Shanks-Tonelli_algorithm
def mod_sqrt_by_shanks_tonelli(prime, arg):
    y = 2
    result = 0
    r = 0
    q = prime - 1
    if jacobi(arg, prime) == -1: return -1
    while jacobi(y, prime) != -1: y += 1
    result = prime - 1
    while q % 2 == 0:
        r += 1
        q /= 2
    result >>= r
    y = pow_mod(y, result, prime)
    result >>= 1
    b = pow_mod(arg, result, prime)
    result = arg * b
    result %= prime
    b *= result
    b %= prime
    while b != 1:   
        t = b * b
        t %= prime
        m = 1
        while t != 1:
            t *= t
            t %= prime
            m += 1
        t = 0
        t |= (1 << (r-m-1))
        t %= prime
        y %= prime
        t = pow_mod(y, t, prime)
        y = t * t
        r = m
        result *= t
        result %= prime
        b *= y % prime
        b %= prime
    return result

#    Express prime p as a sum of squared two integers, based on Serret's algorithm.
#    See p.122 of "The higher arithmetic: an introduction to the theory of numbers".
def square_sum_of_prime_by_serret(p):
    if p == 2: return Pair(1, 1)
    if p % 4 != 1: return None
    q = Pair(p, mod_sqrt_by_shanks_tonelli(p, p - 1))
    if 2 * q.y >= p: q.y = p - q.y
    v = []
    while q.y > 0: 
        v.append(q.x / q.y)
        q.x, q.y = q.y, q.x - (q.x / q.y) * q.y
    q.x = v[len(v) / 2]
    q.y = 1
    for i in range(len(v)/2+1, len(v)):
        q.x, q.y = v[i] * q.x + q.y, q.x
    if 2 * q.x >= p:
        q.x = p - q.x
    q.y = sqrt(p - q.x * q.x)
    if q.y < q.x:
        q.x, q.y = q.y, q.x
    return q

def expand(a, b, c, d, u):
    q = Pair(0, a * d + b * c)
    if a * c >= b * d:
        q.x = a * c - b * d
    else:
        q.x = b * d - a * c
    if q.y < q.x:
        q.x, q.y = q.y, q.x
    for p in u:
        if p.x == q.x: return False
    u.append(q)
    return u

#    Find all pairs x>0 and y>0 which satisfy x^2+y^2=n (x<y).
#    n is the form of exponent expression.
def square_sum_of_integer(v):
    n = 0
    u, v1, v2 = [], [], []
    for p in v:
        if p.base == 2 or p.base % 4 == 1:
            for i in range(p.exp): v1.append(p.base)
            n += p.exp
        else:
            if p.exp % 2 != 0: return False
            for i in range(p.exp / 2): v2.append(p.base)
    if len(v1) == 0:
        q = Pair(1, 0)
        for p in v:
            for i in range(p.exp): q.x *= p.base
        u.append(q)
        return false
    w = []
    r = square_sum_of_prime_by_serret(v1[0])
    if r == None:
        r.x = r.y = v1[0]
    u.append(r)
    for i in range(1, len(v1)):
        r = square_sum_of_prime_by_serret(v1[i])
        if r != None:
            for q in u:
                w = expand(q.x, q.y, r.x, r.y, w)
                w = expand(q.x, q.y, r.y, r.x, w)
        u, w = w, u
        w = []
    for i in range(1, len(v2)):
        for q in u:
            q.x *= v2[i]
            q.y *= v2[i]
    return u

def solve(n):
    r = 0
    t = 10**(n/2)
    v = []
    u = []
    m = t*t + 1

    p = factor(m)
    p.sort()
    q = Pow(p[0], 1)
    for i in range(1, len(p)):
        if p[i] != p[i-1]:
            v.append(q)
            q = Pow(p[i], 1)
        else: q.exp += 1
    v.append(q)

    u = square_sum_of_integer(v)

    for s in u:
        if s.x % 2 != 0: s.x, s.y = s.y, s.x
        x = t / 2 + s.x / 2
        y = (1 + s.y) / 2
        if x >= t / 10 and x < t and y < t:
#            print "  " + str(t * x + y)
            r += t * x + y
        x = t / 2 - s.x / 2
        if x >= t / 10 and x < t and y < t:
#            print "  " + str(t * x + y)
            r += t * x + y
    print str(n) + ': ' + str(r)

for n in range(2, 38, 2):
	solve(n)
solve(68)