Hi, I have implemented the Tonelli and Shanks algorithm which has a
complexity of O(ln^4(p)) and appears to be amazing fast. So far its a
quick and dirty implementation directly in my Sage file and without
error handling. I will ask Martin (when meeting him next time in the
office) how to integrate the implentation properly in Sage. Of course,
if somebody else does it, I am even more happy :-) Here is the current
code which returns x such that x^2 == a mod p:

def step3(b,p,r,x):
        # Step 3: Find exponent
        if GF(p)(b) == GF(p)(1):
                return b,r,x,0
        m = 0
        while GF(p)(b**(2**m)) != 1:
                m = m + 1
        if m == r:
                return b,r,0,0
        return b,r,x,m

def s_root(a,p):
        # Step 0: Determine q:
        q = 0
        e = 0
        while q % 2 != 1:
                e = e+1
                q = (p-1) / 2**e
        # Step 1: Find generator
        n = ZZ.random_element()
        while kronecker(n,p) != -1:
                n = ZZ.random_element()
        n = GF(p)(n)
        z = GF(p)(n**q)
        # Step 2: Initialize
        y = z
        r = e
        a = GF(p)(a)
        x = GF(p)(a**((q-1)/2))
        b = GF(p)(a*(x**2))
        x = GF(p)(a*x)
        # Step 3:
        b,r,x,m = step3(b,p,r,x)
        # Step 4: Reduce exponent
        while ZZ(m) != ZZ(0):
                t = GF(p)(y**(2**(r-m-1)))
                y = GF(p)(t**2)
                r = GF(p)(m)
                x = GF(p)(x*t)
                b = GF(p)(b*y)
                b,r,x,m = step3(b,p,r,x)
        return x

a = GF(17)(13)
print s_root(a,17)


Steffen


