#include "fq_nmod.h"
#include "n_poly.h"
#include "mpoly.h"
#include "fq_nmod_mpoly_factor.h"
#if FLINT_WANT_ASSERT
static void n_fq_polyu_set_n_fq_polyun(
n_polyu_t A,
const n_polyun_t B,
const fq_nmod_ctx_t ctx)
{
slong d = fq_nmod_ctx_degree(ctx);
slong i, j;
A->length = 0;
for (i = 0; i < B->length; i++)
{
for (j = B->coeffs[i].length - 1; j >= 0; j--)
{
if (_n_fq_is_zero(B->coeffs[i].coeffs + d*j, d))
continue;
n_polyu_fit_length(A, d*(A->length + 1));
_n_fq_set(A->coeffs + d*A->length, B->coeffs[i].coeffs + d*j, d);
A->exps[A->length] = B->exps[i] + j;
A->length++;
}
}
}
static void n_fq_polyu_sort_terms(
n_polyu_t A,
const fq_nmod_ctx_t ctx)
{
slong d = fq_nmod_ctx_degree(ctx);
slong i, j;
for (i = 1; i < A->length; i++)
for (j = i; j > 0 && A->exps[j - 1] < A->exps[j]; j--)
{
FLINT_SWAP(ulong, A->exps[j - 1], A->exps[j]);
_n_fq_swap(A->coeffs + d*(j - 1), A->coeffs + d*(j), d);
}
return;
}
static void n_fq_polyu_combine_like_terms(
n_fq_polyu_t A,
const fq_nmod_ctx_t ctx)
{
slong d = fq_nmod_ctx_degree(ctx);
nmod_t mod = fq_nmod_ctx_mod(ctx);
slong in, out;
out = -1;
for (in = 0; in < A->length; in++)
{
FLINT_ASSERT(in > out);
if (out >= 0 && A->exps[out] == A->exps[in])
{
_n_fq_add(A->coeffs + d*out, A->coeffs + d*out, A->coeffs + d*in, d, mod);
}
else
{
if (out < 0 || !_n_fq_is_zero(A->coeffs + d*out, d))
out++;
if (out != in)
{
A->exps[out] = A->exps[in];
_n_fq_set(A->coeffs + d*out, A->coeffs + d*in, d);
}
}
}
if (out < 0 || !_n_fq_is_zero(A->coeffs + d*out, d))
out++;
A->length = out;
}
static int n_fq_polyu_equal(
n_polyu_t A,
n_polyu_t B,
const fq_nmod_ctx_t ctx)
{
slong d = fq_nmod_ctx_degree(ctx);
slong i;
if (A->length != B->length)
return 0;
for (i = 0; i < A->length; i++)
{
if (!_n_fq_equal(A->coeffs + d*i, B->coeffs + d*i, d))
return 0;
if (A->exps[i] != B->exps[i])
return 0;
}
return 1;
}
static void fq_nmod_polyu_mul(
n_polyu_t A,
const n_polyu_t B,
const n_polyu_t C,
const fq_nmod_ctx_t ctx)
{
slong d = fq_nmod_ctx_degree(ctx);
slong Ai, Bi, Ci;
FLINT_ASSERT(A != B);
FLINT_ASSERT(A != C);
n_polyu_fit_length(A, d*B->length*C->length);
Ai = 0;
for (Bi = 0; Bi < B->length; Bi++)
for (Ci = 0; Ci < C->length; Ci++)
{
A->exps[Ai] = B->exps[Bi] + C->exps[Ci];
n_fq_mul(A->coeffs + d*Ai, B->coeffs + d*Bi, C->coeffs + d*Ci, ctx);
Ai++;
}
A->length = Ai;
n_fq_polyu_sort_terms(A, ctx);
n_fq_polyu_combine_like_terms(A, ctx);
}
#endif
static void n_fq_poly_fill_power(
n_fq_poly_t alphapow,
slong e,
const fq_nmod_ctx_t ctx,
ulong * tmp)
{
if (e + 1 > alphapow->length)
{
slong d = fq_nmod_ctx_degree(ctx);
slong k;
slong oldlength = alphapow->length;
FLINT_ASSERT(2 <= oldlength);
n_poly_fit_length(alphapow, d*(e + 1));
for (k = oldlength; k <= e; k++)
{
_n_fq_mul(alphapow->coeffs + d*k, alphapow->coeffs + d*(k - 1),
alphapow->coeffs + d*1, ctx, tmp);
}
alphapow->length = e + 1;
}
}
static void fq_nmod_polyu3_interp_reduce_bpoly(
n_bpoly_t Ap,
const n_polyu_t A,
n_fq_poly_t alphapow,
const fq_nmod_ctx_t ctx)
{
slong d = fq_nmod_ctx_degree(ctx);
slong i;
slong cur0, cur1, e0, e1, e2;
ulong * tmp, * t;
TMP_INIT;
TMP_START;
tmp = (ulong *) TMP_ALLOC(d*(1 + N_FQ_MUL_ITCH)*sizeof(ulong));
t = tmp + d*N_FQ_MUL_ITCH;
n_bpoly_zero(Ap);
FLINT_ASSERT(A->length > 0);
i = 0;
cur0 = extract_exp(A->exps[i], 2, 3);
cur1 = extract_exp(A->exps[i], 1, 3);
e2 = extract_exp(A->exps[i], 0, 3);
n_fq_poly_fill_power(alphapow, e2, ctx, tmp);
_n_fq_mul(t, A->coeffs + d*i, alphapow->coeffs + d*e2, ctx, tmp);
for (i = 1; i < A->length; i++)
{
e0 = extract_exp(A->exps[i], 2, 3);
e1 = extract_exp(A->exps[i], 1, 3);
e2 = extract_exp(A->exps[i], 0, 3);
FLINT_ASSERT(e0 <= cur0);
if (e0 < cur0 || e1 < cur1)
{
n_fq_bpoly_set_coeff_n_fq(Ap, cur0, cur1, t, ctx);
_n_fq_zero(t, d);
}
else
{
FLINT_ASSERT(e0 == cur0);
FLINT_ASSERT(e1 == cur1);
}
cur0 = e0;
cur1 = e1;
n_fq_poly_fill_power(alphapow, e2, ctx, tmp);
_n_fq_addmul(t, t, A->coeffs + d*i, alphapow->coeffs + d*e2, ctx, tmp);
}
n_fq_bpoly_set_coeff_n_fq(Ap, cur0, cur1, t, ctx);
TMP_END;
}
static void fq_nmod_polyu3n_interp_lift_sm_bpoly(
slong * lastdeg,
n_polyun_t T,
const n_bpoly_t A,
const fq_nmod_ctx_t ctx)
{
slong d = fq_nmod_ctx_degree(ctx);
slong lastlength = 0;
slong Ti;
slong Ai, j;
Ti = 0;
for (Ai = A->length - 1; Ai >= 0; Ai--)
{
n_poly_struct * Ac = A->coeffs + Ai;
for (j = Ac->length - 1; j >= 0; j--)
{
if (_n_fq_is_zero(Ac->coeffs + d*j, d))
continue;
n_polyun_fit_length(T, Ti + 1);
T->exps[Ti] = pack_exp3(Ai, j, 0);
n_fq_poly_set_n_fq(T->coeffs + Ti, Ac->coeffs + d*j, ctx);
Ti++;
lastlength = 1;
}
}
T->length = Ti;
*lastdeg = lastlength - 1;
return;
}
static int fq_nmod_polyu3n_interp_crt_sm_bpoly(
slong * lastdeg,
n_polyun_t F,
n_polyun_t T,
const n_bpoly_t A,
const n_poly_t modulus,
n_fq_poly_t alphapow,
const fq_nmod_ctx_t ctx)
{
slong d = fq_nmod_ctx_degree(ctx);
nmod_t mod = fq_nmod_ctx_mod(ctx);
int changed = 0;
slong lastlength = 0;
n_poly_struct * Tcoeffs;
ulong * Texps;
slong Ti;
n_poly_struct * Fcoeffs = F->coeffs;
ulong * Fexps = F->exps;
slong Flen = F->length;
slong Fi;
const n_poly_struct * Acoeffs = A->coeffs;
slong Ai, ai;
ulong * v = FLINT_ARRAY_ALLOC(d, ulong);
FLINT_ASSERT(n_fq_bpoly_is_canonical(A, ctx));
FLINT_ASSERT(n_polyun_fq_is_canonical(F, ctx));
n_polyun_fit_length(T, FLINT_MAX(Flen, A->length));
Tcoeffs = T->coeffs;
Texps = T->exps;
Ti = Fi = 0;
Ai = A->length - 1;
ai = (Ai < 0) ? 0 : n_poly_degree(A->coeffs + Ai);
while (Fi < Flen || Ai >= 0)
{
if (Ti >= T->alloc)
{
slong extra = Flen - Fi;
extra = FLINT_MAX(extra, Ai);
n_polyun_fit_length(T, Ti + extra + 1);
Tcoeffs = T->coeffs;
Texps = T->exps;
}
FLINT_ASSERT(Fi >= Flen || Fcoeffs[Fi].length > 0);
FLINT_ASSERT(Ai < 0 || !_n_fq_is_zero(Acoeffs[Ai].coeffs + d*ai, d));
if (Fi < Flen && Ai >= 0 && Fexps[Fi] == pack_exp3(Ai, ai, 0))
{
n_fq_poly_eval_pow(v, Fcoeffs + Fi, alphapow, ctx);
_n_fq_sub(v, Acoeffs[Ai].coeffs + d*ai, v, d, mod);
if (!_n_fq_is_zero(v, d))
{
changed = 1;
n_fq_poly_scalar_addmul_n_fq(Tcoeffs + Ti,
Fcoeffs + Fi, modulus, v, ctx);
}
else
{
n_fq_poly_set(Tcoeffs + Ti, Fcoeffs + Fi, ctx);
}
Texps[Ti] = Fexps[Fi];
Fi++;
do {
ai--;
} while (ai >= 0 && _n_fq_is_zero(Acoeffs[Ai].coeffs + d*ai, d));
if (ai < 0)
{
do {
Ai--;
} while (Ai >= 0 && Acoeffs[Ai].length == 0);
if (Ai >= 0)
ai = n_poly_degree(Acoeffs + Ai);
}
}
else if (Fi < Flen && (Ai < 0 || Fexps[Fi] > pack_exp3(Ai, ai, 0)))
{
n_fq_poly_eval_pow(v, Fcoeffs + Fi, alphapow, ctx);
if (!_n_fq_is_zero(v, d))
{
changed = 1;
_n_fq_neg(v, v, d, ctx->mod);
n_fq_poly_scalar_addmul_n_fq(Tcoeffs + Ti,
Fcoeffs + Fi, modulus, v, ctx);
}
else
{
n_fq_poly_set(Tcoeffs + Ti, Fcoeffs + Fi, ctx);
}
Texps[Ti] = Fexps[Fi];
Fi++;
}
else
{
FLINT_ASSERT(Ai >= 0 && (Fi >= Flen || Fexps[Fi] < pack_exp3(Ai, ai, 0)));
Texps[Ti] = pack_exp3(Ai, ai, 0);
changed = 1;
n_fq_poly_scalar_mul_n_fq(Tcoeffs + Ti, modulus, Acoeffs[Ai].coeffs + d*ai, ctx);
do {
ai--;
} while (ai >= 0 && _n_fq_is_zero(Acoeffs[Ai].coeffs + d*ai, d));
if (ai < 0)
{
do {
Ai--;
} while (Ai >= 0 && Acoeffs[Ai].length == 0);
if (Ai >= 0)
ai = n_poly_degree(Acoeffs + Ai);
}
}
FLINT_ASSERT(!n_poly_is_zero(Tcoeffs + Ti));
lastlength = FLINT_MAX(lastlength, Tcoeffs[Ti].length);
Ti++;
}
T->length = Ti;
if (changed)
n_polyun_swap(T, F);
FLINT_ASSERT(n_polyun_fq_is_canonical(F, ctx));
flint_free(v);
*lastdeg = lastlength - 1;
return changed;
}
int n_fq_polyu3_hlift(
slong r,
n_polyun_struct * BB,
n_polyu_t A,
n_polyu_struct * B,
const fq_nmod_t beta,
slong degree_inner,
const fq_nmod_ctx_t ctx,
n_poly_bpoly_stack_t St)
{
slong d = fq_nmod_ctx_degree(ctx);
int success, Eok;
slong i, j;
n_polyun_t T;
n_bpoly_struct * Bp;
n_bpoly_t Ap;
n_fq_poly_t modulus, alphapow;
fq_nmod_t alpha;
slong * BBdegZ;
slong AdegY, AdegX, AdegZ;
slong bad_primes_left;
ulong * c = FLINT_ARRAY_ALLOC(d, ulong);
nmod_eval_interp_t E;
fq_nmod_init(alpha, ctx);
FLINT_ASSERT(n_polyu_fq_is_canonical(A, ctx));
for (i = 0; i < r; i++)
FLINT_ASSERT(n_polyu_fq_is_canonical(B + i, ctx));
BBdegZ = FLINT_ARRAY_ALLOC(r, slong);
Bp = FLINT_ARRAY_ALLOC(r, n_bpoly_struct);
for (i = 0; i < r; i++)
n_bpoly_init(Bp + i);
n_polyun_init(T);
n_fq_poly_init(modulus);
n_poly_init2(alphapow, 2*d);
n_bpoly_init(Ap);
nmod_eval_interp_init(E);
Eok = nmod_eval_interp_set_degree_modulus(E, degree_inner, ctx->mod);
n_polyu3_degrees(&AdegY, &AdegX, &AdegZ, A);
if (AdegX != degree_inner)
{
success = -1;
goto cleanup;
}
n_fq_poly_one(modulus, ctx);
fq_nmod_zero(alpha, ctx);
bad_primes_left = FLINT_MAX(5, AdegZ);
choose_prime:
if (fq_nmod_next(alpha, ctx) == 0)
{
success = -1;
goto cleanup;
}
FLINT_ASSERT(alphapow->alloc >= 2*d);
alphapow->length = 2;
_n_fq_one(alphapow->coeffs + d*0, d);
n_fq_set_fq_nmod(alphapow->coeffs + d*1, alpha, ctx);
fq_nmod_polyu3_interp_reduce_bpoly(Ap, A, alphapow, ctx);
for (i = 0; i < r; i++)
fq_nmod_polyu3_interp_reduce_bpoly(Bp + i, B + i, alphapow, ctx);
if (r < 3)
{
if (Eok)
success = n_fq_bpoly_hlift2_cubic(Ap, Bp + 0, Bp + 1, beta, degree_inner, ctx, E, St);
else
success = n_fq_bpoly_hlift2(Ap, Bp + 0, Bp + 1, beta, degree_inner, ctx, St);
}
else
{
success = n_fq_bpoly_hlift(r, Ap, Bp, beta, degree_inner, ctx, St);
}
if (success < 1)
{
if (success == 0 || --bad_primes_left < 0)
goto cleanup;
goto choose_prime;
}
if (n_fq_poly_degree(modulus) > 0)
{
n_fq_poly_eval_pow(c, modulus, alphapow, ctx);
n_fq_inv(c, c, ctx);
n_fq_poly_scalar_mul_n_fq(modulus, modulus, c, ctx);
for (i = 0; i < r; i++)
{
fq_nmod_polyu3n_interp_crt_sm_bpoly(BBdegZ + i, BB + i, T,
Bp + i, modulus, alphapow, ctx);
}
}
else
{
for (i = 0; i < r; i++)
{
fq_nmod_polyu3n_interp_lift_sm_bpoly(BBdegZ + i, BB + i, Bp + i, ctx);
}
}
n_fq_poly_shift_left_scalar_submul(modulus, 1, alphapow->coeffs + d*1, ctx);
j = BBdegZ[0];
for (i = 1; i < r; i++)
j += BBdegZ[i];
if (j > AdegZ)
{
success = 0;
goto cleanup;
}
if (n_fq_poly_degree(modulus) <= AdegZ)
{
goto choose_prime;
}
success = 1;
cleanup:
#if FLINT_WANT_ASSERT
if (success == 1)
{
n_polyu_t T1, T2, T3;
n_polyu_init(T1);
n_polyu_init(T2);
n_polyu_init(T3);
n_fq_polyu_set_n_fq_polyun(T2, BB + 0, ctx);
n_fq_polyu_set_n_fq_polyun(T3, BB + 1, ctx);
fq_nmod_polyu_mul(T1, T2, T3, ctx);
for (i = 2; i < r; i++)
{
n_fq_polyu_set_n_fq_polyun(T3, BB + i, ctx);
fq_nmod_polyu_mul(T2, T1, T3, ctx);
n_polyu_swap(T2, T1);
}
FLINT_ASSERT(n_fq_polyu_equal(A, T1, ctx));
n_polyu_clear(T1);
n_polyu_clear(T2);
n_polyu_clear(T3);
}
#endif
n_polyun_clear(T);
n_bpoly_clear(Ap);
for (i = 0; i < r; i++)
n_bpoly_clear(Bp + i);
flint_free(BBdegZ);
flint_free(Bp);
n_fq_poly_clear(alphapow);
n_fq_poly_clear(modulus);
fq_nmod_clear(alpha, ctx);
flint_free(c);
nmod_eval_interp_clear(E);
return success;
}