#include "nmod.h"
#include "nmod_poly.h"
void
_nmod_poly_power_sums(nn_ptr res, nn_srcptr poly, slong len, slong n,
nmod_t mod)
{
if (10 * n >= len + 75)
_nmod_poly_power_sums_schoenhage(res, poly, len, n, mod);
else
_nmod_poly_power_sums_naive(res, poly, len, n, mod);
}
void
nmod_poly_power_sums(nmod_poly_t res, const nmod_poly_t poly, slong n)
{
slong len = poly->length;
slong i = 0;
if (len == 0)
{
flint_throw(FLINT_ERROR, "(nmod_poly_power_sums_naive): Zero polynomial.\n");
}
while (poly->coeffs[i] == 0)
i++;
if (n <= 0 || len == 1)
nmod_poly_zero(res);
else if (len == i + 1)
{
nmod_poly_fit_length(res, 1);
_nmod_poly_set_length(res, 1);
NMOD_RED(res->coeffs[0], (ulong) (len - 1), poly->mod);
}
else
{
if (*nmod_poly_lead(poly) != 1)
{
nmod_poly_t t;
nmod_poly_init_preinv(t, poly->mod.n, poly->mod.ninv);
nmod_poly_make_monic(t, poly);
nmod_poly_fit_length(res, n);
_nmod_poly_power_sums(res->coeffs, t->coeffs + i,
len - i, n, t->mod);
nmod_poly_clear(t);
}
else if (poly == res)
{
nmod_poly_t t;
nmod_poly_init_preinv(t, poly->mod.n, poly->mod.ninv);
nmod_poly_fit_length(t, n);
_nmod_poly_power_sums(t->coeffs, poly->coeffs + i,
len - i, n, t->mod);
nmod_poly_swap(t, res);
nmod_poly_clear(t);
}
else
{
nmod_poly_fit_length(res, n);
_nmod_poly_power_sums(res->coeffs, poly->coeffs + i,
len - i, n, poly->mod);
}
if (i)
NMOD_RED(res->coeffs[0], (ulong) (len - 1), poly->mod);
_nmod_poly_set_length(res, n);
_nmod_poly_normalise(res);
}
}
void
_nmod_poly_power_sums_naive(nn_ptr res, nn_srcptr poly, slong len, slong n,
nmod_t mod)
{
slong i, k;
NMOD_RED(res[0], (ulong) (len - 1), mod);
for (k = 1; k < FLINT_MIN(n, len); k++)
{
res[k] = nmod_mul(poly[len - 1 - k], k, mod);
for (i = 1; i < k; i++)
res[k] =
nmod_add(res[k], nmod_mul(poly[len - 1 - k + i], res[i], mod),
mod);
res[k] = nmod_neg(res[k], mod);
}
for (k = len; k < n; k++)
{
res[k] = 0;
for (i = k - len + 1; i < k; i++)
res[k] =
nmod_add(res[k], nmod_mul(poly[len - 1 - k + i], res[i], mod),
mod);
res[k] = nmod_neg(res[k], mod);
}
}
void
nmod_poly_power_sums_naive(nmod_poly_t res, const nmod_poly_t poly, slong n)
{
if (poly->length == 0)
{
flint_throw(FLINT_ERROR, "(nmod_poly_power_sums_naive): Zero polynomial.\n");
}
else if (n <= 0 || poly->length == 1)
{
nmod_poly_zero(res);
}
else
{
if (*nmod_poly_lead(poly) != 1)
{
nmod_poly_t t;
nmod_poly_init_preinv(t, poly->mod.n, poly->mod.ninv);
nmod_poly_make_monic(t, poly);
nmod_poly_fit_length(res, n);
_nmod_poly_power_sums_naive(res->coeffs, t->coeffs,
t->length, n, t->mod);
nmod_poly_clear(t);
}
else if (poly == res)
{
nmod_poly_t t;
nmod_poly_init_preinv(t, poly->mod.n, poly->mod.ninv);
nmod_poly_fit_length(t, n);
_nmod_poly_power_sums_naive(t->coeffs, poly->coeffs,
poly->length, n, t->mod);
nmod_poly_swap(t, res);
nmod_poly_clear(t);
}
else
{
nmod_poly_fit_length(res, n);
_nmod_poly_power_sums_naive(res->coeffs, poly->coeffs,
poly->length, n, poly->mod);
}
_nmod_poly_set_length(res, n);
_nmod_poly_normalise(res);
}
}
void
_nmod_poly_power_sums_schoenhage(nn_ptr res, nn_srcptr poly, slong len,
slong n, nmod_t mod)
{
nn_ptr a, b;
a = (nn_ptr) flint_malloc((2 * len - 1) * sizeof(ulong));
b = a + len;
_nmod_poly_reverse(a, poly, len, len);
_nmod_poly_derivative(b, poly, len, mod);
_nmod_poly_reverse(b, b, len - 1, len - 1);
_nmod_poly_div_series(res, b, len - 1, a, len, n, mod);
flint_free(a);
}
void
nmod_poly_power_sums_schoenhage(nmod_poly_t res, const nmod_poly_t poly,
slong n)
{
if (poly->length == 0)
{
flint_throw(FLINT_ERROR, "(nmod_poly_power_sums_schoenhage): Zero polynomial.\n");
}
else if ((n <= 0) || (poly->length == 1))
{
nmod_poly_zero(res);
}
else
{
if (*nmod_poly_lead(poly) != 1)
{
nmod_poly_t t;
nmod_poly_init_preinv(t, poly->mod.n, poly->mod.ninv);
nmod_poly_make_monic(t, poly);
nmod_poly_fit_length(res, n);
_nmod_poly_power_sums_schoenhage(res->coeffs, t->coeffs,
t->length, n, t->mod);
nmod_poly_clear(t);
}
else if (poly == res)
{
nmod_poly_t t;
nmod_poly_init_preinv(t, poly->mod.n, poly->mod.ninv);
nmod_poly_fit_length(t, n);
_nmod_poly_power_sums_schoenhage(t->coeffs, poly->coeffs,
poly->length, n, t->mod);
nmod_poly_swap(t, res);
nmod_poly_clear(t);
}
else
{
nmod_poly_fit_length(res, n);
_nmod_poly_power_sums_schoenhage(res->coeffs, poly->coeffs,
poly->length, n, poly->mod);
}
_nmod_poly_set_length(res, n);
_nmod_poly_normalise(res);
}
}