/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
// package ellipticCurve;
import static java.
lang.
Math.
cbrt; import static java.
lang.
Math.
pow; import static java.
lang.
Math.
sqrt; import java.util.ArrayList;
import java.util.Locale;
/**
*
* @author Trần Hoàng
*/
double x, y;
public static void main
(String[] args
) { // /**-----------------------**/
// ELLIPTIC (E) trên trường hữu hạn F(M): y^2 mod M = x^3 + a*x + b mod M
// Khởi tạo các hệ số của (E), M
long a, b, M;
a = (long)1;
b = (long)1;
M = (long)17;
// Chọn điểm cơ sở
// In điểm cơ sở ra
p.inString("P");
// Tìm bậc của (E)
caculateRankECC(p, a, b, M);
/**-----------------------**/
}
}
// Contructor có đối số của thằng Point
public Point(double x,
double y
) { this.x = x;
this.y = y;
}
// Tìm tạo độ điểm x từ điểm y
public Point calXfromY
(double y,
int a,
int b
) { return new Point(cbrt
(pow
(y,
2) - b
), y
); }
// Tạo một điểm zeroPoint ~ 0
public Point zeroPoint
() { }
// Kiểm tra một điểm có là zeroPoint hay không
public static boolean isZeroPoint
(Point p
) { return p.x > 1e18 || p.x < -1e18;
}
// Tạo điểm đối xứng qua trục hoành
public Point negativePoint
() { return new Point(this.
x,
-this.
y); }
public static long pow_modulo(long base, long exponent, long M) {
if(exponent == 0) return 1;
if(exponent == 1) return base;
long half = pow_modulo(base, exponent/2, M);
if((exponent&1) > 0) return (((half*half)%M)*base)%M;
else return (half*half)%M;
}
public static long modulo_invert_fermat(long n, long M) {
return pow_modulo(n, M-2, M);
}
public static long modNegative(long y, long N) {
if(y < (long)0) {
return ((y+N)%N);
}
return (y%N);
}
public static Point duplicatedPointModN
(Point p,
long a,
long b,
long N
) { if( isZeroPoint(p) ) {
return p;
}
// delta = (3*(xP^2) + a) / ( 2*(yP)
long delta_tu = (long)(3*p.x*p.x) + a;
long delta_mau = 2*(long)p.y;
// vì sau khi tính delta có thể âm nên ta cần mod cho N để hết âm
delta_tu = modNegative(delta_tu, N);
delta_mau = modNegative(delta_mau, N);
// delta = delta_tu/delta_mau => cần tính mod nghịch đảo
// Nếu delta_mau chia hết cho N thì không tồn tại modulo nghịch đảo.
if(delta_mau%N == 0) {
System.
out.
println("Khong ton tai modulo nghich dao"); return new Point().
zeroPoint(); }
else {
delta_mau = modulo_invert_fermat(delta_mau, N);
}
long delta = (delta_tu*delta_mau)%N;
// System.out.println("Nhan delta: " + delta);
// R = 2*P; xR = delta^2 - 2*xP theo (IV)
long xR = ((delta*delta) - 2*(long)p.x);
// System.out.println("xR: " + xR);
xR = modNegative(xR, N);
// System.out.println("xR: " + xR);
// R = 2*P; yR = delta*(xP - xR) - yP theo(V)
long yR = (delta*((long)p.x - xR) - (long)p.y);
// System.out.println("yR: " + yR);
yR = modNegative(yR, N);
return new Point((double)xR,
(double )yR
); }
// check 2 điểm P, Q bằng nhau => dùng công thức nhân đôi
if( p.x == q.x && p.y == q.y )
return duplicatedPointModN(p,a, b, N);
// check zeroPoint P
if( isZeroPoint(q) )
return p;
// check zeroPoint Q
if( isZeroPoint(p) )
return q;
/*
Công thức cộng 2 điểm P(xP, yP) và Q(xQ, yQ) trên đường cong Elliptic
P # Q, P #O, Q # O
delta = delta_tu/ delta_mau = (yQ - yP) / (xQ - xP)
*/
long delta_tu = modNegative(((long)q.y - (long)p.y), N);
long delta_mau = modNegative(((long)q.x - (long)p.x), N);
// delta = delta_tu/delta_mau => cần tính mod nghịch đảo
// Nếu delta_mau chia hết cho N thì không tồn tại modulo nghịch đảo.
if(delta_mau%N == 0) {
System.
out.
println("Khong ton tai modulo nghich dao"); return new Point().
zeroPoint(); }
else {
delta_mau = modulo_invert_fermat(delta_mau, N);
}
long delta = (delta_tu*delta_mau)%N;
// System.out.println("plus - delta: " + delta);
long xR = ((delta*delta) - ((long)p.x + (long)q.x));
// System.out.println("xR: " + xR);
xR = modNegative(xR, N);
long yR = (delta*((long)p.x - (long)xR) - (long)p.y);
// System.out.println("yR: " + yR);
yR = modNegative(yR, N);
return new Point((double)xR,
(double)yR
); }
public void inString
(String msg
) { System.
out.
println( msg
+ "(" + x
+ ", " + y
+ ")"); }
public static void caculateRankECC
(Point p,
long a,
long b,
long N
) { int cnt = 0;
// điểm temp là điểm trung gian, dùng để lưu 1 điểm vừa tính được
// Lặp đến khi tìm thấy bậc của ECC
for( int i = 2;; i++) {
cnt++;
if(i == 2) {
temp = duplicatedPointModN(p, a, b, N);
}
else {
temp = plusModN(p, temp, a, b, N);
}
if(isZeroPoint(temp)) {
msg = i + "P = 0\nBậc của #E" + N + " = " + i;
// Kết thúc việc tìm bậc
return;
}
else msg = i+ "P";
temp.inString(msg);
}
}
}
/*
 * To change this license header, choose License Headers in Project Properties.
 * To change this template file, choose Tools | Templates
 * and open the template in the editor.
 */
