/*
Copyright (C) 2015 Kushagra Singh
This file is part of FLINT.
FLINT is free software: you can redistribute it and/or modify it under
the terms of the GNU Lesser General Public License (LGPL) as published
by the Free Software Foundation; either version 2.1 of the License, or
(at your option) any later version. See .
*/
#include
#include "flint.h"
#include "ulong_extras.h"
/* P (x : z) = 2 * P1 (x0 : z0) */
/*
Coordinates of P :
x = (x0 + z0)^2 * (x0 - z0)^2 mod n
z = 4 * x0 * z0 * ((x0 - z0)^2 + a24 * 4 * x0 * z0) mod n
*/
/* a24 = (a + 2) / 4 mod n */
void
n_factor_ecm_double(mp_limb_t *x, mp_limb_t *z, mp_limb_t x0, mp_limb_t z0,
mp_limb_t n, n_ecm_t n_ecm_inf)
{
mp_limb_t u, v, w;
if (z0 == 0)
{
*x = x0;
*z = 0;
return;
}
u = n_addmod(x0, z0, n);
u = n_mulmod_preinv(u, u, n, n_ecm_inf->ninv, n_ecm_inf->normbits);
v = n_submod(x0, z0, n);
v = n_mulmod_preinv(v, v, n, n_ecm_inf->ninv, n_ecm_inf->normbits);
*x = n_mulmod_preinv(u, v, n, n_ecm_inf->ninv, n_ecm_inf->normbits);
w = n_submod(u, v, n);
u = n_mulmod_preinv(w, n_ecm_inf->a24, n, n_ecm_inf->ninv, n_ecm_inf->normbits);
u = n_addmod(u, v, n);
*z = n_mulmod_preinv(w, u, n, n_ecm_inf->ninv, n_ecm_inf->normbits);
}