/* package whatever; // don't place package name! */
import java.util.* ;
import java.lang.* ;
import java.io.* ;
/**
* Created by Игорь on 25.10.2015.
*/
import java.util.Random ;
public class Main {
public static void main
( String [ ] args
) { //System.out.println(isPrimeAccordingToMillerRabin(19, 5));
generatePossiblePrimeWithLength( 20 ) ;
}
public static int gcd( int a, int b) {
if ( b == 0 )
return gcd( b, a % b) ;
}
/* Символ якоби
1 (проверка взаимной простоты). Если НОД (a, b)≠1, выход из алгоритма с ответом 0.
2 (инициализация). r:=1
3 (переход к положительным числам).
Если a<0 то
a:=-a
Если b mod 4 = 3 то r:=-r
Конец если
4 (избавление от чётности). t:=0
Цикл ПОКА a – чётное
t:=t+1
a:=a/2
Конец цикла
Если t – нечётное, то
Если b mod 8 = 3 или 5, то r:=-r.
Конец если
5 (квадратичный закон взаимности). Если a mod 4 = b mod 4 = 3, то r:=-r.
c:=a; a:=b mod c; b:=c.
6 (выход из алгоритма?). Если a≠0, то идти на шаг 4, иначе выйти из алгоритма с ответом r.
*/
public static int jacobySign( int a, int b) // a - целое, b - натуральное > 1, нечетное
{
if ( gcd ( a, b) != 1 ) {
return 0 ;
}
int r = 1 ;
if ( a < 0 ) {
a = - a;
if ( b % 4 == 3 ) {
r = - r;
}
}
while ( a != 0 )
{
int t = 0 ;
while ( a % 2 == 0 ) {
t++;
a /= 2 ;
}
if ( t % 2 != 0 )
{
int mod = b % 8 ;
if ( mod == 3 || mod == 5 )
{
r = - r;
}
}
int modA = a % 4 ;
if ( ( modA == b % 4 ) && modA == 3 ) {
r = - r;
}
int c = a;
a = b % c;
b = c;
}
return r;
}
public static int randomIntInRange( int min, int max) {
int randomNum = rand.nextInt ( ( max - min) + 1 ) + min;
return randomNum;
}
public static int pow( int foundation, int degree, int modeFoundation) {
int answer = 1 ;
for ( int i = 0 ; i < degree; i++ ) {
answer *= foundation;
answer %= modeFoundation;
}
return answer;
}
/*
Алгоритм Соловея — Штрассена [6] параметризуется количеством раундов k.
В каждом раунде случайным образом выбирается число a < n. Если НОД(a,n) > 1, то выносится решение, что n составное.
Иначе проверяется справедливость сравнения \textstyle a^{(n-1)/2} \equiv \left( {a\over n} \right) \pmod{n}.
Если оно не выполняется, то выносится решение, что n — составное. Если это сравнение выполняется, то a является свидетелем простоты числа n.
Далее выбирается другое случайное a и процедура повторяется.
После нахождения k свидетелей простоты в k раундах выносится заключение, что n является простым числом с вероятностью 1 - 2^{-k} .
Вход: n > 2, тестируемое нечётное натуральное число;
k, параметр, определяющий точность теста.
Выход: составное, означает, что n точно составное;
вероятно простое, означает, что n вероятно является простым.
for i = 1, 2, ..., k:
a = случайное целое от 2 до n - 1, включительно;
если НОД(a, n) > 1, тогда:
вывести, что n — составное, и остановиться.
если a^{(n-1)/2} \not\equiv \left( {a\over n} \right) \pmod{n}, тогда:
вывести, что n — составное, и остановиться.
вывести, что n — простое с вероятностью \textstyle 1 - 2^{-k} , и остановиться.
*/
public static boolean isPrimeAccordingToSoloveiShtrassen( int number, int accuracy) { //number > 2, нечётное натуральное число
for ( int i = 0 ; i < accuracy; i++ ) {
int randomInt = randomIntInRange( 2 , number - 1 ) ;
if ( gcd( randomInt, number) > 1 ) {
return false ;
}
int checkNumberW = pow( randomInt, ( number - 1 ) / 2 , number) ;
int jacobySignum = ( jacobySign( randomInt, number) + number) % number;
if ( checkNumberW != jacobySignum) {
return false ;
}
}
return true ;
}
/*
Алгоритм Миллера — Рабина параметризуется количеством раундов r. Рекомендуется брать r порядка величины \log_2(m), где m — проверяемое число.
Для данного m находятся такие целое число s и целое нечётное число t, что m-1 = 2^s t.
Выбирается случайное число a, 1 < a < m. Если a не является свидетелем простоты числа m, то выдаётся ответ «m — составное», и алгоритм завершается. Иначе, выбирается новое случайное число a и процедура проверки повторяется.
После нахождения r свидетелей простоты, выдаётся ответ «m — вероятно простое», и алгоритм завершается.
Ввод: m > 2, нечётное натуральное число, которое необходимо проверить на простоту;
r — количество раундов.
Вывод: составное, означает, что m является составным числом;
вероятно простое, означает, что m с высокой вероятностью является простым числом.
Представить m − 1 в виде 2s·t, где t нечётно, можно сделать последовательным делением m - 1 на 2.
цикл А: повторить r раз:
Выбрать случайное целое число a в отрезке [2, m − 2]
x ← at mod m, вычисляется с помощью алгоритма возведения в степень по модулю
если x = 1 или x = m − 1, то перейти на следующую итерацию цикла А
цикл B: повторить s − 1 раз
x ← x2 mod m
если x = 1, то вернуть составное
если x = m − 1, то перейти на следующую итерацию цикла A
вернуть составное
вернуть вероятно простое
*/
// Рекомендуется брать r порядка величины log2(m), где m — проверяемое число. number > 2, нечётное натуральное число
public static boolean isPrimeAccordingToMillerRabin( int number, int roundCount) {
int numberMinus1 = number - 1 ;
int t = numberMinus1;
int s = 0 ;
while ( t % 2 == 0 )
{
t /= 2 ;
s++;
}
for ( int i = 0 ; i < roundCount; i++ ) {
int randomInt = randomIntInRange( 2 , number - 2 ) ;
int x = pow( randomInt, t, number) ;
if ( x == 1 || x == numberMinus1) continue ;
boolean flagToStop = true ;
for ( int j = 0 ; j < s - 1 ; j++ ) {
x = pow( x, 2 , number) ;
if ( x == 1 ) {
return false ;
}
if ( x == numberMinus1) {
flagToStop = false ;
break ;
}
}
if ( flagToStop)
{
return false ;
}
}
return true ;
}
public static boolean isPrime( int n) {
boolean r = true ;
for ( int i
= 2 ; i
<= ( int ) Math .
sqrt ( n
) ; i
++ ) { if ( n % i == 0 ) {
r = false ;
break ;
}
}
return r;
}
public static void generatePossiblePrimeWithLength( int length) {
int primeNumber = 2 , roundCount = 5 ;
boolean f = false ;
while ( ! f) {
primeNumber
= 2 + rnd.
nextInt ( ( int ) Math .
pow ( 2 , length
) - 2 ) ;
for ( int i = 2 ; i < 7 && isPrime( i) ; i++ ) {
if ( primeNumber % i == 0 ) {
f = false ;
break ;
}
else {
f = true ;
}
}
if ( ! isPrimeAccordingToMillerRabin( primeNumber, roundCount) ) {
f = false ;
} ;
}
System .
out .
println ( "Number: " + primeNumber
) ; }
/*
Простое число p называется сильно простым, если существуют целые числа r,s,t, удовлетвоярющие следующим условиям:
1. p+1 имеет большой простой делитель s;
2. p-1 имеет большой простой делитель r;
3. r-1 имеет большой простой делитель t.
В этом определении величина делителей зависит от вида атаки, от которой необходимо защититься.
Алгоритм Гордона (Gordon’s algorithm) для генерации сильно простого числа
Выход: Сильно простое число p.
1. Сгенерировать два больших простых числа s и t одной длины;
2. Выбрать число Gus39.gif. Найти первое простое число в последовательности 2it+1, где i= Gus39.gif, i=i+1. Обозначить это число r=2it+1;
3. Вычислить Gus40.gif;
4. Выбрать целое число Gus42.gif. Найти первое простое число в последовательности Gus41.gif, где j = Gus42.gif, j=j+1. Обозначить это число p=Gus41.gif;
5. Вернуть (p).
*/
public static void generateStrongPrime( ) {
int k = 9 ;
int i0 = 1 ;
int j0 = 1 ;
int s = 0 ;
int t = 0 ;
boolean isp1 = false ;
boolean isp2 = false ;
while ( ! isp1) {
s
= ( int ) Math .
pow ( 2 , k
- 1 ) + rnd.
nextInt ( ( int ) Math .
pow ( 2 , k
) - ( int ) Math .
pow ( 2 , k
- 1 ) ) ; isp1 = isPrime( s) ;
}
while ( ! isp2) {
t
= ( int ) Math .
pow ( 2 , k
- 1 ) + rnd.
nextInt ( ( int ) Math .
pow ( 2 , k
) - ( int ) Math .
pow ( 2 , k
- 1 ) ) ; isp2 = isPrime( t) ;
}
int i = i0;
while ( ! isPrime( 2 * i * t + 1 ) ) {
i++;
}
int r = 2 * i * t + 1 ;
int p0 = 2 * pow( s, r - 2 , r) * s - 1 ;
int j = j0;
while ( ! isPrime( p0 + 2 * j * r * s) ) {
j++;
}
int p = p0 + 2 * j * r * s;
System .
out .
println ( "Number :" + p
) ; }
}
/* package whatever; // don't place package name! */

