#include "nmod.h"
#include "nmod_vec.h"
#include "nmod_poly.h"
#include "gr_poly.h"
void
_nmod_poly_divexact(nn_ptr Q, nn_srcptr A, slong lenA, nn_srcptr B, slong lenB, nmod_t mod)
{
if (lenA == lenB)
{
Q[0] = nmod_div(A[lenA - 1], B[lenB - 1], mod);
}
else if (lenB == 1)
{
_nmod_vec_scalar_mul_nmod(Q, A, lenA, nmod_inv(B[0], mod), mod);
}
else
{
gr_ctx_t ctx;
_gr_ctx_init_nmod(ctx, &mod);
GR_MUST_SUCCEED(_gr_poly_divexact(Q, A, lenA, B, lenB, ctx));
}
}
void
nmod_poly_divexact(nmod_poly_t Q,
const nmod_poly_t A, const nmod_poly_t B)
{
nmod_poly_t tQ;
nn_ptr q;
slong A_len, B_len;
B_len = B->length;
if (B_len == 0)
{
if (nmod_poly_modulus(B) == 1)
{
nmod_poly_set(Q, A);
return;
}
else
{
flint_throw(FLINT_DIVZERO, "Exception (nmod_poly_divexact). Division by zero.\n");
}
}
A_len = A->length;
if (A_len < B_len)
{
nmod_poly_zero(Q);
return;
}
if (Q == A || Q == B)
{
nmod_poly_init2(tQ, A->mod.n, A_len - B_len + 1);
q = tQ->coeffs;
}
else
{
nmod_poly_fit_length(Q, A_len - B_len + 1);
q = Q->coeffs;
}
_nmod_poly_divexact(q, A->coeffs, A_len, B->coeffs, B_len, A->mod);
if (Q == A || Q == B)
{
nmod_poly_swap(tQ, Q);
nmod_poly_clear(tQ);
}
Q->length = A_len - B_len + 1;
}