On 9 Nov., 21:50, "John Cremona" <[EMAIL PROTECTED]> wrote:
> I see.  In my example a was
>
> sage: type(a)
> <type 'sage.rings.integer_mod.IntegerMod_int64'>
>
> John
>
> On 09/11/2007, Robert Bradshaw <[EMAIL PROTECTED]> wrote:
>
>
>
>
>
> > On Nov 9, 2007, at 1:43 PM, John Cremona wrote:
>
> > > [Hello Robert -- are you in Bristol yet?]
>
> > Just got in.
>
> > > I'm puzzled now.  My comment on inefficiency and Steffen's were based
> > > on the code
> > >>                 for i from 0 <= i <= n/2:
> > >>                     if (i*i) % n == self.ivalue:
> > >>                         return self._new_c(i)
>
> > > but now when I do a.sqrt?? I do not see that code.  Steffen, are you
> > > using 2.8.12?
>
> > This depends on the size of the modulus. There are three types--
> > IntegerMod_int32, IntegerMod_int64, and IntegerMod_gmp. The 32-bit
> > one has that code, and only for small p. If your modulus is beg
> > enough, a.sqrt?? won't give you that at all.
>
> > > John
>
> > > On 09/11/2007, Steffen <[EMAIL PROTECTED]> wrote:
>
> > >> Thx, I am wondering why I did not try the command a.sqrt?? on my own.
>
> > >> However, it seems as the implemented algorithm is not the most
> > >> efficient one. My result from a.sqrt?? from the latest release:
>
> > >> def sqrt(self, extend=True, all=False):
> > >>         cdef int_fast32_t i, n = self.__modulus.int32
> > >>         if n > 100:
> > >>             moduli = self._parent.factored_order()
> > >>         # Unless the modulus is tiny, test to see if we're in the
> > >> really
> > >>         # easy case of n prime, n = 3 mod 4.
> > >>         if n > 100 and n % 4 == 3 and len(moduli) == 1 and moduli[0]
> > >> [1] == 1:
> > >>             if jacobi_int(self.ivalue, self.__modulus.int32) == 1:
> > >>                 # it's a non-zero square, sqrt(a) = a^(p+1)/4
> > >>                 i = mod_pow_int(self.ivalue, (self.__modulus.int32
> > >> +1)/
> > >> 4, n)
> > >>                 if i > n/2:
> > >>                     i = n-i
> > >>                 if all:
> > >>                     return [self._new_c(i), self._new_c(n-i)]
> > >>                 else:
> > >>                     return self._new_c(i)
> > >>             elif self.ivalue == 0:
> > >>                 return self
> > >>             elif not extend:
> > >>                 raise ValueError, "self must be a square"
> > >>         # Now we use a heuristic to guess whether or not it will
> > >>         # be faster to just brute-force search for squares in a c
> > >> loop...
> > >>         # TODO: more tuning?
> > >>         elif n <= 100 or n / (1 << len(moduli)) < 5000:
> > >>             if all:
> > >>                 return [self._new_c(i) for i from 0 <= i < n if (i*i)
> > >> % n == self.ivalue]
> > >>             else:
> > >>                 for i from 0 <= i <= n/2:
> > >>                     if (i*i) % n == self.ivalue:
> > >>                         return self._new_c(i)
> > >>                 if not extend:
> > >>                     raise ValueError, "self must be a square"
> > >>         # Either it failed but extend was True, or the generic
> > >> algorithm is better
> > >>         return IntegerMod_abstract.sqrt(self, extend=extend, all=all)
>
> > >> The easier %4 == 3 case seems to be implemented efficiently, but the
> > >> %4 == 1 not. The algo from Tonelli and Shanks might be a good
> > >> solution
> > >> here. Any thoughts on other/better algorithm?
>
> > >> Steffen
>
> > >> On 9 Nov., 18:24, "John Cremona" <[EMAIL PROTECTED]> wrote:
> > >>> Use the ?? operator to see the algorithm:
>
> > >>> sage: a=GF(next_prime(10^6)).random_element()^2;
> > >>> sage: a.sqrt??
>
> > >>> Type:           builtin_function_or_method
> > >>> Base Class:     <type 'builtin_function_or_method'>
> > >>> String Form:    <built-in method sqrt of
> > >>> sage.rings.integer_mod.IntegerMod_int64 object at 0x99055a4>
> > >>> Namespace:      Interactive
> > >>> Source:
> > >>>     def sqrt(self, extend=True, all=False):
> > >>>         r"""
> > >>>         Returns square root or square roots of self modulo n.
>
> > >>>         INPUT:
> > >>>             extend -- bool (default: True); if True, return a square
> > >>>                  root in an extension ring, if necessary. Otherwise,
> > >>>                  raise a ValueError if the square is not in the
> > >>> base ring.
> > >>>             all -- bool (default: False); if True, return *all*
> > >>> square
> > >>>                    roots of self, instead of just one.
>
> > >>>         ALGORITHM: Calculates the square roots mod $p$ for each
> > >>> of the
> > >>>         primes $p$ dividing the order of the ring, then lifts them
> > >>>         p-adically and uses the CRT to find a square root mod $n$.
>
> > >>>         See also \code{square_root_mod_prime_power} and
> > >>>         \code{square_root_mod_prime} (in this module) for more
> > >>>         algorithmic details.
>
> > >>>         EXAMPLES:
> > >>>             sage: mod(-1, 17).sqrt()
> > >>>             4
> > >>>             sage: mod(5, 389).sqrt()
> > >>>             86
> > >>>             sage: mod(7, 18).sqrt()
> > >>>             5
> > >>>             sage: a = mod(14, 5^60).sqrt()
> > >>>             sage: a*a
> > >>>             14
> > >>>             sage: mod(15, 389).sqrt(extend=False)
> > >>>             Traceback (most recent call last):
> > >>>             ...
> > >>>             ValueError: self must be a square
> > >>>             sage: Mod(1/9, next_prime(2^40)).sqrt()^(-2)
> > >>>             9
> > >>>             sage: Mod(1/25, next_prime(2^90)).sqrt()^(-2)
> > >>>             25
>
> > >>>             sage: a = Mod(3,5); a
> > >>>             3
> > >>>             sage: x = Mod(-1, 360)
> > >>>             sage: x.sqrt(extend=False)
> > >>>             Traceback (most recent call last):
> > >>>             ...
> > >>>             ValueError: self must be a square
> > >>>             sage: y = x.sqrt(); y
> > >>>             sqrt359
> > >>>             sage: y.parent()
> > >>>             Univariate Quotient Polynomial Ring in sqrt359 over Ring
> > >>> of integers modulo 360 with modulus x^2 + 1
> > >>>             sage: y^2
> > >>>             359
>
> > >>>         We compute all square roots in several cases:
> > >>>             sage: R = Integers(5*2^3*3^2); R
> > >>>             Ring of integers modulo 360
> > >>>             sage: R(40).sqrt(all=True)
> > >>>             [20, 160, 200, 340]
> > >>>             sage: [x for x in R if x^2 == 40]  # Brute force
> > >>> verification
> > >>>             [20, 160, 200, 340]
> > >>>             sage: R(1).sqrt(all=True)
> > >>>             [1, 19, 71, 89, 91, 109, 161, 179, 181, 199, 251, 269,
> > >>> 271, 289, 341, 359]
> > >>>             sage: R(0).sqrt(all=True)
> > >>>             [0, 60, 120, 180, 240, 300]
>
> > >>>             sage: R = Integers(5*13^3*37); R
> > >>>             Ring of integers modulo 406445
> > >>>             sage: v = R(-1).sqrt(all=True); v
> > >>>             [78853, 111808, 160142, 193097, 213348, 246303,
> > >>> 294637, 327592]
> > >>>             sage: [x^2 for x in v]
> > >>>             [406444, 406444, 406444, 406444, 406444, 406444,
> > >>> 406444, 406444]
> > >>>             sage: v = R(169).sqrt(all=True); min(v), -max(v), len(v)
> > >>>             (13, 13, 104)
> > >>>             sage: all([x^2==169 for x in v])
> > >>>             True
>
> > >>>         Modulo a power of 2:
> > >>>             sage: R = Integers(2^7); R
> > >>>             Ring of integers modulo 128
> > >>>             sage: a = R(17)
> > >>>             sage: a.sqrt()
> > >>>             23
> > >>>             sage: a.sqrt(all=True)
> > >>>             [23, 41, 87, 105]
> > >>>             sage: [x for x in R if x^2==17]
> > >>>             [23, 41, 87, 105]
>
> > >>>         """
> > >>>         if self.is_one():
> > >>>             if all:
> > >>>                 return list(self.parent().square_roots_of_one())
> > >>>             else:
> > >>>                 return self
>
> > >>>         if not self.is_square_c():
> > >>>             if extend:
>
> > >>>  and so on!
>
> > >>> John
>
> > >>> On 09/11/2007, Steffen <[EMAIL PROTECTED]> wrote:
>
> > >>>> Hi, I need to find square roots in GF(prime). I did it like this:
>
> > >>>> y = sqrt(GF(prime)(ySquare))
>
> > >>>> So far I am not quite happy with the calculation periods and I
> > >>>> would
> > >>>> like to know which algorithm is used. In my case is prime % 4 == 1,
> > >>>> which is the hardest case to find the square root mod p. I am sage
> > >>>> newbie and probably its again only a command that I cant find. I
> > >>>> tried
> > >>>> sqrt? and similar things, but only got the information that sqrt
> > >>>> is a
> > >>>> symbolic function. Could somebody tell me which algo is
> > >>>> implemented or
> > >>>> better how to find the implemented algo.
>
> > >>>> Cheers, Steffen
>
> > >>> --
> > >>> John Cremona
>
> > > --
> > > John Cremona
>
> --
> John Cremona


--~--~---------~--~----~------------~-------~--~----~
To post to this group, send email to sage-devel@googlegroups.com
To unsubscribe from this group, send email to [EMAIL PROTECTED]
For more options, visit this group at http://groups.google.com/group/sage-devel
URLs: http://sage.scipy.org/sage/ and http://modular.math.washington.edu/sage/
-~----------~----~----~----~------~----~------~--~---

Reply via email to