import java.util.*;
import java.lang.*;
import java.io.*;

/**
 * Created by Игорь on 25.10.2015.
 */
import  java.util.Random;
public class Main {
    public static void main(String[] args){
        //System.out.println(isPrimeAccordingToMillerRabin(19, 5));
        generatePossiblePrimeWithLength(20);
    }


    public static int gcd(int a, int b) {
        if (b == 0)
            return Math.abs(a);
        return gcd(b, a % b);
    }

    /* Символ якоби
    1 (проверка взаимной простоты). Если НОД (a, b)≠1, выход из алгоритма с ответом 0.

2 (инициализация). r:=1

3 (переход к положительным числам).
 Если a<0 то
  a:=-a
  Если b mod 4 = 3 то r:=-r
 Конец если

4 (избавление от чётности). t:=0
 Цикл ПОКА a – чётное
  t:=t+1
  a:=a/2
 Конец цикла
 Если t – нечётное, то
  Если b mod 8 = 3 или 5, то r:=-r.
 Конец если

5 (квадратичный закон взаимности). Если a mod 4 = b mod 4 = 3, то r:=-r.
  c:=a; a:=b mod c; b:=c.

6 (выход из алгоритма?). Если a≠0, то идти на шаг 4, иначе выйти из алгоритма с ответом r.
     */
    public static int jacobySign(int a, int b) // a - целое, b - натуральное > 1, нечетное
    {
        if (gcd (a, b) != 1){
            return 0;
        }

        int r = 1;

        if (a < 0){
            a = -a;

            if (b % 4 == 3){
                r = -r;
            }
        }

        while (a != 0)
        {
            int t = 0;

            while (a % 2 == 0){
                t++;
                a /= 2;
            }

            if (t % 2 != 0)
            {
                int mod = b % 8;

                if (mod == 3 || mod == 5)
                {
                    r = -r;
                }
            }

            int modA = a % 4;

            if ((modA == b % 4) && modA  == 3) {
                r = -r;
            }
            int c = a;
            a = b % c;
            b = c;
        }

        return r;
    }


