#include "fmpz_vec.h"
#include "fmpz_poly.h"
#include "fmpq.h"
#include "fmpq_poly.h"
#define FLINT_NEWTON_INIT(from, to) \
{ \
slong __steps[FLINT_BITS], __i, __from, __to; \
__steps[__i = 0] = __to = (to); \
__from = (from); \
while (__to > __from) \
__steps[++__i] = (__to = (__to + 1) / 2); \
#define FLINT_NEWTON_BASECASE(bc_to) { slong bc_to = __to;
#define FLINT_NEWTON_END_BASECASE }
#define FLINT_NEWTON_LOOP(step_from, step_to) \
{ \
for (__i--; __i >= 0; __i--) \
{ \
slong step_from = __steps[__i+1]; \
slong step_to = __steps[__i]; \
#define FLINT_NEWTON_END_LOOP }}
#define FLINT_NEWTON_END }
#define FMPQ_POLY_INV_NEWTON_CUTOFF 24
static void
_fmpq_poly_inv_series_basecase_rev(fmpz * Qinv, fmpz_t Qinvden,
fmpz * W, fmpz_t Wden,
const fmpz * Q, const fmpz_t Qden, slong Qlen, slong n)
{
slong Wlen;
fmpz *Qrev;
Qlen = FLINT_MIN(Qlen, n);
Wlen = n + Qlen - 1;
Qrev = W + Wlen;
_fmpz_poly_reverse(Qrev, Q, Qlen, Qlen);
_fmpz_vec_zero(W, Wlen - 1);
fmpz_one(W + Wlen - 1);
fmpz_one(Wden);
_fmpq_poly_div(Qinv, Qinvden, W, Wden, Wlen, Qrev, Qden, Qlen, NULL);
_fmpq_poly_canonicalise(Qinv, Qinvden, n);
_fmpz_poly_reverse(Qinv, Qinv, n, n);
}
#define MULLOW(z, x, xn, y, yn, nn) \
if ((xn) >= (yn)) \
_fmpz_poly_mullow(z, x, xn, y, yn, nn); \
else \
_fmpz_poly_mullow(z, y, yn, x, xn, nn); \
void
_fmpq_poly_inv_series_newton(fmpz * Qinv, fmpz_t Qinvden,
const fmpz * Q, const fmpz_t Qden, slong Qlen, slong n)
{
if (fmpz_is_pm1(Q) && fmpz_is_one(Qden))
{
_fmpz_poly_inv_series(Qinv, Q, Qlen, n);
fmpz_one(Qinvden);
return;
}
Qlen = FLINT_MIN(Qlen, n);
if (Qlen == 1)
{
fmpz_set(Qinv, Qden);
fmpz_set(Qinvden, Q);
_fmpq_canonicalise(Qinv, Qinvden);
_fmpz_vec_zero(Qinv + 1, n - 1);
}
else
{
slong alloc, Qnlen, Wlen, W2len;
fmpz * W;
fmpz_t Wden;
alloc = FLINT_MAX(n, 3 * FMPQ_POLY_INV_NEWTON_CUTOFF);
W = _fmpz_vec_init(alloc);
fmpz_init(Wden);
FLINT_NEWTON_INIT(FMPQ_POLY_INV_NEWTON_CUTOFF, n)
FLINT_NEWTON_BASECASE(n)
_fmpq_poly_inv_series_basecase_rev(Qinv, Qinvden, W, Wden, Q, Qden, Qlen, n);
FLINT_NEWTON_END_BASECASE
FLINT_NEWTON_LOOP(m, n)
Qnlen = FLINT_MIN(Qlen, n);
Wlen = FLINT_MIN(Qnlen + m - 1, n);
W2len = Wlen - m;
_fmpz_poly_mulmid(W, Q, Qnlen, Qinv, m, m, Wlen);
fmpz_mul(Wden, Qden, Qinvden);
MULLOW(Qinv + m, Qinv, m, W, W2len, n - m);
fmpz_mul(Qinvden, Qinvden, Wden);
_fmpz_vec_scalar_mul_fmpz(Qinv, Qinv, m, Wden);
_fmpz_vec_neg(Qinv + m, Qinv + m, n - m);
_fmpq_poly_canonicalise(Qinv, Qinvden, n);
FLINT_NEWTON_END_LOOP
FLINT_NEWTON_END
_fmpz_vec_clear(W, alloc);
fmpz_clear(Wden);
}
}
void
fmpq_poly_inv_series_newton(fmpq_poly_t Qinv, const fmpq_poly_t Q, slong n)
{
slong Qlen = Q->length;
Qlen = FLINT_MIN(Qlen, n);
if (Qlen == 0)
{
flint_throw(FLINT_DIVZERO, "Exception (fmpq_poly_inv_series_newton). Division by zero.\n");
}
if (Qinv != Q)
{
fmpq_poly_fit_length(Qinv, n);
_fmpq_poly_inv_series_newton(Qinv->coeffs, Qinv->den, Q->coeffs, Q->den, Qlen, n);
}
else
{
fmpq_poly_t t;
fmpq_poly_init2(t, n);
_fmpq_poly_inv_series_newton(t->coeffs, t->den, Q->coeffs, Q->den, Qlen, n);
fmpq_poly_swap(Qinv, t);
fmpq_poly_clear(t);
}
_fmpq_poly_set_length(Qinv, n);
_fmpq_poly_normalise(Qinv);
}