#include "fq_zech_mpoly_factor.h"
static int _hlift_quartic2(
slong m,
fq_zech_mpoly_struct * f,
slong r,
const fq_zech_struct * alpha,
const fq_zech_mpoly_t A,
const slong * degs,
const fq_zech_mpoly_ctx_t ctx)
{
int success;
slong i, j;
fq_zech_mpoly_t Aq, t, t2, t3, xalpha;
fq_zech_mpoly_struct * deltas;
fq_zech_mpoly_pfrac_t I;
fq_zech_mpolyv_struct B[2];
fq_zech_mpoly_struct betas[2];
slong tdeg;
flint_bitcnt_t bits = A->bits;
FLINT_ASSERT(r == 2);
r = 2;
fq_zech_mpoly_init(t, ctx);
fq_zech_mpoly_init(t2, ctx);
fq_zech_mpoly_init(t3, ctx);
fq_zech_mpoly_init(xalpha, ctx);
fq_zech_mpoly_init(Aq, ctx);
fq_zech_mpoly_gen(xalpha, m, ctx);
fq_zech_mpoly_sub_fq_zech(xalpha, xalpha, alpha + m - 1, ctx);
fq_zech_mpoly_repack_bits_inplace(xalpha, bits, ctx);
for (i = 0; i < r; i++)
{
fq_zech_mpolyv_init(B + i, ctx);
fq_zech_mpoly_repack_bits_inplace(f + i, bits, ctx);
fq_zech_mpoly_to_mpolyv(B + i, f + i, xalpha, ctx);
fq_zech_mpolyv_fit_length(B + i, degs[m] + 1, ctx);
for (j = B[i].length; j <= degs[m]; j++)
fq_zech_mpoly_zero(B[i].coeffs + j, ctx);
betas[i] = B[i].coeffs[0];
}
success = fq_zech_mpoly_pfrac_init(I, bits, r, m - 1, betas, alpha, ctx);
FLINT_ASSERT(success == 1);
deltas = I->deltas + (m - 1)*I->r;
fq_zech_mpoly_divrem(Aq, t, A, xalpha, ctx);
#if FLINT_WANT_ASSERT
fq_zech_mpoly_one(t2, ctx);
for (i = 0; i < r; i++)
fq_zech_mpoly_mul(t2, t2, betas + i, ctx);
FLINT_ASSERT(fq_zech_mpoly_equal(t, t2, ctx));
#endif
for (j = 1; j <= degs[m]; j++)
{
fq_zech_mpoly_divrem(t2, t, Aq, xalpha, ctx);
fq_zech_mpoly_swap(Aq, t2, ctx);
for (i = 0; i <= j; i++)
{
fq_zech_mpoly_mul(t2, B[0].coeffs + i, B[1].coeffs + j - i, ctx);
fq_zech_mpoly_sub(t3, t, t2, ctx);
fq_zech_mpoly_swap(t, t3, ctx);
}
if (fq_zech_mpoly_is_zero(t, ctx))
continue;
success = fq_zech_mpoly_pfrac(m - 1, t, degs, I, ctx);
if (success < 1)
{
success = 0;
goto cleanup;
}
tdeg = 0;
for (i = 0; i < r; i++)
{
fq_zech_mpoly_add(t3, B[i].coeffs + j, deltas + i, ctx);
fq_zech_mpoly_swap(B[i].coeffs + j, t3, ctx);
if (!fq_zech_mpoly_is_zero(B[i].coeffs + j, ctx))
B[i].length = FLINT_MAX(B[i].length, j + 1);
FLINT_ASSERT(B[i].length > 0);
tdeg += B[i].length - 1;
}
if (tdeg > degs[m])
{
success = 0;
goto cleanup;
}
}
success = 1;
cleanup:
fq_zech_mpoly_pfrac_clear(I, ctx);
for (i = 0; i < r; i++)
{
if (success)
fq_zech_mpoly_from_mpolyv(f + i, B + i, xalpha, ctx);
fq_zech_mpolyv_clear(B + i, ctx);
}
fq_zech_mpoly_clear(t, ctx);
fq_zech_mpoly_clear(t2, ctx);
fq_zech_mpoly_clear(t3, ctx);
fq_zech_mpoly_clear(xalpha, ctx);
fq_zech_mpoly_clear(Aq, ctx);
return success;
}
static int _hlift_quartic(
slong m,
fq_zech_mpoly_struct * f,
slong r,
const fq_zech_struct * alpha,
const fq_zech_mpoly_t A,
const slong * degs,
const fq_zech_mpoly_ctx_t ctx)
{
int success;
slong i, j, k;
fq_zech_mpoly_t Aq, t, t1, t2, t3, xalpha;
fq_zech_mpoly_struct * betas, * deltas;
fq_zech_mpoly_pfrac_t I;
fq_zech_mpolyv_struct * B, * U;
slong tdeg;
flint_bitcnt_t bits = A->bits;
FLINT_ASSERT(r > 2);
betas = FLINT_ARRAY_ALLOC(r, fq_zech_mpoly_struct);
B = FLINT_ARRAY_ALLOC(2*r, fq_zech_mpolyv_struct);
U = B + r;
fq_zech_mpoly_init(t, ctx);
fq_zech_mpoly_init(t1, ctx);
fq_zech_mpoly_init(t2, ctx);
fq_zech_mpoly_init(t3, ctx);
fq_zech_mpoly_init(xalpha, ctx);
fq_zech_mpoly_init(Aq, ctx);
fq_zech_mpoly_gen(xalpha, m, ctx);
fq_zech_mpoly_sub_fq_zech(xalpha, xalpha, alpha + m - 1, ctx);
fq_zech_mpoly_repack_bits_inplace(xalpha, bits, ctx);
for (k = 0; k < r; k++)
{
fq_zech_mpolyv_init(U + k, ctx);
fq_zech_mpolyv_fit_length(U + k, degs[m] + 1, ctx);
for (j = 0; j <= degs[m]; j++)
fq_zech_mpoly_zero(U[k].coeffs + j, ctx);
fq_zech_mpolyv_init(B + k, ctx);
fq_zech_mpoly_repack_bits_inplace(f + k, bits, ctx);
fq_zech_mpoly_to_mpolyv(B + k, f + k, xalpha, ctx);
fq_zech_mpolyv_fit_length(B + k, degs[m] + 1, ctx);
for (j = B[k].length; j <= degs[m]; j++)
fq_zech_mpoly_zero(B[k].coeffs + j, ctx);
betas[k] = B[k].coeffs[0];
}
success = fq_zech_mpoly_pfrac_init(I, bits, r, m - 1, betas, alpha, ctx);
FLINT_ASSERT(success == 1);
deltas = I->deltas + (m - 1)*I->r;
k = r - 2;
fq_zech_mpoly_mul(U[k].coeffs + 0, B[k].coeffs + 0, B[k + 1].coeffs + 0, ctx);
for (k--; k >= 1; k--)
fq_zech_mpoly_mul(U[k].coeffs + 0, B[k].coeffs + 0, U[k + 1].coeffs + 0, ctx);
fq_zech_mpoly_divrem(Aq, t, A, xalpha, ctx);
#if FLINT_WANT_ASSERT
fq_zech_mpoly_one(t2, ctx);
for (i = 0; i < r; i++)
fq_zech_mpoly_mul(t2, t2, betas + i, ctx);
FLINT_ASSERT(fq_zech_mpoly_equal(t, t2, ctx));
#endif
for (j = 1; j <= degs[m]; j++)
{
k = r - 2;
fq_zech_mpoly_zero(U[k].coeffs + j, ctx);
for (i = 0; i <= j; i++)
{
fq_zech_mpoly_mul(t1, B[k].coeffs + i, B[k + 1].coeffs + j - i, ctx);
fq_zech_mpoly_add(U[k].coeffs + j, U[k].coeffs + j, t1, ctx);
}
for (k--; k >= 1; k--)
{
fq_zech_mpoly_zero(U[k].coeffs + j, ctx);
for (i = 0; i <= j; i++)
{
fq_zech_mpoly_mul(t1, B[k].coeffs + i, U[k + 1].coeffs + j - i, ctx);
fq_zech_mpoly_add(U[k].coeffs + j, U[k].coeffs + j, t1, ctx);
}
}
fq_zech_mpoly_divrem(t2, t, Aq, xalpha, ctx);
fq_zech_mpoly_swap(Aq, t2, ctx);
for (i = 0; i <= j; i++)
{
fq_zech_mpoly_mul(t2, B[0].coeffs + i, U[1].coeffs + j - i, ctx);
fq_zech_mpoly_sub(t3, t, t2, ctx);
fq_zech_mpoly_swap(t, t3, ctx);
}
if (fq_zech_mpoly_is_zero(t, ctx))
continue;
success = fq_zech_mpoly_pfrac(m - 1, t, degs, I, ctx);
if (success < 1)
{
success = 0;
goto cleanup;
}
tdeg = 0;
for (i = 0; i < r; i++)
{
fq_zech_mpoly_add(t3, B[i].coeffs + j, deltas + i, ctx);
fq_zech_mpoly_swap(B[i].coeffs + j, t3, ctx);
if (!fq_zech_mpoly_is_zero(B[i].coeffs + j, ctx))
B[i].length = FLINT_MAX(B[i].length, j + 1);
FLINT_ASSERT(B[i].length > 0);
tdeg += B[i].length - 1;
}
if (tdeg > degs[m])
{
success = 0;
goto cleanup;
}
k = r - 2;
fq_zech_mpoly_mul(t, B[k].coeffs + 0, deltas + k + 1, ctx);
fq_zech_mpoly_mul(t1, deltas + k, B[k + 1].coeffs + 0, ctx);
fq_zech_mpoly_add(t, t, t1, ctx);
fq_zech_mpoly_add(U[k].coeffs + j, U[k].coeffs + j, t, ctx);
for (k--; k >= 1; k--)
{
fq_zech_mpoly_mul(t1, B[k].coeffs + 0, t, ctx);
fq_zech_mpoly_swap(t, t1, ctx);
fq_zech_mpoly_mul(t1, deltas + k, U[k + 1].coeffs + 0, ctx);
fq_zech_mpoly_add(t, t, t1, ctx);
fq_zech_mpoly_add(U[k].coeffs + j, U[k].coeffs + j, t, ctx);
}
}
success = 1;
cleanup:
fq_zech_mpoly_pfrac_clear(I, ctx);
flint_free(betas);
for (i = 0; i < r; i++)
{
if (success)
fq_zech_mpoly_from_mpolyv(f + i, B + i, xalpha, ctx);
fq_zech_mpolyv_clear(B + i, ctx);
fq_zech_mpolyv_clear(U + i, ctx);
}
flint_free(B);
fq_zech_mpoly_clear(t, ctx);
fq_zech_mpoly_clear(t1, ctx);
fq_zech_mpoly_clear(t2, ctx);
fq_zech_mpoly_clear(t3, ctx);
fq_zech_mpoly_clear(xalpha, ctx);
fq_zech_mpoly_clear(Aq, ctx);
return success;
}
static int _hlift_quintic(
slong m,
fq_zech_mpoly_struct * f,
slong r,
const fq_zech_struct * alpha,
const fq_zech_mpoly_t A,
const slong * degs,
const fq_zech_mpoly_ctx_t ctx)
{
int success;
slong i, j;
fq_zech_mpoly_t e, t, pow, xalpha, q;
fq_zech_mpoly_struct * betas, * deltas;
fq_zech_mpoly_pfrac_t I;
flint_bitcnt_t bits = A->bits;
FLINT_ASSERT(r > 1);
fq_zech_mpoly_init(e, ctx);
fq_zech_mpoly_init(t, ctx);
fq_zech_mpoly_init(pow, ctx);
fq_zech_mpoly_init(xalpha, ctx);
fq_zech_mpoly_init(q, ctx);
betas = (fq_zech_mpoly_struct *) flint_malloc(r*sizeof(fq_zech_mpoly_struct));
for (i = 0; i < r; i++)
{
fq_zech_mpoly_init(betas + i, ctx);
fq_zech_mpoly_repack_bits_inplace(f + i, bits, ctx);
fq_zech_mpoly_evaluate_one_fq_zech(betas + i, f + i, m, alpha + m - 1, ctx);
}
fq_zech_mpoly_mul(t, f + 0, f + 1, ctx);
for (i = 2; i < r; i++)
fq_zech_mpoly_mul(t, t, f + i, ctx);
fq_zech_mpoly_sub(e, A, t, ctx);
fq_zech_mpoly_one(pow, ctx);
fq_zech_mpoly_repack_bits_inplace(pow, bits, ctx);
fq_zech_mpoly_gen(xalpha, m, ctx);
fq_zech_mpoly_sub_fq_zech(xalpha, xalpha, alpha + m - 1, ctx);
fq_zech_mpoly_repack_bits_inplace(xalpha, bits, ctx);
fq_zech_mpoly_pfrac_init(I, bits, r, m - 1, betas, alpha, ctx);
deltas = I->deltas + (m - 1)*I->r;
for (j = 1; j <= degs[m]; j++)
{
if (fq_zech_mpoly_is_zero(e, ctx))
{
success = 1;
goto cleanup;
}
fq_zech_mpoly_mul(pow, pow, xalpha, ctx);
success = fq_zech_mpoly_divides(q, e, pow, ctx);
FLINT_ASSERT(success);
fq_zech_mpoly_evaluate_one_fq_zech(t, q, m, alpha + m - 1, ctx);
success = fq_zech_mpoly_pfrac(m - 1, t, degs, I, ctx);
if (success < 1)
{
success = 0;
goto cleanup;
}
for (i = 0; i < r; i++)
{
fq_zech_mpoly_mul(t, deltas + i, pow, ctx);
fq_zech_mpoly_add(f + i, f + i, t, ctx);
}
fq_zech_mpoly_mul(t, f + 0, f + 1, ctx);
for (i = 2; i < r; i++)
fq_zech_mpoly_mul(t, t, f + i, ctx);
fq_zech_mpoly_sub(e, A, t, ctx);
}
success = fq_zech_mpoly_is_zero(e, ctx);
cleanup:
fq_zech_mpoly_pfrac_clear(I, ctx);
fq_zech_mpoly_clear(e, ctx);
fq_zech_mpoly_clear(t, ctx);
fq_zech_mpoly_clear(pow, ctx);
fq_zech_mpoly_clear(xalpha, ctx);
fq_zech_mpoly_clear(q, ctx);
for (i = 0; i < r; i++)
fq_zech_mpoly_clear(betas + i, ctx);
flint_free(betas);
return success;
}
int fq_zech_mpoly_hlift(
slong m,
fq_zech_mpoly_struct * f,
slong r,
const fq_zech_struct * alpha,
const fq_zech_mpoly_t A,
const slong * degs,
const fq_zech_mpoly_ctx_t ctx)
{
FLINT_ASSERT(r >= 2);
FLINT_ASSERT(A->bits <= FLINT_BITS);
FLINT_ASSERT(ctx->minfo->ord == ORD_LEX);
if (r == 2)
return _hlift_quartic2(m, f, r, alpha, A, degs, ctx);
else if (r < 20)
return _hlift_quartic(m, f, r, alpha, A, degs, ctx);
else
return _hlift_quintic(m, f, r, alpha, A, degs, ctx);
}