    public static int randomIntInRange(int min, int max){
        Random rand = new Random();
        int randomNum = rand.nextInt((max - min) + 1) + min;

        return randomNum;
    }


    public static int pow(int foundation, int degree, int modeFoundation){
        int answer = 1;

        for (int i = 0; i < degree; i++){
            answer *= foundation;
            answer %= modeFoundation;
        }

        return  answer;
    }


    /*
    Алгоритм Соловея — Штрассена [6] параметризуется количеством раундов k.
    В каждом раунде случайным образом выбирается число a < n. Если НОД(a,n) > 1, то выносится решение, что n составное.
    Иначе проверяется справедливость сравнения \textstyle a^{(n-1)/2} \equiv \left( {a\over n} \right) \pmod{n}.
    Если оно не выполняется, то выносится решение, что n — составное. Если это сравнение выполняется, то a является свидетелем простоты числа n.
    Далее выбирается другое случайное a и процедура повторяется.
    После нахождения k свидетелей простоты в k раундах выносится заключение, что n является простым числом с вероятностью  1 - 2^{-k}  .

    Вход: n > 2, тестируемое нечётное натуральное число;
      k, параметр, определяющий точность теста.
Выход: составное, означает, что n точно составное;
       вероятно простое, означает, что n вероятно является простым.

for i = 1, 2, ..., k:
   a = случайное целое от 2 до n - 1, включительно;
   если НОД(a, n) > 1, тогда:
       вывести, что n — составное, и остановиться.
   если a^{(n-1)/2} \not\equiv \left( {a\over n} \right) \pmod{n}, тогда:
       вывести, что n — составное, и остановиться.

вывести, что n —  простое  с вероятностью \textstyle 1 - 2^{-k} , и остановиться.
     */

