
    xhZ:                         d dl mZmZ d dlmZ d dlmZmZmZm	Z
 d dlmZ d dlmZ d dlmZ  G d d      Z G d	 d
      Zd Zd Zd Zd Zd Zd ZddZddZy)    )explog)_randint)	bit_scan1gcdinvertsqrt)_perfect_power)isprime)_sqrt_mod_prime_powerc                       e Zd Zd Zd Zd Zy)SievePolynomialc                 h    || _         || _        |dz  | _        d|z  |z  | _        |dz  |z
  | _        y)a4  This class denotes the sieve polynomial.
        Provide methods to compute `(a*x + b)**2 - N` and
        `a*x + b` when given `x`.

        Parameters
        ==========

        a : parameter of the sieve polynomial
        b : parameter of the sieve polynomial
        N : number to be factored

           N)aba2abb2)selfr   r   Ns       N/var/www/html/eduruby.in/venv/lib/python3.12/site-packages/sympy/ntheory/qs.py__init__zSievePolynomial.__init__
   s;     Q$A#a%Q$(    c                 :    | j                   |z  | j                  z   S N)r   r   r   xs     r   eval_uzSievePolynomial.eval_u   s    vvax$&&  r   c                 Z    | j                   |z  | j                  z   |z  | j                  z   S r   )r   r   r   r   s     r   eval_vzSievePolynomial.eval_v    s'    	DGG#Q&00r   N)__name__
__module____qualname__r   r   r!    r   r   r   r   	   s    &!1r   r   c                       e Zd ZdZd Zy)FactorBaseElemz7This class stores an element of the `factor_base`.
    c                 X    || _         || _        || _        d| _        d| _        d| _        y)z
        Initialization of factor_base_elem.

        Parameters
        ==========

        prime : prime number of the factor_base
        tmem_p : Integer square root of x**2 = n mod prime
        log_p : Compute Natural Logarithm of the prime
        N)primetmem_plog_psoln1soln2b_ainv)r   r)   r*   r+   s       r   r   zFactorBaseElem.__init__'   s0     

 

r   N)r"   r#   r$   __doc__r   r%   r   r   r'   r'   $   s    r   r'   c                 \   ddl m} g }d\  }}|j                  d|       D ]  }t        ||dz
  dz  |      dk(  s|dkD  r|t	        |      dz
  }|dkD  r|t	        |      dz
  }t        ||d      d   }t        t        |      dz        }|j                  t        |||              |||fS )	a  Generate `factor_base` for Quadratic Sieve. The `factor_base`
    consists of all the points whose ``legendre_symbol(n, p) == 1``
    and ``p < num_primes``. Along with the prime `factor_base` also stores
    natural logarithm of prime and the residue n modulo p.
    It also returns the of primes numbers in the `factor_base` which are
    close to 1000 and 5000.

    Parameters
    ==========

    prime_bound : upper prime bound of the factor_base
    n : integer to be factored
    r   )sieve)NN   r   i  i     )
sympy.ntheory.generater1   
primerangepowlenr   roundr   appendr'   )	prime_boundnr1   factor_baseidx_1000idx_5000r)   residuer+   s	            r   _generate_factor_baser@   <   s     -K#Hh!!![1 Fq519"E*a/t| 0{+a/t| 0{+a/+Aua8;G#e*U*+E~eWeDEF X{**r   c              #     K   t        d| z        dz  t        |      z
  }|xs d}|xs t        |      dz
  }	 d\  }	}
}t        d      D ]  }d}g }t        |      |k  rSd}|dk(  s||v r |||      }|dk(  r||v r||   j                  }||z  }|j	                  |       t        |      |k  rSt        t        |      |z
        }|t        |dz
        t        |dz
        k  s|}
