/* Copyright (C) 2010 Sebastian Pancratz Copyright (C) 2010 William Hart Copyright (C) 2011 Fredrik Johansson Copyright (C) 2012 Lina Kulakova Copyright (C) 2013 Martin Lee 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 . */ #undef ulong #define ulong ulongxx/* interferes with system includes */ #include #undef ulong #include #define ulong mp_limb_t #include "flint.h" #include "fmpz_vec.h" #include "fmpz_mod_poly.h" void _fmpz_mod_poly_powmod_fmpz_binexp_preinv(fmpz * res, const fmpz * poly, const fmpz_t e, const fmpz * f, slong lenf, const fmpz* finv, slong lenfinv, const fmpz_t p) { fmpz * T, * Q; slong lenT, lenQ; slong i; if (lenf == 2) { fmpz_powm(res, poly, e, p); return; } lenT = 2 * lenf - 3; lenQ = lenT - lenf + 1; T = _fmpz_vec_init(lenT + lenQ); Q = T + lenT; _fmpz_vec_set(res, poly, lenf - 1); for (i = fmpz_sizeinbase(e, 2) - 2; i >= 0; i--) { _fmpz_mod_poly_sqr(T, res, lenf - 1, p); _fmpz_mod_poly_divrem_newton_n_preinv(Q, res, T, 2 * lenf - 3, f, lenf, finv, lenfinv, p); if (fmpz_tstbit(e, i)) { _fmpz_mod_poly_mul(T, res, lenf - 1, poly, lenf - 1, p); _fmpz_mod_poly_divrem_newton_n_preinv(Q, res, T, 2 * lenf - 3, f, lenf, finv, lenfinv, p); } } _fmpz_vec_clear(T, lenT + lenQ); } void fmpz_mod_poly_powmod_fmpz_binexp_preinv(fmpz_mod_poly_t res, const fmpz_mod_poly_t poly, const fmpz_t e, const fmpz_mod_poly_t f, const fmpz_mod_poly_t finv, const fmpz_mod_ctx_t ctx) { fmpz * q; slong len = poly->length; slong lenf = f->length; slong trunc = lenf - 1; int qcopy = 0; if (lenf == 0) { flint_printf("Exception (fmpz_mod_poly_powmod_fmpz_binexp_preinv)." "Divide by zero.\n"); flint_abort(); } if (lenf == 1) { fmpz_mod_poly_zero(res, ctx); return; } if (fmpz_sgn(e) < 0) { flint_printf("Exception (fmpz_mod_poly_powmod_fmpz_binexp_preinv)." "Negative exp not implemented\n"); flint_abort(); } if (len >= lenf) { fmpz_mod_poly_t t, r; fmpz_mod_poly_init(t, ctx); fmpz_mod_poly_init(r, ctx); fmpz_mod_poly_divrem(t, r, poly, f, ctx); fmpz_mod_poly_powmod_fmpz_binexp_preinv(res, r, e, f, finv, ctx); fmpz_mod_poly_clear(t, ctx); fmpz_mod_poly_clear(r, ctx); return; } if (fmpz_abs_fits_ui(e)) { ulong exp = fmpz_get_ui(e); if (exp <= 2) { if (exp == UWORD (0)) { fmpz_mod_poly_fit_length(res, 1, ctx); fmpz_one(res->coeffs); _fmpz_mod_poly_set_length(res, 1); } else if (exp == UWORD (1)) { fmpz_mod_poly_set(res, poly, ctx); } else { fmpz_mod_poly_mulmod_preinv(res, poly, poly, f, finv, ctx); } return; } } if (len == 0) { fmpz_mod_poly_zero(res, ctx); return; } if (poly->length < trunc) { q = _fmpz_vec_init(trunc); _fmpz_vec_set(q, poly->coeffs, len); _fmpz_vec_zero(q + len, trunc - len); qcopy = 1; } else q = poly->coeffs; if ((res == poly && !qcopy) || (res == f) || (res == finv)) { fmpz_mod_poly_t t; fmpz_mod_poly_init2(t, 2*lenf - 3, ctx); _fmpz_mod_poly_powmod_fmpz_binexp_preinv(t->coeffs, q, e, f->coeffs, lenf, finv->coeffs, finv->length, fmpz_mod_ctx_modulus(ctx)); fmpz_mod_poly_swap(res, t, ctx); fmpz_mod_poly_clear(t, ctx); } else { fmpz_mod_poly_fit_length(res, 2*lenf - 3, ctx); _fmpz_mod_poly_powmod_fmpz_binexp_preinv(res->coeffs, q, e, f->coeffs, lenf, finv->coeffs, finv->length, fmpz_mod_ctx_modulus(ctx)); } if (qcopy) _fmpz_vec_clear(q, trunc); _fmpz_mod_poly_set_length(res, trunc); _fmpz_mod_poly_normalise(res); }