    public static boolean isPrimeAccordingToSoloveiShtrassen(int number, int accuracy){ //number > 2, нечётное натуральное число
        for (int i = 0; i < accuracy; i++){
            int randomInt = randomIntInRange(2, number - 1);

            if (gcd(randomInt, number) > 1){
                return false;
            }

            int checkNumberW = pow(randomInt, (number - 1) / 2, number);
            int jacobySignum = (jacobySign(randomInt, number) + number) % number;

            if (checkNumberW != jacobySignum){
                return false;
            }
        }

        return true;
    }


    /*
    Алгоритм Миллера — Рабина параметризуется количеством раундов r. Рекомендуется брать r порядка величины \log_2(m), где m — проверяемое число.

Для данного m находятся такие целое число s и целое нечётное число t, что m-1 = 2^s t.
Выбирается случайное число a, 1 < a < m. Если a не является свидетелем простоты числа m, то выдаётся ответ «m — составное», и алгоритм завершается. Иначе, выбирается новое случайное число a и процедура проверки повторяется.
После нахождения r свидетелей простоты, выдаётся ответ «m — вероятно простое», и алгоритм завершается.

     Ввод: m > 2, нечётное натуральное число, которое необходимо проверить на простоту;
       r — количество раундов.
Вывод: составное, означает, что m является составным числом;
       вероятно простое, означает, что m с высокой вероятностью является простым числом.
Представить m − 1 в виде 2s·t, где t нечётно, можно сделать последовательным делением m - 1 на 2.
цикл А: повторить r раз:
   Выбрать случайное целое число a в отрезке [2, m − 2]
   x ← at mod m, вычисляется с помощью алгоритма возведения в степень по модулю
   если x = 1 или x = m − 1, то перейти на следующую итерацию цикла А
   цикл B: повторить s − 1 раз
      x ← x2 mod m
      если x = 1, то вернуть составное
      если x = m − 1, то перейти на следующую итерацию цикла A
   вернуть составное
вернуть вероятно простое
     */

