#include "fmpz_vec.h"
#include "mpoly.h"
#include "fmpz_mod_mpoly.h"
void fmpz_mod_mpolyl_lead_coeff(
fmpz_mod_mpoly_t c,
const fmpz_mod_mpoly_t A,
slong num_vars,
const fmpz_mod_mpoly_ctx_t ctx)
{
slong i, j, off, shift;
ulong mask, first_mask;
slong N = mpoly_words_per_exp_sp(A->bits, ctx->minfo);
ulong * Aexps = A->exps;
ulong * cexps;
slong Alen = A->length;
FLINT_ASSERT(c != A);
FLINT_ASSERT(A->length > 0);
FLINT_ASSERT(ctx->minfo->ord == ORD_LEX);
FLINT_ASSERT(0 < num_vars && num_vars < ctx->minfo->nvars);
mpoly_gen_offset_shift_sp(&off, &shift, num_vars - 1, A->bits, ctx->minfo);
mask = (-UWORD(1)) << shift;
i = 0;
first_mask = (Aexps + N*i)[off] & mask;
for (i = 1; i < Alen; i++)
{
if (((Aexps + N*i)[off] & mask) != first_mask)
goto break_outer;
for (j = off + 1; j < N; j++)
if ((Aexps + N*(i - 1))[j] != (Aexps + N*i)[j])
goto break_outer;
}
break_outer:
fmpz_mod_mpoly_fit_length_reset_bits(c, i, A->bits, ctx);
c->length = i;
cexps = c->exps;
_fmpz_vec_set(c->coeffs, A->coeffs, c->length);
mask = ~mask;
for (i = 0; i < c->length; i++)
{
for (j = 0; j < off; j++)
(cexps + N*i)[j] = (Aexps + N*i)[j];
(cexps + N*i)[off] = mask & (Aexps + N*i)[off];
for (j = off + 1; j < N; j++)
(cexps + N*i)[j] = 0;
}
}