|}	|} |	}|
}g }|D ]W  }||   j                  }||   j                  t        ||z  |      z  |z  }d|z  |kD  r||z
  }|j	                  ||z  |z         Y t        |      }t        |||       }|D ]  }||j                  z  dk(  rd|_        t        ||j                        }|D cg c]  }d|z  |z  |j                  z   c}|_        ||j                  |z
  z  |j                  z  |_        ||j                   |z
  z  |j                  z  |_         | t        ddt        |      dz
  z        D ]  }t        |      }d||dz   z	  dz  z  dz
  }|j                  d|z  ||   z  z   }|j                   }t        |||       }|D ]q  }|j                  |j                  ||j                  |   z  z
  |j                  z  |_        |j                  ||j                  |   z  z
  |j                  z  |_        s |  c c}w w)a6   Generate sieve polynomials indefinitely.
    Information such as `soln1` in the `factor_base` associated with
    the polynomial is modified in place.

    Parameters
    ==========

    N : Number to be factored
    M : sieve interval
    factor_base : factor_base primes
    idx_1000 : index of prime number in the factor_base near 1000
    idx_5000 : index of prime number in the factor_base near to 5000
    randint : A callable that takes two integers (a, b) and returns a random integer
              n such that a <= n <= b, similar to `random.randint`.
    r   r   r2   )NNN2   N)r   r7   ranger)   r9   r   absr*   r   sumr   r,   r.   r-   r   r   r   )r   Mr<   r=   r>   randint
approx_valstartendbest_abest_q
best_ratio_r   qrand_ppratioBvalq_lgammar   gfba_invb_elemivneg_pows                                 r   _generate_polynomialr^   Y   sF      QqS!c!f$JME

,s;'!+C
%5"
r 	#AAAa&:%kVq[$UC0F kVq['--Q  a&:% A+,E!S^c*q.6I%I"
	#"  	#Cc"((C$++fQ#Xs.CCcIEw}eHHQVE\"	# FAq!$ 	;B288|q 1bhh'EABCv6%"((2CBIryy1}-9BH		zA~."((:BH	;  q!c!fQh-( 	A!A!A,!+,q0Gai!n$AA1a(A! H88#HHwryy|';;rxxGHHwryy|';;rxxG	H
 G	U H Ds,   A6L9L>3L26L)B;L$L  ELc                    dgd| z  dz   z  }|D ]  }|j                   t        | |j                   z   |j                  z  d| z  |j                        D ]  }||xx   |j                  z  cc<    |j                  dk(  rpt        | |j                  z   |j                  z  d| z  |j                        D ]  }||xx   |j                  z  cc<     |S )a  Sieve Stage of the Quadratic Sieve. For every prime in the factor_base
    that does not divide the coefficient `a` we add log_p over the sieve_array
    such that ``-M <= soln1 + i*p <=  M`` and ``-M <= soln2 + i*p <=  M`` where `i`
    is an integer. When p = 2 then log_p is only added using
    ``-M <= soln1 + i*p <=  M``.

    Parameters
    ==========

    M : sieve interval
    factor_base : factor_base primes
    r   r   r2   )r,   rC   r)   r+   r-   )rF   r<   sieve_arrayfactoridxs        r   _gen_sieve_arrayrc      s     #qsQw-K 	-<<!fll*fll:AaCN 	-C,	-<<1!fll*fll:AaCN 	-C,	-	- r   c                     | dk  r| dz  } d}nd}t        |d      D ]m  \  }}| |j                  z  rd}| |j                  z  } | |j                  z  dk(  r'|dz  }| |j                  z  } | |j                  z  dk(  r'|dz  sf|d|z  z  }o || fS )z Check if `num` is smooth with respect to the given `factor_base`
    and compute its factorization vector.

    Parameters
    ==========

    num : integer whose smootheness is to be checked
    factor_base : factor_base primes
    r   r2   r   )	enumerater)   )numr<   vecr[   rX   es         r   _check_smoothnessrj      s     Qwr	;* 	2>BHHn!FABHHC BHHn! q516MC	 8Or   c                 ~   t        |      t        |       dz  z   |z
  dz  }g }t               }	d|d   j                  z  }