// package ellipticCurve;
import static java.lang.Math.cbrt;
import static java.lang.Math.pow;
import static java.lang.Math.sqrt;
import java.util.ArrayList;
import java.util.Locale;

/**
 *
 * @author Trần Hoàng
 */

class Point {
    
    double x, y;
    
    public static void main(String[] args) {
//      /**-----------------------**/
//		ELLIPTIC (E) trên trường hữu hạn F(M):            y^2 mod M = x^3 + a*x + b mod M
//		Khởi tạo các hệ số của (E), M
		long a, b, M;
		a = (long)1; 
		b = (long)1;
		M = (long)17;
//		Chọn điểm cơ sở
        Point p = new Point(16, 13);
//		In điểm cơ sở ra
        p.inString("P");
//		Tìm bậc của (E)
        caculateRankECC(p, a, b, M);
        /**-----------------------**/    
    }

    public Point() {
    }
    
    // Contructor có đối số của thằng Point
    public Point(double x, double y) {
        this.x = x;
        this.y = y;
    }
    // Tìm tạo độ điểm x từ điểm y
    public Point calXfromY(double y, int a, int b) {
        return new Point(cbrt(pow(y,2) - b), y);
    }
    // Tạo một điểm zeroPoint ~ 0
    public Point zeroPoint() {
        return new Point(Double.POSITIVE_INFINITY, Double.POSITIVE_INFINITY);
    }
    // Kiểm tra một điểm có là zeroPoint hay không
    public static boolean isZeroPoint(Point p) {
        return p.x > 1e18 || p.x < -1e18;
    }
    // Tạo điểm đối xứng qua trục hoành
    public Point negativePoint() {
        return new Point(this.x, -this.y);
    }
    
    public static long pow_modulo(long base, long exponent, long M) {
        if(exponent == 0) return 1;
        if(exponent == 1) return base;
        long half = pow_modulo(base, exponent/2, M);
        if((exponent&1) > 0) return (((half*half)%M)*base)%M;
        else return (half*half)%M; 
    }
    
    public static long modulo_invert_fermat(long n, long M) {
        return pow_modulo(n, M-2, M);
    }
    
    public static long modNegative(long y, long N) {
        if(y < (long)0) {
            y = N*Math.abs(y/N) + y;
            return ((y+N)%N);
        }
        return (y%N);
    }
    
