#include "ulong_extras.h"
#include "fq_nmod.h"
#include "n_poly.h"
#include "fq_nmod_mpoly_factor.h"
int fq_nmod_mpoly_factor_irred_smprime_wang(
fq_nmod_mpolyv_t fac,
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)
{
slong d = fq_nmod_ctx_degree(ctx->fqctx);
int success;
int alphas_tries_remaining, alphabetas_tries_remaining, alphabetas_length;
const slong n = ctx->minfo->nvars - 1;
slong i, j, k, r;
fq_nmod_struct * alpha;
n_poly_struct * alphabetas;
fq_nmod_mpoly_struct * Aevals;
slong * degs, * degeval;
fq_nmod_mpolyv_t tfac;
fq_nmod_mpoly_t t, Acopy;
fq_nmod_mpoly_struct * newA;
n_poly_t Abfc;
n_bpoly_t Ab;
n_tpoly_t Abfp;
fq_nmod_mpoly_t m, mpow;
fq_nmod_mpolyv_t new_lcs, lc_divs;
FLINT_ASSERT(n > 1);
FLINT_ASSERT(A->length > 1);
FLINT_ASSERT(_n_fq_is_one(A->coeffs + d*0, d));
FLINT_ASSERT(A->bits <= FLINT_BITS);
fq_nmod_mpoly_init(Acopy, ctx);
fq_nmod_mpoly_init(m, ctx);
fq_nmod_mpoly_init(mpow, ctx);
fq_nmod_mpolyv_init(new_lcs, ctx);
fq_nmod_mpolyv_init(lc_divs, ctx);
n_poly_init(Abfc);
n_tpoly_init(Abfp);
n_bpoly_init(Ab);
degs = FLINT_ARRAY_ALLOC(n + 1, slong);
degeval = FLINT_ARRAY_ALLOC(n + 1, slong);
alpha = FLINT_ARRAY_ALLOC(n, fq_nmod_struct);
alphabetas = FLINT_ARRAY_ALLOC(n, n_poly_struct);
Aevals = FLINT_ARRAY_ALLOC(n, fq_nmod_mpoly_struct);
for (i = 0; i < n; i++)
{
fq_nmod_init(alpha + i, ctx->fqctx);
n_poly_init(alphabetas + i);
fq_nmod_mpoly_init(Aevals + i, ctx);
}
fq_nmod_mpolyv_init(tfac, ctx);
fq_nmod_mpoly_init(t, ctx);
alphabetas_length = 2;
alphas_tries_remaining = 10;
fq_nmod_mpoly_degrees_si(degs, A, ctx);
next_alpha:
if (--alphas_tries_remaining < 0)
{
success = 0;
goto cleanup;
}
for (i = 0; i < n; i++)
{
fq_nmod_rand(alpha + i, state, ctx->fqctx);
}
for (i = n - 1; i >= 0; i--)
{
fq_nmod_mpoly_evaluate_one_fq_nmod(Aevals + i,
i == n - 1 ? A : Aevals + i + 1, i + 1, alpha + i, ctx);
fq_nmod_mpoly_degrees_si(degeval, Aevals + i, ctx);
for (j = 0; j <= i; j++)
if (degeval[j] != degs[j])
goto next_alpha;
}
fq_nmod_mpoly_derivative(t, Aevals + 0, 0, ctx);
fq_nmod_mpoly_gcd(t, t, Aevals + 0, ctx);
if (!fq_nmod_mpoly_is_one(t, ctx))
goto next_alpha;
alphabetas_tries_remaining = 2 + alphabetas_length;
next_alphabetas:
if (--alphabetas_tries_remaining < 0)
{
if (++alphabetas_length > 10)
{
success = 0;
goto cleanup;
}
goto next_alpha;
}
for (i = 0; i < n; i++)
{
n_poly_fit_length(alphabetas + i, d*alphabetas_length);
n_fq_set_fq_nmod(alphabetas[i].coeffs + d*0, alpha + i, ctx->fqctx);
for (j = d; j < d*alphabetas_length; j++)
alphabetas[i].coeffs[j] = n_urandint(state, ctx->fqctx->mod.n);
alphabetas[i].length = alphabetas_length;
_n_fq_poly_normalise(alphabetas + i, d);
}
_fq_nmod_mpoly_eval_rest_to_n_fq_bpoly(Ab, A, alphabetas, ctx);
success = n_fq_bpoly_factor_smprime(Abfc, Abfp, Ab, 0, ctx->fqctx);
if (!success)
{
FLINT_ASSERT(0 && "this should not happen");
goto next_alpha;
}
r = Abfp->length;
if (r < 2)
{
fq_nmod_mpolyv_fit_length(fac, 1, ctx);
fac->length = 1;
fq_nmod_mpoly_set(fac->coeffs + 0, A, ctx);
success = 1;
goto cleanup;
}
fq_nmod_mpolyv_fit_length(lc_divs, r, ctx);
lc_divs->length = r;
if (lcAfac->num > 0)
{
success = fq_nmod_mpoly_factor_lcc_wang(lc_divs->coeffs, lcAfac,
Abfc, Abfp->coeffs, r, alphabetas, ctx);
if (!success)
goto next_alphabetas;
}
else
{
for (i = 0; i < r; i++)
fq_nmod_mpoly_one(lc_divs->coeffs + i, ctx);
}
success = fq_nmod_mpoly_divides(m, lcA, lc_divs->coeffs + 0, ctx);
FLINT_ASSERT(success);
for (i = 1; i < r; i++)
{
success = fq_nmod_mpoly_divides(m, m, lc_divs->coeffs + i, ctx);
FLINT_ASSERT(success);
}
fq_nmod_mpoly_pow_ui(mpow, m, r - 1, ctx);
if (fq_nmod_mpoly_is_one(mpow, ctx))
{
newA = (fq_nmod_mpoly_struct *) A;
}
else
{
newA = Acopy;
fq_nmod_mpoly_mul(newA, A, mpow, ctx);
}
if (newA->bits > FLINT_BITS)
{
success = 0;
goto cleanup;
}
fq_nmod_mpoly_degrees_si(degs, newA, ctx);
fq_nmod_mpoly_set(t, mpow, ctx);
for (i = n - 1; i >= 0; i--)
{
fq_nmod_mpoly_evaluate_one_fq_nmod(t, mpow, i + 1, alpha + i, ctx);
fq_nmod_mpoly_swap(t, mpow, ctx);
fq_nmod_mpoly_mul(Aevals + i, Aevals + i, mpow, ctx);
}
fq_nmod_mpolyv_fit_length(new_lcs, (n + 1)*r, ctx);
i = n;
for (j = 0; j < r; j++)
{
fq_nmod_mpoly_mul(new_lcs->coeffs + i*r + j, lc_divs->coeffs + j, m, ctx);
}
for (i = n - 1; i >= 0; i--)
{
for (j = 0; j < r; j++)
{
fq_nmod_mpoly_evaluate_one_fq_nmod(new_lcs->coeffs + i*r + j,
new_lcs->coeffs + (i + 1)*r + j, i + 1, alpha + i, ctx);
}
}
fq_nmod_mpolyv_fit_length(fac, r, ctx);
fac->length = r;
for (i = 0; i < r; i++)
{
fq_nmod_t q, qt;
fq_nmod_init(q, ctx->fqctx);
fq_nmod_init(qt, ctx->fqctx);
FLINT_ASSERT(fq_nmod_mpoly_is_fq_nmod(new_lcs->coeffs + 0*r + i, ctx));
FLINT_ASSERT(fq_nmod_mpoly_length(new_lcs->coeffs + 0*r + i, ctx) == 1);
_fq_nmod_mpoly_set_n_fq_bpoly_gen1_zero(fac->coeffs + i, newA->bits, Abfp->coeffs + i, 0, ctx);
FLINT_ASSERT(fac->coeffs[i].length > 0);
n_fq_get_fq_nmod(qt, fac->coeffs[i].coeffs + d*0, ctx->fqctx);
fq_nmod_inv(q, qt, ctx->fqctx);
n_fq_get_fq_nmod(qt, new_lcs->coeffs[0*r + i].coeffs + 0, ctx->fqctx);
fq_nmod_mul(q, q, qt, ctx->fqctx);
fq_nmod_mpoly_scalar_mul_fq_nmod(fac->coeffs + i, fac->coeffs + i, q, ctx);
fq_nmod_clear(q, ctx->fqctx);
fq_nmod_clear(qt, ctx->fqctx);
}
fq_nmod_mpolyv_fit_length(tfac, r, ctx);
tfac->length = r;
for (k = 1; k <= n; k++)
{
for (i = 0; i < r; i++)
{
_fq_nmod_mpoly_set_lead0(tfac->coeffs + i, fac->coeffs + i,
new_lcs->coeffs + k*r + i, ctx);
}
success = fq_nmod_mpoly_hlift(k, tfac->coeffs, r, alpha,
k < n ? Aevals + k : newA, degs, ctx);
if (!success)
goto next_alphabetas;
fq_nmod_mpolyv_swap(tfac, fac, ctx);
}
if (!fq_nmod_mpoly_is_fq_nmod(m, ctx))
{
for (i = 0; i < r; i++)
{
FLINT_ASSERT(fac->coeffs[i].bits <= FLINT_BITS);
if (!fq_nmod_mpolyl_content(t, fac->coeffs + i, 1, ctx))
{
success = -1;
goto cleanup;
}
success = fq_nmod_mpoly_divides(fac->coeffs + i, fac->coeffs + i, t, ctx);
FLINT_ASSERT(success);
}
}
for (i = 0; i < r; i++)
fq_nmod_mpoly_make_monic(fac->coeffs + i, fac->coeffs + i, ctx);
success = 1;
cleanup:
fq_nmod_mpolyv_clear(new_lcs, ctx);
fq_nmod_mpolyv_clear(lc_divs, ctx);
n_bpoly_clear(Ab);
n_poly_clear(Abfc);
n_tpoly_clear(Abfp);
for (i = 0; i < n; i++)
{
fq_nmod_clear(alpha + i, ctx->fqctx);
fq_nmod_mpoly_clear(Aevals + i, ctx);
n_poly_clear(alphabetas + i);
}
flint_free(alphabetas);
flint_free(alpha);
flint_free(Aevals);
flint_free(degs);
flint_free(degeval);
fq_nmod_mpolyv_clear(tfac, ctx);
fq_nmod_mpoly_clear(t, ctx);
fq_nmod_mpoly_clear(Acopy, ctx);
fq_nmod_mpoly_clear(m, ctx);
fq_nmod_mpoly_clear(mpow, ctx);
#if FLINT_WANT_ASSERT
if (success)
{
fq_nmod_mpoly_t prod;
fq_nmod_mpoly_init(prod, ctx);
fq_nmod_mpoly_one(prod, ctx);
for (i = 0; i < fac->length; i++)
fq_nmod_mpoly_mul(prod, prod, fac->coeffs + i, ctx);
FLINT_ASSERT(fq_nmod_mpoly_equal(prod, A, ctx));
fq_nmod_mpoly_clear(prod, ctx);
}
#endif
return success;
}