t        ||       D ]  \  }}||k  r|j	                  |      }t        ||      \  }}|dk(  r$|j                  |j                  |      ||f       U||
k  s[t        |      sg| |z  dk(  r|	j                  |       |j                  |      }||v rO|j                  |      \  }}}||z  t        ||       z  | z  }||z  |dz  z  }||z  }|j                  |||f       |||f||<    ||	fS )a)  Trial division stage. Here we trial divide the values generetated
    by sieve_poly in the sieve interval and if it is a smooth number then
    it is stored in `smooth_relations`. Moreover, if we find two partial relations
    with same large prime then they are combined to form a smooth relation.
    First we iterate over sieve array and look for values which are greater
    than accumulated_val, as these values have a high chance of being smooth
    number. Then using these values we find smooth relations.
    In general, let ``t**2 = u*p modN`` and ``r**2 = v*p modN`` be two partial relations
    with the same large prime p. Then they can be combined ``(t*r/p)**2 = u*v modN``
    to form a smooth relation.

    Parameters
    ==========

    N : Number to be factored
    M : sieve interval
    factor_base : factor_base primes
    sieve_array : stores log_p values
    sieve_poly : polynomial from which we find smooth relations
    partial_relations : stores partial relations with one large prime
    ERROR_TERM : error term for accumulated_val
    r   r3      re   r2   r   )r   setr)   rf   r!   rj   r9   r   r   addpopr   )r   rF   r<   r`   
sieve_polypartial_relations
ERROR_TERMaccumulated_valsmooth_relationsproper_factorpartial_relation_upper_boundr   rT   r\   rh   rg   uu_prevv_prevvec_prevs                       r   _trial_division_stager{      su   . 1vAq(:5>OEM#&{2'<'<#< K!, 53 a $Q4S!8##Z%6%6q%91c$BC//GCL3w!|!!#&!!!$A''+<+@+@+E(fHVC^+a/fHQ&x ''As4*+Q!#&'5( ]**r   c              #   4  K   |D cg c]  }|d   	 }}t        |      }dg|z  }t        |      D ]_  }d|z  }t        |      D ]J  }	||	   |z  x}
s|
||	   z  }|||	<   d||	<   t        |	dz   |      D ]  }||   |z  s||xx   |z  cc<     _ a t        |||      D ]o  \  }}}|r
|d   |d   }}t        |||      D ]  \  }}}|s
||z  s||d   z  }||d   z  }! t        |      }dt	        ||z
  |       x}cxk  r| k  sin l| q yc c}w w)a   Finds proper factor of N using fast gaussian reduction for modulo 2 matrix.

    Parameters
    ==========

    N : Number to be factored
    smooth_relations : Smooth relations vectors matrix
    col : Number of columns in the matrix

    Reference
    ==========

    .. [1] A fast algorithm for gaussian elimination over GF(2) and
    its implementation on the GAPP. Cetin K.Koc, Sarath N.Arachchige
    r   Fr2   Tr   N)r7   rC   zipisqrtr   )r   rt   col
s_relationmatrixrowmarkposmr[   rQ   add_coljmatrelrw   r\   m1mat1rel1rW   s                        r   _find_factorr     sx      /??
jm?F?
f+C7S=DSz 
Hs 	A1IM!q!fQi-q	Qq1uc* -Aay1}q	W,	- 	
 4)9: 31vs1v1!$0@A 	NBdcDjT!WT!W	
 !HSQ]"'a'G @s.   DD>D,D?ADD7D	Dc           	      2    t        t        | ||||            S )a  Performs factorization using Self-Initializing Quadratic Sieve.
    In SIQS, let N be a number to be factored, and this N should not be a
    perfect power. If we find two integers such that ``X**2 = Y**2 modN`` and
    ``X != +-Y modN``, then `gcd(X + Y, N)` will reveal a proper factor of N.
    In order to find these integers X and Y we try to find relations of form
    t**2 = u modN where u is a product of small primes. If we have enough of
    these relations then we can form ``(t1*t2...ti)**2 = u1*u2...ui modN`` such that
    the right hand side is a square, thus we found a relation of ``X**2 = Y**2 modN``.

    Here, several optimizations are done like using multiple polynomials for
    sieving, fast changing between polynomials and using partial relations.
    The use of partial relations can speeds up the factoring by 2 times.

    Parameters
    ==========

    N : Number to be Factored
    prime_bound : upper bound for primes in the factor base
    M : Sieve Interval
    ERROR_TERM : Error term for checking smoothness
    seed : seed of random number generator

    Returns
    =======

    set(int) : A set of factors of N without considering multiplicity.
               Returns ``{N}`` if factorization fails.

    Examples
    ========

    >>> from sympy.ntheory import qs
    >>> qs(25645121643901801, 2000, 10000)
    {5394769, 4753701529}
    >>> qs(9804659461513846513, 2000, 10000)
    {4641991, 2112166839943}

    See Also
    ========

    qs_factor

    References
    ==========

    .. [1] https://pdfs.semanticscholar.org/5c52/8a975c1405bd35c65993abf5a4edb667c1db.pdf
    .. [2] https://www.rieselprime.de/ziki/Self-initializing_quadratic_sieve
    )rm   	qs_factor)r   r:   rF   rr   seeds        r   qsr   :  s    b yKJ=>>r   c           
         | dk  rt        d      i }g }i }| dz  dk(  r'd}| dz  } | dz  dk(  r| dz  } |dz  }| dz  dk(  r||d<   t        |       rd|| <   |S t        | d      x}	r|	\  }
}|||
<   |S | }t        |      }t	        ||       \  }}}t        |      dz  dz  }t        | |||||      D ]k  }t        ||      }t        | ||||||      \  }}||z  }|D ]/  }||z  r	d}||z  }||z  dk(  r||z  }|dz  }||z  dk(  r|||<   1 |t        |      k  sk n t        | |t        |      dz         D ]D  }||z  dk(  sd}||z  }||z  dk(  r||z  }|dz  }||z  dk(  r|||<   |dk(  st        |      sD n |dk7  rd||<   |S )a   Performs factorization using Self-Initializing Quadratic Sieve.

    Parameters
    ==========

    N : Number to be Factored
    prime_bound : upper bound for primes in the factor base
    M : Sieve Interval
    ERROR_TERM : Error term for checking smoothness
    seed : seed of random number generator

    Returns
    =======

    dict[int, int] : Factors of N.
                     Returns ``{N: 1}`` if factorization fails.
                     Note that the key is not always a prime number.

    Examples
    ========

    >>> from sympy.ntheory import qs_factor
    >>> qs_factor(1009 * 100003, 2000, 10000)
    {1009: 1, 100003: 1}

    See Also
    ========

    qs

    r   zN should be greater than 1r   r2      i   d   )

ValueErrorr   r
   r   r@   r7   r^   rc   r{   r   )r   r:   rF   rr   r   factorsrt   rq   ri   resultr;   N_copyrG   r=   r>   r<   	thresholdrW   r`   s_relp_frQ   ra   s                          r   r   r   n  sK   @ 	1u566G 	1uz	a!eqj!GAFA !eqj 
qz
1%%v%1
FtnG&;K&K#HhK 3&+I!!QXxQ &q+6*1ak1N_akl
sE! 	AzAqLF1*/1Q 1*/ GAJ	 ,--  q"2C4Dq4HI 	F?aAvF6/Q&6!Q 6/Q&  GFO{gfo	 {Nr   N)   i  )mathr   r   sympy.core.randomr   sympy.external.gmpyr   r   r   r	   r~   sympy.ntheory.factor_r
   sympy.ntheory.primetestr   sympy.ntheory.residue_ntheoryr   r   r'   r@   r^   rc   rj   r{   r   r   r   r%   r   r   <module>r      s\     & E E 0 + ?1 16 0+:HV68/+d*Z1?hUr   