#include "fmpz.h"
#include "fq_nmod.h"
#include "n_poly.h"
#include "mpoly.h"
#include "fq_nmod_mpoly_factor.h"
static void _map_poly(
fq_nmod_mpoly_t eA,
const fq_nmod_mpoly_ctx_t ectx,
const fq_nmod_mpoly_t A,
const fq_nmod_mpoly_ctx_t ctx,
const bad_fq_nmod_embed_t emb)
{
slong smd = fq_nmod_ctx_degree(ctx->fqctx);
slong lgd = fq_nmod_ctx_degree(ectx->fqctx);
slong N = mpoly_words_per_exp(A->bits, ectx->minfo);
slong i;
FLINT_ASSERT(eA != A);
fq_nmod_mpoly_fit_length_reset_bits(eA, A->length, A->bits, ectx);
mpoly_copy_monomials(eA->exps, A->exps, A->length, N);
for (i = 0; i < A->length; i++)
bad_n_fq_embed_sm_elem_to_lg(eA->coeffs + lgd*i, A->coeffs + smd*i, emb);
eA->length = A->length;
}
static void _frob_combine(
fq_nmod_mpolyv_t Af,
fq_nmod_mpolyv_t eAf,
const fq_nmod_mpoly_ctx_t ctx,
const fq_nmod_mpoly_ctx_t ectx,
const bad_fq_nmod_embed_t emb)
{
slong smd = fq_nmod_ctx_degree(ctx->fqctx);
slong lgd = fq_nmod_ctx_degree(ectx->fqctx);
slong i, j;
fq_nmod_mpolyv_t tfac;
fq_nmod_mpoly_t t;
fq_nmod_mpoly_struct * s;
n_poly_t c;
slong k = lgd/smd;
fmpz_t q;
FLINT_ASSERT(k > 1);
fmpz_init(q);
fq_nmod_mpoly_init(t, ectx);
fq_nmod_mpolyv_init(tfac, ectx);
n_poly_init(c);
fmpz_ui_pow_ui(q, fq_nmod_ctx_prime(ctx->fqctx), fq_nmod_ctx_degree(ctx->fqctx));
Af->length = 0;
while (eAf->length > 0)
{
eAf->length--;
fq_nmod_mpoly_swap(t, eAf->coeffs + eAf->length, ectx);
fq_nmod_mpolyv_fit_length(tfac, 1, ectx);
fq_nmod_mpoly_set(tfac->coeffs + 0, t, ectx);
tfac->length = 1;
for (i = 1; i < k; i++)
{
for (j = 0; j < t->length; j++)
n_fq_pow_fmpz(t->coeffs + lgd*j, t->coeffs + lgd*j, q, ectx->fqctx);
for (j = 0; j < eAf->length; j++)
{
if (fq_nmod_mpoly_equal(t, eAf->coeffs + j, ectx))
break;
}
if (j >= eAf->length)
continue;
fq_nmod_mpolyv_fit_length(tfac, tfac->length + 1, ectx);
fq_nmod_mpoly_swap(tfac->coeffs + tfac->length, eAf->coeffs + j, ectx);
tfac->length++;
eAf->length--;
fq_nmod_mpoly_swap(eAf->coeffs + j, eAf->coeffs + eAf->length, ectx);
}
fq_nmod_mpoly_swap(t, tfac->coeffs + 0, ectx);
for (i = 1; i < tfac->length; i++)
fq_nmod_mpoly_mul(t, t, tfac->coeffs + i, ectx);
fq_nmod_mpolyv_fit_length(Af, Af->length + 1, ctx);
s = Af->coeffs + Af->length;
Af->length++;
fq_nmod_mpoly_fit_length_reset_bits(s, t->length, t->bits, ctx);
s->length = t->length;
mpoly_copy_monomials(s->exps, t->exps,
mpoly_words_per_exp(t->bits, ectx->minfo), t->length);
for (i = 0; i < t->length; i++)
{
bad_n_fq_embed_lg_to_sm(c, t->coeffs + lgd*i, emb);
if (c->length != 1)
{
flint_throw(FLINT_ERROR, "fatal error in _frob_combine");
}
_n_fq_set(s->coeffs + smd*i, c->coeffs + smd*0, smd);
}
}
n_poly_clear(c);
fq_nmod_mpolyv_clear(tfac, ectx);
fq_nmod_mpoly_clear(t, ectx);
fmpz_clear(q);
}
int fq_nmod_mpoly_factor_irred_lgprime_zassenhaus(
fq_nmod_mpolyv_t Af,
const fq_nmod_mpoly_t A,
const fq_nmod_mpoly_ctx_t ctx,
flint_rand_t state)
{
int success;
fq_nmod_mpolyv_t eAf;
fq_nmod_mpoly_t eA;
bad_fq_nmod_mpoly_embed_chooser_t embc;
bad_fq_nmod_embed_struct * cur_emb;
fq_nmod_mpoly_ctx_t ectx;
cur_emb = bad_fq_nmod_mpoly_embed_chooser_init(embc, ectx, ctx, state);
fq_nmod_mpoly_init(eA, ectx);
fq_nmod_mpolyv_init(eAf, ectx);
goto have_prime;
choose_prime:
cur_emb = bad_fq_nmod_mpoly_embed_chooser_next(embc, ectx, ctx, state);
if (cur_emb == NULL)
{
success = 0;
goto cleanup;
}
have_prime:
_map_poly(eA, ectx, A, ctx, cur_emb);
success = fq_nmod_mpoly_factor_irred_smprime_zassenhaus(eAf, eA, ectx, state);
if (success == 0)
goto choose_prime;
if (success < 0)
goto cleanup;
_frob_combine(Af, eAf, ctx, ectx, cur_emb);
success = 1;
cleanup:
fq_nmod_mpoly_clear(eA, ectx);
fq_nmod_mpolyv_clear(eAf, ectx);
bad_fq_nmod_mpoly_embed_chooser_clear(embc, ectx, ctx, state);
return success;
}
static int _map_fac(
fq_nmod_mpoly_factor_t eAfac,
const fq_nmod_mpoly_ctx_t ectx,
const fq_nmod_mpoly_factor_t Afac,
const fq_nmod_mpoly_ctx_t ctx,
const bad_fq_nmod_embed_t emb)
{
int success;
slong i, j;
fq_nmod_mpoly_t t;
fq_nmod_mpoly_factor_t tfac;
fq_nmod_mpoly_init(t, ectx);
fq_nmod_mpoly_factor_init(tfac, ectx);
bad_fq_nmod_embed_sm_elem_to_lg(eAfac->constant, Afac->constant, emb);
eAfac->num = 0;
for (i = 0; i < Afac->num; i++)
{
_map_poly(t, ectx, Afac->poly + i, ctx, emb);
success = fq_nmod_mpoly_factor(tfac, t, ectx);
if (!success)
goto cleanup;
FLINT_ASSERT(fq_nmod_is_one(tfac->constant, ectx->fqctx));
fq_nmod_mpoly_factor_fit_length(eAfac, eAfac->num + tfac->num, ectx);
for (j = 0; j < tfac->num; j++)
{
fq_nmod_mpoly_swap(eAfac->poly + eAfac->num, tfac->poly + j, ectx);
fmpz_mul(eAfac->exp + eAfac->num, tfac->exp + j, Afac->exp + i);
eAfac->num++;
}
}
success = 1;
cleanup:
fq_nmod_mpoly_clear(t, ectx);
fq_nmod_mpoly_factor_clear(tfac, ectx);
return success;
}
int fq_nmod_mpoly_factor_irred_lgprime_wang(
fq_nmod_mpolyv_t Af,
const fq_nmod_mpoly_t A,
const fq_nmod_mpoly_factor_t lcAfac,
const fq_nmod_mpoly_t lcA,
const fq_nmod_mpoly_ctx_t ctx,
flint_rand_t state)
{
int success;
fq_nmod_mpoly_factor_t elcAfac;
fq_nmod_mpolyv_t eAf;
fq_nmod_mpoly_t eA, elcA;
bad_fq_nmod_mpoly_embed_chooser_t embc;
bad_fq_nmod_embed_struct * cur_emb;
fq_nmod_mpoly_ctx_t ectx;
FLINT_ASSERT(A->length > 0);
FLINT_ASSERT(_n_fq_is_one(A->coeffs + 0, fq_nmod_ctx_degree(ctx->fqctx)));
FLINT_ASSERT(ctx->minfo->ord == ORD_LEX);
cur_emb = bad_fq_nmod_mpoly_embed_chooser_init(embc, ectx, ctx, state);
fq_nmod_mpoly_init(eA, ectx);
fq_nmod_mpolyv_init(eAf, ectx);
fq_nmod_mpoly_init(elcA, ectx);
fq_nmod_mpoly_factor_init(elcAfac, ectx);
goto have_prime;
choose_prime:
cur_emb = bad_fq_nmod_mpoly_embed_chooser_next(embc, ectx, ctx, state);
if (cur_emb == NULL)
{
success = 0;
goto cleanup;
}
have_prime:
_map_poly(eA, ectx, A, ctx, cur_emb);
_map_poly(elcA, ectx, lcA, ctx, cur_emb);
_map_fac(elcAfac, ectx, lcAfac, ctx, cur_emb);
success = fq_nmod_mpoly_factor_irred_smprime_wang(eAf, eA, elcAfac, elcA, ectx, state);
if (success == 0)
goto choose_prime;
if (success < 0)
goto cleanup;
_frob_combine(Af, eAf, ctx, ectx, cur_emb);
success = 1;
cleanup:
fq_nmod_mpoly_clear(eA, ectx);
fq_nmod_mpolyv_clear(eAf, ectx);
fq_nmod_mpoly_clear(elcA, ectx);
fq_nmod_mpoly_factor_clear(elcAfac, ectx);
bad_fq_nmod_mpoly_embed_chooser_clear(embc, ectx, ctx, state);
return success;
}
int fq_nmod_mpoly_factor_irred_lgprime_zippel(
fq_nmod_mpolyv_t Af,
const fq_nmod_mpoly_t A,
const fq_nmod_mpoly_factor_t lcAfac,
const fq_nmod_mpoly_t lcA,
const fq_nmod_mpoly_ctx_t ctx,
flint_rand_t state)
{
int success;
fq_nmod_mpoly_factor_t elcAfac;
fq_nmod_mpolyv_t eAf;
fq_nmod_mpoly_t eA, elcA;
bad_fq_nmod_mpoly_embed_chooser_t embc;
bad_fq_nmod_embed_struct * cur_emb;
fq_nmod_mpoly_ctx_t ectx;
FLINT_ASSERT(A->length > 0);
FLINT_ASSERT(_n_fq_is_one(A->coeffs + 0, fq_nmod_ctx_degree(ctx->fqctx)));
FLINT_ASSERT(ctx->minfo->ord == ORD_LEX);
cur_emb = bad_fq_nmod_mpoly_embed_chooser_init(embc, ectx, ctx, state);
fq_nmod_mpoly_init(eA, ectx);
fq_nmod_mpolyv_init(eAf, ectx);
fq_nmod_mpoly_init(elcA, ectx);
fq_nmod_mpoly_factor_init(elcAfac, ectx);
goto have_prime;
choose_prime:
cur_emb = bad_fq_nmod_mpoly_embed_chooser_next(embc, ectx, ctx, state);
if (cur_emb == NULL)
{
success = 0;
goto cleanup;
}
have_prime:
_map_poly(eA, ectx, A, ctx, cur_emb);
_map_poly(elcA, ectx, lcA, ctx, cur_emb);
_map_fac(elcAfac, ectx, lcAfac, ctx, cur_emb);
success = fq_nmod_mpoly_factor_irred_smprime_zippel(eAf, eA, elcAfac, elcA, ectx, state);
if (success == 0)
goto choose_prime;
if (success < 0)
goto cleanup;
_frob_combine(Af, eAf, ctx, ectx, cur_emb);
success = 1;
cleanup:
fq_nmod_mpoly_clear(eA, ectx);
fq_nmod_mpolyv_clear(eAf, ectx);
fq_nmod_mpoly_clear(elcA, ectx);
fq_nmod_mpoly_factor_clear(elcAfac, ectx);
bad_fq_nmod_mpoly_embed_chooser_clear(embc, ectx, ctx, state);
return success;
}