    // Рекомендуется брать r порядка величины log2(m), где m — проверяемое число. number > 2, нечётное натуральное число
    public static boolean isPrimeAccordingToMillerRabin(int number, int roundCount){
        int numberMinus1 = number - 1;
        int t = numberMinus1;
        int s = 0;

        while (t % 2 == 0)
        {
            t /= 2;
            s++;
        }

        for (int i = 0; i < roundCount; i++){
            int randomInt = randomIntInRange(2, number - 2);
            int x = pow(randomInt, t, number);

            if (x == 1 || x == numberMinus1) continue;

            boolean flagToStop = true;

            for (int j = 0; j < s - 1; j++){
                x = pow(x, 2, number);

                if (x == 1){
                    return false;
                }

                if (x == numberMinus1){
                    flagToStop = false;
                    break;
                }
            }

            if (flagToStop)
            {
                return false;
            }
        }

        return true;
    }

    public static boolean isPrime(int n) {
        boolean r = true;

        for (int i = 2; i <= (int) Math.sqrt(n); i++) {
            if (n % i == 0) {
                r = false;
                break;
            }
        }
        return r;
    }


    public static void generatePossiblePrimeWithLength(int length) {
        int primeNumber = 2, roundCount = 5;
        boolean f = false;
        Random rnd = new Random();

        while(!f) {
            primeNumber = 2 + rnd.nextInt((int) Math.pow(2, length) - 2);

            for(int i = 2; i < 7 && isPrime(i); i++){
                if(primeNumber % i == 0){
                    f = false;
                    break;
                }
                else {
                    f = true;
                }
            }

            if(!isPrimeAccordingToMillerRabin(primeNumber, roundCount)){
                f = false;
            };
        }

        System.out.println("Number: " + primeNumber);
    }


    /*
    Простое число p называется сильно простым, если существуют целые числа r,s,t, удовлетвоярющие следующим условиям:
1. p+1 имеет большой простой делитель s;
2. p-1 имеет большой простой делитель r;
3. r-1 имеет большой простой делитель t.
В этом определении величина делителей зависит от вида атаки, от которой необходимо защититься.
Алгоритм Гордона (Gordon’s algorithm) для генерации сильно простого числа
Выход: Сильно простое число p.
1. Сгенерировать два больших простых числа s и t одной длины;
2. Выбрать число Gus39.gif. Найти первое простое число в последовательности 2it+1, где i= Gus39.gif, i=i+1. Обозначить это число r=2it+1;
3. Вычислить Gus40.gif;
4. Выбрать целое число Gus42.gif. Найти первое простое число в последовательности Gus41.gif, где j = Gus42.gif, j=j+1. Обозначить это число p=Gus41.gif;
5. Вернуть (p).
     */

    public static void generateStrongPrime() {
        int k = 9;
        int i0 = 1;
        int j0 = 1;
        int s = 0;
        int t = 0;

        boolean isp1 = false;
        boolean isp2 = false;

        Random rnd = new Random();

        while(!isp1) {
            s = (int) Math.pow(2, k - 1) + rnd.nextInt((int) Math.pow(2, k) - (int) Math.pow(2, k - 1));
            isp1 = isPrime(s);
        }

        while(!isp2) {
            t = (int) Math.pow(2, k - 1) + rnd.nextInt((int) Math.pow(2, k) - (int) Math.pow(2, k - 1));
            isp2 = isPrime(t);
        }
        int i = i0;

        while(!isPrime(2 * i * t + 1)) {
            i++;
        }
        int r = 2 * i * t + 1;
        int p0 =  2 * pow(s, r - 2, r) * s - 1;
        int j = j0;

        while(!isPrime(p0 + 2 * j * r * s)) {
            j++;
        }
        int p = p0 + 2 * j * r * s;

        System.out.println("Number :" + p);
    }
}