#include "fq_zech.h"
#include "fq_zech_poly.h"
#include "mpoly.h"
#include "fq_zech_mpoly_factor.h"
void fq_zech_bpoly_clear(fq_zech_bpoly_t A, const fq_zech_ctx_t ctx)
{
slong i;
if (A->alloc > 0)
{
FLINT_ASSERT(A->coeffs != NULL);
for (i = 0; i < A->alloc; i++)
fq_zech_poly_clear(A->coeffs + i, ctx);
flint_free(A->coeffs);
}
else
{
FLINT_ASSERT(A->coeffs == NULL);
}
}
void fq_zech_bpoly_normalise(fq_zech_bpoly_t A, const fq_zech_ctx_t ctx)
{
while (A->length > 0 && fq_zech_poly_is_zero(A->coeffs + A->length - 1, ctx))
A->length--;
}
void fq_zech_bpoly_realloc(fq_zech_bpoly_t A, slong len, const fq_zech_ctx_t ctx)
{
slong i;
slong old_alloc = A->alloc;
slong new_alloc = FLINT_MAX(len, old_alloc + 1 + old_alloc/2);
FLINT_ASSERT(A->alloc >= 0);
if (len <= A->alloc)
return;
if (A->alloc > 0)
{
A->coeffs = (fq_zech_poly_struct *) flint_realloc(A->coeffs,
new_alloc * sizeof(fq_zech_poly_struct));
}
else
{
FLINT_ASSERT(A->coeffs == NULL);
A->coeffs = (fq_zech_poly_struct *) flint_malloc(
new_alloc * sizeof(fq_zech_poly_struct));
}
for (i = old_alloc; i < new_alloc; i++)
fq_zech_poly_init(A->coeffs + i, ctx);
A->alloc = len;
}
int fq_zech_bpoly_is_canonical(const fq_zech_bpoly_t A, const fq_zech_ctx_t ctx)
{
slong i;
if (A->length < 0)
return 0;
for (i = 0; i < A->length; i++)
{
if (i + 1 == A->length && fq_zech_poly_is_zero(A->coeffs + i, ctx))
return 0;
}
return 1;
}
slong fq_zech_bpoly_degree1(const fq_zech_bpoly_t A, const fq_zech_ctx_t FLINT_UNUSED(ctx))
{
slong i, len = 0;
for (i = 0; i < A->length; i++)
len = FLINT_MAX(len, A->coeffs[i].length);
return len - 1;
}
int fq_zech_bpoly_equal(
const fq_zech_bpoly_t A,
const fq_zech_bpoly_t B,
const fq_zech_ctx_t ctx)
{
slong i;
if (A->length != B->length)
return 0;
for (i = 0; i < A->length; i++)
{
if (!fq_zech_poly_equal(A->coeffs + i, B->coeffs + i, ctx))
return 0;
}
return 1;
}
void fq_zech_bpoly_get_coeff(fq_zech_t c, const fq_zech_bpoly_t A, slong e0, slong e1, const fq_zech_ctx_t ctx)
{
if (e0 >= A->length)
fq_zech_zero(c, ctx);
else
fq_zech_poly_get_coeff(c, A->coeffs + e0, e1, ctx);
}
void fq_zech_bpoly_set_coeff_fq_zech(
fq_zech_bpoly_t A,
slong xi,
slong yi,
const fq_zech_t c,
const fq_zech_ctx_t ctx)
{
slong i;
if (xi >= A->length)
{
fq_zech_bpoly_fit_length(A, xi + 1, ctx);
for (i = A->length; i <= xi; i++)
fq_zech_poly_zero(A->coeffs + i, ctx);
A->length = xi + 1;
}
fq_zech_poly_set_coeff(A->coeffs + xi, yi, c, ctx);
fq_zech_bpoly_normalise(A, ctx);
}
void fq_zech_bpoly_set_fq_zech_poly_var0(
fq_zech_bpoly_t A,
const fq_zech_poly_t B,
const fq_zech_ctx_t ctx)
{
slong i;
fq_zech_bpoly_fit_length(A, B->length, ctx);
for (i = 0; i < B->length; i++)
fq_zech_poly_set_fq_zech(A->coeffs + i, B->coeffs + i, ctx);
A->length = B->length;
fq_zech_bpoly_normalise(A, ctx);
}
void fq_zech_bpoly_set_fq_zech_poly_var1(
fq_zech_bpoly_t A,
const fq_zech_poly_t B,
const fq_zech_ctx_t ctx)
{
fq_zech_bpoly_fit_length(A, 1, ctx);
fq_zech_poly_set(A->coeffs + 0, B, ctx);
A->length = !fq_zech_poly_is_zero(A->coeffs + 0, ctx);
}
void fq_zech_bpoly_mul(
fq_zech_bpoly_t A,
const fq_zech_bpoly_t B,
const fq_zech_bpoly_t C,
const fq_zech_ctx_t ctx)
{
slong i, j;
fq_zech_poly_t t;
FLINT_ASSERT(A != B);
FLINT_ASSERT(A != C);
if (B->length < 1 || C->length < 1)
{
A->length = 0;
return;
}
fq_zech_poly_init(t, ctx);
fq_zech_bpoly_fit_length(A, B->length + C->length - 1, ctx);
for (i = 0; i < B->length + C->length - 1; i++)
fq_zech_poly_zero(A->coeffs + i, ctx);
for (i = 0; i < B->length; i++)
{
for (j = 0; j < C->length; j++)
{
fq_zech_poly_mul(t, B->coeffs + i, C->coeffs + j, ctx);
fq_zech_poly_add(A->coeffs + i + j, A->coeffs + i + j, t, ctx);
}
}
fq_zech_poly_clear(t, ctx);
A->length = B->length + C->length - 1;
fq_zech_bpoly_normalise(A, ctx);
}
void fq_zech_bpoly_mul_series(
fq_zech_bpoly_t A,
const fq_zech_bpoly_t B,
const fq_zech_bpoly_t C,
slong order,
const fq_zech_ctx_t ctx)
{
slong i, j;
fq_zech_poly_t t;
FLINT_ASSERT(A != B);
FLINT_ASSERT(A != C);
fq_zech_poly_init(t, ctx);
fq_zech_bpoly_fit_length(A, B->length + C->length - 1, ctx);
for (i = 0; i < B->length + C->length - 1; i++)
fq_zech_poly_zero(A->coeffs + i, ctx);
for (i = 0; i < B->length; i++)
{
for (j = 0; j < C->length; j++)
{
fq_zech_poly_mullow(t, B->coeffs + i, C->coeffs + j, order, ctx);
fq_zech_poly_add(A->coeffs + i + j, A->coeffs + i + j, t, ctx);
}
}
fq_zech_poly_clear(t, ctx);
A->length = B->length + C->length - 1;
fq_zech_bpoly_normalise(A, ctx);
}
void fq_zech_bpoly_add(
fq_zech_bpoly_t A,
const fq_zech_bpoly_t B,
const fq_zech_bpoly_t C,
const fq_zech_ctx_t ctx)
{
slong i;
slong Alen = FLINT_MAX(B->length, C->length);
fq_zech_bpoly_fit_length(A, Alen, ctx);
for (i = 0; i < Alen; i++)
{
if (i < B->length)
{
if (i < C->length)
{
fq_zech_poly_add(A->coeffs + i, B->coeffs + i, C->coeffs + i, ctx);
}
else
{
fq_zech_poly_set(A->coeffs + i, B->coeffs + i, ctx);
}
}
else
{
FLINT_ASSERT(i < C->length);
fq_zech_poly_set(A->coeffs + i, C->coeffs + i, ctx);
}
}
A->length = Alen;
fq_zech_bpoly_normalise(A, ctx);
}
void fq_zech_bpoly_one(fq_zech_bpoly_t A, const fq_zech_ctx_t ctx)
{
fq_zech_bpoly_fit_length(A, 1, ctx);
A->length = 1;
fq_zech_poly_one(A->coeffs + 0, ctx);
}
void fq_zech_bpoly_sub(
fq_zech_bpoly_t A,
const fq_zech_bpoly_t B,
const fq_zech_bpoly_t C,
const fq_zech_ctx_t ctx)
{
slong i;
slong Alen = FLINT_MAX(B->length, C->length);
fq_zech_bpoly_fit_length(A, Alen, ctx);
for (i = 0; i < Alen; i++)
{
if (i < B->length)
{
if (i < C->length)
{
fq_zech_poly_sub(A->coeffs + i, B->coeffs + i, C->coeffs + i, ctx);
}
else
{
fq_zech_poly_set(A->coeffs + i, B->coeffs + i, ctx);
}
}
else
{
FLINT_ASSERT(i < C->length);
fq_zech_poly_neg(A->coeffs + i, C->coeffs + i, ctx);
}
}
A->length = Alen;
fq_zech_bpoly_normalise(A, ctx);
}
void fq_zech_bpoly_derivative(
fq_zech_bpoly_t A,
const fq_zech_bpoly_t B,
const fq_zech_ctx_t ctx)
{
slong i;
slong Blen = B->length;
fq_zech_t c;
if (Blen < 2)
{
fq_zech_bpoly_zero(A, ctx);
return;
}
fq_zech_init(c, ctx);
fq_zech_bpoly_fit_length(A, Blen - 1, ctx);
for (i = 1; i < Blen; i++)
{
fq_zech_set_ui(c, i, ctx);
fq_zech_poly_scalar_mul_fq_zech(A->coeffs + i - 1, B->coeffs + i, c, ctx);
}
A->length = Blen - 1;
fq_zech_bpoly_normalise(A, ctx);
fq_zech_clear(c, ctx);
}
void fq_zech_bpoly_divrem_series(
fq_zech_bpoly_t Q,
fq_zech_bpoly_t R,
const fq_zech_bpoly_t A,
const fq_zech_bpoly_t B,
slong order,
const fq_zech_ctx_t ctx)
{
slong i, qoff;
fq_zech_poly_t q, t, binv;
FLINT_ASSERT(R != A);
FLINT_ASSERT(R != B);
FLINT_ASSERT(Q != A);
FLINT_ASSERT(Q != B);
FLINT_ASSERT(B->length > 0);
fq_zech_poly_init(q, ctx);
fq_zech_poly_init(t, ctx);
fq_zech_poly_init(binv, ctx);
fq_zech_bpoly_set(R, A, ctx);
for (i = 0; i < R->length; i++)
fq_zech_poly_truncate(R->coeffs + i, order, ctx);
fq_zech_bpoly_normalise(R, ctx);
fq_zech_poly_inv_series(binv, B->coeffs + B->length - 1, order, ctx);
Q->length = 0;
while (R->length >= B->length)
{
fq_zech_poly_mullow(q, R->coeffs + R->length - 1, binv, order, ctx);
for (i = 0; i < B->length; i++)
{
fq_zech_poly_mullow(t, B->coeffs + i, q, order, ctx);
fq_zech_poly_sub(R->coeffs + i + R->length - B->length,
R->coeffs + i + R->length - B->length, t, ctx);
}
qoff = R->length - B->length;
FLINT_ASSERT(qoff >= 0);
if (qoff >= Q->length)
{
fq_zech_bpoly_fit_length(Q, qoff + 1, ctx);
for (i = Q->length; i <= qoff; i++)
fq_zech_poly_zero(Q->coeffs + i, ctx);
Q->length = qoff + 1;
}
fq_zech_poly_set(Q->coeffs + qoff, q, ctx);
FLINT_ASSERT(fq_zech_poly_is_zero(R->coeffs + R->length - 1, ctx));
fq_zech_bpoly_normalise(R, ctx);
}
fq_zech_poly_clear(q, ctx);
fq_zech_poly_clear(t, ctx);
fq_zech_poly_clear(binv, ctx);
}
int fq_zech_bpoly_divides(
fq_zech_bpoly_t Q,
const fq_zech_bpoly_t A,
const fq_zech_bpoly_t B,
const fq_zech_ctx_t ctx)
{
slong i, qoff;
int divides;
fq_zech_poly_t q, t;
fq_zech_bpoly_t R;
FLINT_ASSERT(Q != A);
FLINT_ASSERT(Q != B);
FLINT_ASSERT(B->length > 0);
fq_zech_poly_init(q, ctx);
fq_zech_poly_init(t, ctx);
fq_zech_bpoly_init(R, ctx);
fq_zech_bpoly_set(R, A, ctx);
Q->length = 0;
while (R->length >= B->length)
{
fq_zech_poly_divrem(q, t, R->coeffs + R->length - 1,
B->coeffs + B->length - 1, ctx);
if (!fq_zech_poly_is_zero(t, ctx))
{
divides = 0;
goto cleanup;
}
for (i = 0; i < B->length; i++)
{
fq_zech_poly_mul(t, B->coeffs + i, q, ctx);
fq_zech_poly_sub(R->coeffs + i + R->length - B->length,
R->coeffs + i + R->length - B->length, t, ctx);
}
qoff = R->length - B->length;
FLINT_ASSERT(qoff >= 0);
if (qoff >= Q->length)
{
fq_zech_bpoly_fit_length(Q, qoff + 1, ctx);
for (i = Q->length; i <= qoff; i++)
fq_zech_poly_zero(Q->coeffs + i, ctx);
Q->length = qoff + 1;
}
fq_zech_poly_set(Q->coeffs + qoff, q, ctx);
FLINT_ASSERT(fq_zech_poly_is_zero(R->coeffs + R->length - 1, ctx));
fq_zech_bpoly_normalise(R, ctx);
}
divides = (R->length == 0);
cleanup:
fq_zech_poly_clear(q, ctx);
fq_zech_poly_clear(t, ctx);
fq_zech_bpoly_clear(R, ctx);
return divides;
}
void fq_zech_bpoly_set(
fq_zech_bpoly_t A,
const fq_zech_bpoly_t B,
const fq_zech_ctx_t ctx)
{
slong i;
if (A == B)
return;
fq_zech_bpoly_fit_length(A, B->length, ctx);
A->length = B->length;
for (i = 0; i < B->length; i++)
fq_zech_poly_set(A->coeffs + i, B->coeffs + i, ctx);
}
void fq_zech_bpoly_make_primitive(
fq_zech_poly_t g,
fq_zech_bpoly_t A,
const fq_zech_ctx_t ctx)
{
slong Alen = A->length;
slong i;
fq_zech_poly_t q, r;
fq_zech_poly_init(q, ctx);
fq_zech_poly_init(r, ctx);
fq_zech_poly_zero(g, ctx);
for (i = 0; i < Alen; i++)
{
fq_zech_poly_gcd(q, g, A->coeffs + i, ctx);
fq_zech_poly_swap(g, q, ctx);
}
for (i = 0; i < Alen; i++)
{
fq_zech_poly_divrem(q, r, A->coeffs + i, g, ctx);
fq_zech_poly_set(A->coeffs + i, q, ctx);
}
if (Alen > 0)
{
fq_zech_poly_struct * Alead = A->coeffs + Alen - 1;
fq_zech_struct * c_ = Alead->coeffs + Alead->length - 1;
fq_zech_t c;
fq_zech_init(c, ctx);
if (!fq_zech_is_one(c_, ctx))
{
fq_zech_poly_scalar_mul_fq_zech(g, g, c_, ctx);
fq_zech_inv(c, c_, ctx);
for (i = 0; i < Alen; i++)
fq_zech_poly_scalar_mul_fq_zech(A->coeffs + i,
A->coeffs + i, c, ctx);
}
fq_zech_clear(c, ctx);
}
fq_zech_poly_clear(q, ctx);
fq_zech_poly_clear(r, ctx);
}
static void _fq_zech_poly_taylor_shift_horner(
fq_zech_struct * poly,
const fq_zech_t c,
slong n,
const fq_zech_ctx_t ctx)
{
slong i, j;
fq_zech_t p;
fq_zech_init(p, ctx);
for (i = n - 2; i >= 0; i--)
{
for (j = i; j < n - 1; j++)
{
fq_zech_mul(p, poly + (j + 1), c, ctx);
fq_zech_add(poly + j, poly + j, p, ctx);
}
}
fq_zech_clear(p, ctx);
}
void fq_zech_bpoly_taylor_shift_var1(
fq_zech_bpoly_t A,
const fq_zech_bpoly_t B,
const fq_zech_t c,
const fq_zech_ctx_t ctx)
{
slong i;
fq_zech_bpoly_set(A, B, ctx);
for (i = A->length - 1; i >= 0; i--)
_fq_zech_poly_taylor_shift_horner(A->coeffs[i].coeffs, c,
A->coeffs[i].length, ctx);
}
void fq_zech_bpoly_taylor_shift_var0(
fq_zech_bpoly_t A,
const fq_zech_t alpha,
const fq_zech_ctx_t ctx)
{
slong n, i, j;
fq_zech_poly_t t;
if (fq_zech_is_zero(alpha, ctx))
return;
fq_zech_poly_init(t, ctx);
n = A->length;
for (i = n - 2; i >= 0; i--)
{
for (j = i; j < n - 1; j++)
{
fq_zech_poly_scalar_mul_fq_zech(t, A->coeffs + j + 1, alpha, ctx);
fq_zech_poly_add(A->coeffs + j, A->coeffs + j, t, ctx);
}
}
fq_zech_poly_clear(t, ctx);
}
void fq_zech_mpoly_get_fq_zech_bpoly(
fq_zech_bpoly_t A,
const fq_zech_mpoly_t B,
slong varx,
slong vary,
const fq_zech_mpoly_ctx_t ctx)
{
slong j;
slong NB;
ulong Bexpx, Bexpy;
slong Boffx, Bshiftx, Boffy, Bshifty;
ulong mask;
FLINT_ASSERT(B->bits <= FLINT_BITS);
NB = mpoly_words_per_exp_sp(B->bits, ctx->minfo);
mpoly_gen_offset_shift_sp(&Boffx, &Bshiftx, varx, B->bits, ctx->minfo);
mpoly_gen_offset_shift_sp(&Boffy, &Bshifty, vary, B->bits, ctx->minfo);
mask = (-UWORD(1)) >> (FLINT_BITS - B->bits);
fq_zech_bpoly_zero(A, ctx->fqctx);
for (j = 0; j < B->length; j++)
{
Bexpx = ((B->exps + NB*j)[Boffx] >> Bshiftx) & mask;
Bexpy = ((B->exps + NB*j)[Boffy] >> Bshifty) & mask;
fq_zech_bpoly_set_coeff_fq_zech(A, Bexpx, Bexpy, B->coeffs + j, ctx->fqctx);
}
}
void fq_zech_mpoly_set_fq_zech_bpoly(
fq_zech_mpoly_t A,
flint_bitcnt_t Abits,
const fq_zech_bpoly_t B,
slong varx,
slong vary,
const fq_zech_mpoly_ctx_t ctx)
{
slong n = ctx->minfo->nvars;
slong i, j;
slong NA;
slong Alen;
fq_zech_struct * Acoeff;
ulong * Aexp;
slong Aalloc;
ulong * Aexps;
TMP_INIT;
FLINT_ASSERT(B->length > 0);
FLINT_ASSERT(Abits <= FLINT_BITS);
TMP_START;
Aexps = (ulong *) TMP_ALLOC(n*sizeof(ulong));
for (i = 0; i < n; i++)
Aexps[i] = 0;
NA = mpoly_words_per_exp(Abits, ctx->minfo);
fq_zech_mpoly_fit_bits(A, Abits, ctx);
A->bits = Abits;
Acoeff = A->coeffs;
Aexp = A->exps;
Aalloc = A->alloc;
Alen = 0;
for (i = 0; i < B->length; i++)
{
fq_zech_poly_struct * Bc = B->coeffs + i;
_fq_zech_mpoly_fit_length(&Acoeff, &Aexp, &Aalloc, Alen + Bc->length, NA, ctx->fqctx);
for (j = 0; j < Bc->length; j++)
{
if (fq_zech_is_zero(Bc->coeffs + j, ctx->fqctx))
continue;
Aexps[varx] = i;
Aexps[vary] = j;
fq_zech_set(Acoeff + Alen, Bc->coeffs + j, ctx->fqctx);
mpoly_set_monomial_ui(Aexp + NA*Alen, Aexps, Abits, ctx->minfo);
Alen++;
}
}
A->coeffs = Acoeff;
A->exps = Aexp;
A->alloc = Aalloc;
A->length = Alen;
TMP_END;
fq_zech_mpoly_sort_terms(A, ctx);
}