#include "ulong_extras.h"
#include "nmod_poly.h"
#include "fmpz.h"
#include "fq_zech.h"
#include "mpoly.h"
#include "nmod_mpoly_factor.h"
#include "fq_zech_mpoly_factor.h"
static void _frob_combine(
nmod_mpolyv_t Af,
fq_zech_mpolyv_t eAf,
const nmod_mpoly_ctx_t ctx,
const fq_zech_mpoly_ctx_t ectx)
{
slong i, j, N;
fq_zech_mpolyv_t tfac;
fq_zech_mpoly_t t;
nmod_mpoly_struct * s;
slong k = fq_zech_ctx_degree(ectx->fqctx);
FLINT_ASSERT(k > 1);
fq_zech_mpoly_init(t, ectx);
fq_zech_mpolyv_init(tfac, ectx);
Af->length = 0;
while (eAf->length > 0)
{
eAf->length--;
fq_zech_mpoly_swap(t, eAf->coeffs + eAf->length, ectx);
fq_zech_mpolyv_fit_length(tfac, 1, ectx);
fq_zech_mpoly_set(tfac->coeffs + 0, t, ectx);
tfac->length = 1;
for (i = 1; i < k; i++)
{
for (j = 0; j < t->length; j++)
{
fq_zech_pow_ui(t->coeffs + j, t->coeffs + j,
ctx->mod.n, ectx->fqctx);
}
for (j = 0; j < eAf->length; j++)
{
if (fq_zech_mpoly_equal(t, eAf->coeffs + j, ectx))
break;
}
if (j >= eAf->length)
continue;
fq_zech_mpolyv_fit_length(tfac, tfac->length + 1, ectx);
fq_zech_mpoly_swap(tfac->coeffs + tfac->length, eAf->coeffs + j, ectx);
tfac->length++;
eAf->length--;
fq_zech_mpoly_swap(eAf->coeffs + j, eAf->coeffs + eAf->length, ectx);
}
fq_zech_mpoly_swap(t, tfac->coeffs + 0, ectx);
for (i = 1; i < tfac->length; i++)
fq_zech_mpoly_mul(t, t, tfac->coeffs + i, ectx);
nmod_mpolyv_fit_length(Af, Af->length + 1, ctx);
s = Af->coeffs + Af->length;
Af->length++;
nmod_mpoly_fit_length_reset_bits(s, t->length, t->bits, ctx);
s->length = t->length;
N = mpoly_words_per_exp(t->bits, ectx->minfo);
mpoly_copy_monomials(s->exps, t->exps, t->length, N);
for (i = 0; i < t->length; i++)
{
nmod_poly_t asdf;
nmod_poly_init_mod(asdf, ctx->mod);
fq_zech_get_nmod_poly(asdf, t->coeffs + i, ectx->fqctx);
if (asdf->length != 1)
{
flint_throw(FLINT_ERROR, "fatal error in _frob_combine");
}
s->coeffs[i] = asdf->coeffs[0];
nmod_poly_clear(asdf);
}
}
fq_zech_mpolyv_clear(tfac, ectx);
fq_zech_mpoly_clear(t, ectx);
}
int nmod_mpoly_factor_irred_medprime_zassenhaus(
nmod_mpolyv_t Af,
const nmod_mpoly_t A,
const nmod_mpoly_ctx_t ctx,
flint_rand_t state)
{
int success;
fq_zech_mpolyv_t eAf;
fq_zech_mpoly_t eA;
fq_zech_mpoly_ctx_t ectx;
slong edeg, max_degree = n_flog(1000000, ctx->mod.n);
FLINT_ASSERT(A->length > 0);
FLINT_ASSERT(A->coeffs[0] == 1);
FLINT_ASSERT(ctx->minfo->ord == ORD_LEX);
edeg = 1 + n_clog(A->length + 1, ctx->mod.n)/2;
edeg = FLINT_MAX(2, edeg);
if (edeg > max_degree)
return 0;
fq_zech_mpoly_ctx_init_deg(ectx, ctx->minfo->nvars, ORD_LEX, ctx->mod.n, edeg);
fq_zech_mpoly_init(eA, ectx);
fq_zech_mpolyv_init(eAf, ectx);
goto have_prime;
choose_prime:
edeg++;
if (edeg > max_degree)
{
success = 0;
goto cleanup;
}
fq_zech_mpoly_ctx_change_modulus(ectx, edeg);
have_prime:
_fq_zech_mpoly_set_nmod_mpoly(eA, ectx, A, ctx);
success = fq_zech_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);
success = 1;
cleanup:
fq_zech_mpoly_clear(eA, ectx);
fq_zech_mpolyv_clear(eAf, ectx);
fq_zech_mpoly_ctx_clear(ectx);
return success;
}
static int _map_fac(
fq_zech_mpoly_factor_t eAfac,
const fq_zech_mpoly_ctx_t ectx,
const nmod_mpoly_factor_t Afac,
const nmod_mpoly_ctx_t ctx)
{
int success;
slong i, j;
fq_zech_mpoly_t t;
fq_zech_mpoly_factor_t tfac;
fq_zech_mpoly_init(t, ectx);
fq_zech_mpoly_factor_init(tfac, ectx);
fq_zech_set_ui(eAfac->constant, Afac->constant, ectx->fqctx);
eAfac->num = 0;
for (i = 0; i < Afac->num; i++)
{
_fq_zech_mpoly_set_nmod_mpoly(t, ectx, Afac->poly + i, ctx);
success = fq_zech_mpoly_factor(tfac, t, ectx);
if (!success)
goto cleanup;
FLINT_ASSERT(fq_zech_is_one(tfac->constant, ectx->fqctx));
fq_zech_mpoly_factor_fit_length(eAfac, eAfac->num + tfac->num, ectx);
for (j = 0; j < tfac->num; j++)
{
fq_zech_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_zech_mpoly_clear(t, ectx);
fq_zech_mpoly_factor_clear(tfac, ectx);
return success;
}
int nmod_mpoly_factor_irred_medprime_wang(
nmod_mpolyv_t Af,
const nmod_mpoly_t A,
const nmod_mpoly_factor_t lcAfac,
const nmod_mpoly_t lcA,
const nmod_mpoly_ctx_t ctx,
flint_rand_t state)
{
int success;
const slong n = ctx->minfo->nvars - 1;
fq_zech_mpoly_factor_t elcAfac;
fq_zech_mpolyv_t eAf;
fq_zech_mpoly_t eA, elcA;
fq_zech_mpoly_ctx_t ectx;
slong edeg, max_degree = n_flog(1000000, ctx->mod.n);
FLINT_ASSERT(A->length > 0);
FLINT_ASSERT(A->coeffs[0] == 1);
FLINT_ASSERT(ctx->minfo->ord == ORD_LEX);
edeg = 1 + n_clog(A->length + 1, ctx->mod.n)/2;
edeg = FLINT_MAX(2, edeg);
if (edeg > max_degree)
return 0;
fq_zech_mpoly_ctx_init_deg(ectx, n + 1, ORD_LEX, ctx->mod.n, edeg);
fq_zech_mpoly_init(eA, ectx);
fq_zech_mpolyv_init(eAf, ectx);
fq_zech_mpoly_init(elcA, ectx);
fq_zech_mpoly_factor_init(elcAfac, ectx);
fq_zech_mpoly_factor_fit_length(elcAfac, lcAfac->num, ectx);
elcAfac->num = lcAfac->num;
goto have_prime;
choose_prime:
edeg++;
if (edeg > max_degree)
{
success = 0;
goto cleanup;
}
fq_zech_mpoly_ctx_change_modulus(ectx, edeg);
have_prime:
_fq_zech_mpoly_set_nmod_mpoly(eA, ectx, A, ctx);
_fq_zech_mpoly_set_nmod_mpoly(elcA, ectx, lcA, ctx);
_map_fac(elcAfac, ectx, lcAfac, ctx);
success = fq_zech_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);
success = 1;
cleanup:
fq_zech_mpoly_clear(eA, ectx);
fq_zech_mpolyv_clear(eAf, ectx);
fq_zech_mpoly_clear(elcA, ectx);
fq_zech_mpoly_factor_clear(elcAfac, ectx);
fq_zech_mpoly_ctx_clear(ectx);
return success;
}
int nmod_mpoly_factor_irred_medprime_zippel(
nmod_mpolyv_t Af,
const nmod_mpoly_t A,
const nmod_mpoly_factor_t lcAfac,
const nmod_mpoly_t lcA,
const nmod_mpoly_ctx_t ctx,
flint_rand_t state)
{
int success;
const slong n = ctx->minfo->nvars - 1;
fq_zech_mpoly_factor_t elcAfac;
fq_zech_mpolyv_t eAf;
fq_zech_mpoly_t eA, elcA;
fq_zech_mpoly_ctx_t ectx;
slong edeg, max_degree = n_flog(1000000, ctx->mod.n);
FLINT_ASSERT(A->length > 0);
FLINT_ASSERT(A->coeffs[0] == 1);
FLINT_ASSERT(ctx->minfo->ord == ORD_LEX);
edeg = 1 + n_clog(A->length + 1, ctx->mod.n);
edeg = FLINT_MAX(2, edeg);
if (edeg > max_degree)
return 0;
fq_zech_mpoly_ctx_init_deg(ectx, n + 1, ORD_LEX, ctx->mod.n, edeg);
fq_zech_mpoly_init(eA, ectx);
fq_zech_mpolyv_init(eAf, ectx);
fq_zech_mpoly_init(elcA, ectx);
fq_zech_mpoly_factor_init(elcAfac, ectx);
fq_zech_mpoly_factor_fit_length(elcAfac, lcAfac->num, ectx);
elcAfac->num = lcAfac->num;
goto have_prime;
choose_prime:
edeg++;
if (edeg > max_degree)
{
success = 0;
goto cleanup;
}
fq_zech_mpoly_ctx_change_modulus(ectx, edeg);
have_prime:
_fq_zech_mpoly_set_nmod_mpoly(eA, ectx, A, ctx);
_fq_zech_mpoly_set_nmod_mpoly(elcA, ectx, lcA, ctx);
_map_fac(elcAfac, ectx, lcAfac, ctx);
success = fq_zech_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);
success = 1;
cleanup:
fq_zech_mpoly_clear(eA, ectx);
fq_zech_mpolyv_clear(eAf, ectx);
fq_zech_mpoly_clear(elcA, ectx);
fq_zech_mpoly_factor_clear(elcAfac, ectx);
fq_zech_mpoly_ctx_clear(ectx);
return success;
}