/*============================================================================= This file is part of ARB. ARB is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. ARB is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with ARB; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA =============================================================================*/ /****************************************************************************** Copyright (C) 2012 Fredrik Johansson ******************************************************************************/ #include "fmprb_poly.h" static void _fmprb_vec_zero(fmprb_struct * A, long n) { long i; for (i = 0; i < n; i++) fmprb_zero(A + i); } void _fmprb_vec_set(fmprb_struct * res, const fmprb_struct * vec, long len) { long i; for (i = 0; i < len; i++) fmprb_set(res + i, vec + i); } static void _fmprb_vec_neg(fmprb_struct * B, const fmprb_struct * A, long n, long prec) { long i; for (i = 0; i < n; i++) fmprb_neg(B + i, A + i); } static void _fmprb_vec_sub(fmprb_struct * C, const fmprb_struct * A, const fmprb_struct * B, long n, long prec) { long i; for (i = 0; i < n; i++) fmprb_sub(C + i, A + i, B + i, prec); } static void _fmprb_vec_add(fmprb_struct * C, const fmprb_struct * A, const fmprb_struct * B, long n, long prec) { long i; for (i = 0; i < n; i++) fmprb_add(C + i, A + i, B + i, prec); } #define NEWTON_EXP_CUTOFF 6 /* with inverse=1 simultaneously computes g = exp(-x) to length n with inverse=0 uses g as scratch space, computing g = exp(-x) only to length (n+1)/2 */ static void _fmprb_poly_exp_series_newton(fmprb_struct * f, fmprb_struct * g, const fmprb_struct * h, long n, long prec, int inverse) { long a[FLINT_BITS]; long i, m, m2, l, alloc; fmprb_struct * T, * U, * hprime; alloc = n; T = _fmprb_vec_init(alloc); U = _fmprb_vec_init(alloc); hprime = _fmprb_vec_init(alloc); _fmprb_poly_derivative(hprime, h, n, prec); fmprb_zero(hprime + n - 1); for (i = 1; (1L << i) < n; i++); a[i = 0] = n; while (n >= NEWTON_EXP_CUTOFF) a[++i] = (n = (n + 1) / 2); /* f := exp(h) + O(x^m), g := exp(-h) + O(x^m2) */ _fmprb_poly_exp_series_basecase(f, h, n, n, prec); _fmprb_poly_inv_series(g, f, (n + 1) / 2, prec); for (i--; i >= 0; i--) { m = n; /* previous length */ n = a[i]; /* new length */ m2 = (m + 1) / 2; l = m - 1; /* shifted for derivative */ /* g := exp(-h) + O(x^m) */ _fmprb_poly_mullow(T, f, m, g, m2, m, prec); _fmprb_poly_mullow(g + m2, g, m2, T + m2, m - m2, m - m2, prec); _fmprb_vec_neg(g + m2, g + m2, m - m2, prec); /* U := h' + g (f' - f h') + O(x^(n-1)) Note: should replace h' by h' mod x^(m-1) */ _fmprb_vec_zero(f + m, n - m); _fmprb_poly_mullow(T, f, n, hprime, n, n, prec); /* should be mulmid */ _fmprb_poly_derivative(U, f, n, prec); fmprb_zero(U + n - 1); /* should skip low terms */ _fmprb_vec_sub(U + l, U + l, T + l, n - l, prec); _fmprb_poly_mullow(T + l, g, n - m, U + l, n - m, n - m, prec); _fmprb_vec_add(U + l, hprime + l, T + l, n - m, prec); /* f := f + f * (h - int U) + O(x^n) = exp(h) + O(x^n) */ _fmprb_poly_integral(U, U, n, prec); /* should skip low terms */ _fmprb_vec_sub(U + m, h + m, U + m, n - m, prec); _fmprb_poly_mullow(f + m, f, n - m, U + m, n - m, n - m, prec); /* g := exp(-h) + O(x^n) */ /* not needed if we only want exp(x) */ if (i == 0 && inverse) { _fmprb_poly_mullow(T, f, n, g, m, n, prec); _fmprb_poly_mullow(g + m, g, m, T + m, n - m, n - m, prec); _fmprb_vec_neg(g + m, g + m, n - m, prec); } } _fmprb_vec_clear(hprime, alloc); _fmprb_vec_clear(T, alloc); _fmprb_vec_clear(U, alloc); } void _fmprb_poly_exp_series(fmprb_struct * f, const fmprb_struct * h, long hlen, long n, long prec) { hlen = FLINT_MIN(hlen, n); if (n < NEWTON_EXP_CUTOFF) { _fmprb_poly_exp_series_basecase(f, h, hlen, n, prec); } else { fmprb_struct *g, *t; fmprb_t u; int fix; g = _fmprb_vec_init((n + 1) / 2); fix = (hlen < n || h == f || !fmprb_is_zero(h)); if (fix) { t = _fmprb_vec_init(n); _fmprb_vec_set(t + 1, h + 1, hlen - 1); } else t = (fmprb_struct *) h; fmprb_init(u); fmprb_exp(u, h, prec); _fmprb_poly_exp_series_newton(f, g, t, n, prec, 0); if (!fmprb_is_one(u)) _fmprb_vec_scalar_mul(f, f, n, u, prec); _fmprb_vec_clear(g, (n + 1) / 2); if (fix) _fmprb_vec_clear(t, n); fmprb_clear(u); } } void fmprb_poly_exp_series(fmprb_poly_t f, const fmprb_poly_t h, long n, long prec) { long hlen = h->length; if (n == 0) { fmprb_poly_zero(f); return; } if (hlen == 0) { fmprb_poly_one(f); return; } fmprb_poly_fit_length(f, n); _fmprb_poly_exp_series(f->coeffs, h->coeffs, hlen, n, prec); _fmprb_poly_set_length(f, n); _fmprb_poly_normalise(f); }