    public static Point duplicatedPointModN(Point p, long a, long b, long N) {
        if( isZeroPoint(p) ) {
            return p;
        }
        // delta = (3*(xP^2) + a) / ( 2*(yP)
        long delta_tu = (long)(3*p.x*p.x) + a;
        long delta_mau =  2*(long)p.y;
        // vì sau khi tính delta có thể âm nên ta cần mod cho N để hết âm
        delta_tu = modNegative(delta_tu, N);
        delta_mau = modNegative(delta_mau, N);
        // delta = delta_tu/delta_mau => cần tính mod nghịch đảo
        // Nếu delta_mau chia hết cho N thì không tồn tại modulo nghịch đảo.
        if(delta_mau%N == 0) {
            System.out.println("Khong ton tai modulo nghich dao");
            return new Point().zeroPoint();
        }
        else {
            delta_mau = modulo_invert_fermat(delta_mau, N);
        }
        long delta = (delta_tu*delta_mau)%N;
//        System.out.println("Nhan delta: " + delta);
        // R = 2*P; xR = delta^2 - 2*xP theo (IV)
        long xR = ((delta*delta) - 2*(long)p.x);
//        System.out.println("xR: " + xR);
        xR = modNegative(xR, N);
//        System.out.println("xR: " + xR);
        // R = 2*P; yR = delta*(xP - xR) - yP theo(V)
        long yR = (delta*((long)p.x - xR) - (long)p.y);
//        System.out.println("yR: " + yR);
        yR = modNegative(yR, N);
        return new Point((double)xR, (double )yR);
    }
    
    public static Point plusModN(Point p, Point q, long a, long b, long N) {
        // check 2 điểm P, Q bằng nhau => dùng công thức nhân đôi
        if( p.x == q.x && p.y == q.y ) 
            return duplicatedPointModN(p,a, b, N);
        // check zeroPoint P
        if( isZeroPoint(q) )
            return p;
        // check zeroPoint Q
        if( isZeroPoint(p) )
            return q;
        /*
            Công thức cộng 2 điểm P(xP, yP) và Q(xQ, yQ) trên đường cong Elliptic
            P # Q, P #O, Q # O
            delta = delta_tu/ delta_mau = (yQ - yP) / (xQ - xP)
        */
        long delta_tu = modNegative(((long)q.y - (long)p.y), N);
        long delta_mau = modNegative(((long)q.x - (long)p.x), N);
        
        // delta = delta_tu/delta_mau => cần tính mod nghịch đảo
        // Nếu delta_mau chia hết cho N thì không tồn tại modulo nghịch đảo.
        if(delta_mau%N == 0) {
            System.out.println("Khong ton tai modulo nghich dao");
            return new Point().zeroPoint();
        }
        else {
            delta_mau = modulo_invert_fermat(delta_mau, N);
        }
        long delta = (delta_tu*delta_mau)%N;
//        System.out.println("plus - delta: " + delta);
        long xR = ((delta*delta) - ((long)p.x + (long)q.x));
//        System.out.println("xR: " + xR);
        xR = modNegative(xR, N);
        long yR = (delta*((long)p.x - (long)xR) - (long)p.y);
//        System.out.println("yR: " + yR);
        yR = modNegative(yR, N);
        return new Point((double)xR, (double)yR);
    }
    
    public void inString(String msg) {
        System.out.println( msg + "(" + x + ", " + y + ")");
    }

    public static void caculateRankECC(Point p, long a, long b, long N) {
        int cnt = 0;
        // điểm temp là điểm trung gian, dùng để lưu 1 điểm vừa tính được
        Point temp = new Point().zeroPoint();
        // Lặp đến khi tìm thấy bậc của ECC
        for( int i = 2;; i++) {
            cnt++;
            if(i == 2) {
                temp = duplicatedPointModN(p, a, b, N);
            }
            else {
                temp = plusModN(p, temp, a, b, N);
            }
            String msg = "";
            if(isZeroPoint(temp)) {
                msg = i + "P = 0\nBậc của #E" + N + " = " + i;
                System.out.println(msg);
                // Kết thúc việc tìm bậc
                return;
            }
            else msg = i+ "P";
            temp.inString(msg);
        }
    }
}
