|
1 | 1 | /* |
2 | 2 | Copyright (C) 2016 William Hart |
| 3 | + Copyright (C) 2026 Fredrik Johansson |
3 | 4 |
|
4 | 5 | This file is part of FLINT. |
5 | 6 |
|
|
12 | 13 | #include "fmpz.h" |
13 | 14 | #include "fmpz_vec.h" |
14 | 15 | #include "fmpz_poly.h" |
| 16 | +#include "fmpz_mod.h" |
| 17 | +#include "fmpz_mod_vec.h" |
| 18 | +#include "fmpz_mod_poly.h" |
15 | 19 |
|
16 | 20 | void fmpz_poly_divlow_smodp(fmpz * res, const fmpz_poly_t f, |
17 | | - const fmpz_poly_t g, const fmpz_t p, slong n) |
| 21 | + const fmpz_poly_t g, const fmpz_t p, slong n) |
18 | 22 | { |
19 | | - fmpz_t d, cinv; |
20 | | - slong i = 0, k, zeroes; |
21 | | - fmpz_poly_t tf; |
22 | | - |
23 | | - fmpz_init(d); |
24 | | - fmpz_init(cinv); |
25 | | - |
26 | | - while (fmpz_is_zero(g->coeffs + i)) |
27 | | - i++; |
28 | | - |
29 | | - zeroes = i; |
30 | | - |
31 | | - fmpz_poly_init2(tf, n + zeroes); |
32 | | - |
33 | | - fmpz_poly_set(tf, f); |
34 | | - |
35 | | - if (fmpz_sgn(g->coeffs + zeroes) >= 0) |
36 | | - fmpz_gcdinv(d, cinv, g->coeffs + zeroes, p); |
37 | | - else |
38 | | - { |
39 | | - fmpz_t temp; |
40 | | - |
41 | | - fmpz_init(temp); |
42 | | - |
43 | | - fmpz_add(temp, g->coeffs + zeroes, p); |
44 | | - fmpz_gcdinv(d, cinv, temp, p); |
45 | | - |
46 | | - fmpz_clear(temp); |
47 | | - } |
48 | | - |
49 | | - if (!fmpz_is_one(d)) |
50 | | - { |
51 | | - flint_throw(FLINT_ERROR, "Exception (fmpz_poly_divlow_smodp). Impossible inverse.\n"); |
52 | | - } |
53 | | - |
54 | | - for (k = 0; k < n; i++, k++) |
55 | | - { |
56 | | - fmpz_mul(res + k, tf->coeffs + i, cinv); |
57 | | - |
58 | | - fmpz_smod(res + k, res + k, p); |
59 | | - |
60 | | - _fmpz_vec_scalar_submul_fmpz(tf->coeffs + i, g->coeffs + zeroes, |
61 | | - FLINT_MIN(g->length - zeroes, n - k), res + k); |
62 | | - _fmpz_vec_scalar_smod_fmpz(tf->coeffs + i, tf->coeffs + i, |
63 | | - FLINT_MIN(g->length - zeroes, n - k), p); |
64 | | - } |
65 | | - |
66 | | - fmpz_poly_clear(tf); |
67 | | - fmpz_clear(cinv); |
68 | | - fmpz_clear(d); |
| 23 | + fmpz_mod_ctx_t ctx; |
| 24 | + fmpz *tA, *tB; |
| 25 | + slong zeroes, glen, flen, Alen, Blen; |
| 26 | + |
| 27 | + zeroes = 0; |
| 28 | + while (fmpz_is_zero(g->coeffs + zeroes)) |
| 29 | + zeroes++; |
| 30 | + |
| 31 | + flen = f->length - zeroes; |
| 32 | + glen = g->length - zeroes; |
| 33 | + Alen = FLINT_MAX(0, FLINT_MIN(flen, n)); |
| 34 | + Blen = FLINT_MIN(glen, n); |
| 35 | + |
| 36 | + if (Alen == 0) |
| 37 | + { |
| 38 | + _fmpz_vec_zero(res, n); |
| 39 | + return; |
| 40 | + } |
| 41 | + |
| 42 | + fmpz_mod_ctx_init(ctx, p); |
| 43 | + tA = _fmpz_vec_init(Alen); |
| 44 | + tB = _fmpz_vec_init(Blen); |
| 45 | + |
| 46 | + _fmpz_mod_vec_set_fmpz_vec(tA, f->coeffs + zeroes, Alen, ctx); |
| 47 | + _fmpz_mod_vec_set_fmpz_vec(tB, g->coeffs + zeroes, Blen, ctx); |
| 48 | + _fmpz_mod_poly_div_series(res, tA, Alen, tB, Blen, n, ctx); |
| 49 | + _fmpz_mod_vec_get_fmpz_vec_smod(res, res, n, ctx); |
| 50 | + |
| 51 | + _fmpz_vec_clear(tA, Alen); |
| 52 | + _fmpz_vec_clear(tB, Blen); |
| 53 | + fmpz_mod_ctx_clear(ctx); |
69 | 54 | } |
| 55 | + |
0 commit comments