#include "fmpz_mod_mpoly_factor.h"
int fmpz_mod_mpoly_pfrac_init(
fmpz_mod_mpoly_pfrac_t I,
flint_bitcnt_t bits,
slong r,
slong w,
const fmpz_mod_mpoly_struct * betas,
const fmpz * alpha,
const fmpz_mod_mpoly_ctx_t ctx)
{
int success = 1;
slong i, j, k;
fmpz_mod_poly_t p;
fmpz_mod_poly_t G, S, pq;
FLINT_ASSERT(bits <= FLINT_BITS);
I->bits = bits;
I->r = r;
I->w = w;
I->dbetas = FLINT_ARRAY_ALLOC(r, fmpz_mod_poly_struct);
I->inv_prod_dbetas = FLINT_ARRAY_ALLOC(r, fmpz_mod_poly_struct);
I->prod_mbetas = FLINT_ARRAY_ALLOC((w + 1)*r, fmpz_mod_mpoly_struct);
I->prod_mbetas_coeffs = FLINT_ARRAY_ALLOC((w + 1)*r, fmpz_mod_mpolyv_struct);
I->mbetas = FLINT_ARRAY_ALLOC((w + 1)*r, fmpz_mod_mpoly_struct);
I->deltas = FLINT_ARRAY_ALLOC((w + 1)*r, fmpz_mod_mpoly_struct);
I->xalpha = FLINT_ARRAY_ALLOC(w + 1, fmpz_mod_mpoly_struct);
I->q = FLINT_ARRAY_ALLOC(w + 1, fmpz_mod_mpoly_struct);
I->G = FLINT_ARRAY_ALLOC(w + 1, fmpz_mod_mpoly_geobucket_struct);
I->qt = FLINT_ARRAY_ALLOC(w + 1, fmpz_mod_mpoly_struct);
I->newt = FLINT_ARRAY_ALLOC(w + 1, fmpz_mod_mpoly_struct);
I->delta_coeffs = FLINT_ARRAY_ALLOC((w + 1)*r, fmpz_mod_mpolyv_struct);
for (i = 0; i <= w; i++)
{
fmpz_mod_mpoly_init(I->xalpha + i, ctx);
fmpz_mod_mpoly_init(I->q + i, ctx);
fmpz_mod_mpoly_geobucket_init(I->G + i, ctx);
fmpz_mod_mpoly_init(I->qt + i, ctx);
fmpz_mod_mpoly_init(I->newt + i, ctx);
for (j = 0; j < r; j++)
{
fmpz_mod_mpoly_init(I->deltas + i*r + j, ctx);
fmpz_mod_mpolyv_init(I->delta_coeffs + i*r + j, ctx);
}
if (i < 1)
continue;
fmpz_mod_mpoly_gen(I->xalpha + i, i, ctx);
fmpz_mod_mpoly_sub_fmpz(I->xalpha + i, I->xalpha + i, alpha + i - 1, ctx);
fmpz_mod_mpoly_repack_bits_inplace(I->xalpha + i, I->bits, ctx);
}
fmpz_mod_poly_init(p, ctx->ffinfo);
fmpz_mod_poly_init(G, ctx->ffinfo);
fmpz_mod_poly_init(S, ctx->ffinfo);
fmpz_mod_poly_init(pq, ctx->ffinfo);
i = w;
for (j = 0; j < r; j++)
{
fmpz_mod_mpoly_init(I->mbetas + i*r + j, ctx);
fmpz_mod_mpoly_set(I->mbetas + i*r + j, betas + j, ctx);
}
for (i--; i >= 0; i--)
{
for (j = 0; j < r; j++)
{
fmpz_mod_mpoly_init(I->mbetas + i*r + j, ctx);
fmpz_mod_mpoly_evaluate_one_fmpz(I->mbetas + i*r + j,
I->mbetas + (i + 1)*r + j, i + 1, alpha + i, ctx);
}
}
for (j = 0; j < r; j++)
{
fmpz_mod_poly_init(I->dbetas + j, ctx->ffinfo);
if (!fmpz_mod_mpoly_get_fmpz_mod_poly(I->dbetas + j,
I->mbetas + 0*r + j, 0, ctx))
{
success = 0;
}
}
for (i = w; i >= 0; i--)
{
for (j = 0; j < r; j++)
{
fmpz_mod_mpoly_init(I->prod_mbetas + i*r + j, ctx);
fmpz_mod_mpoly_one(I->prod_mbetas + i*r + j, ctx);
for (k = 0; k < r; k++)
{
if (k == j)
continue;
fmpz_mod_mpoly_mul(I->prod_mbetas + i*r + j,
I->prod_mbetas + i*r + j, I->mbetas + i*r + k, ctx);
}
fmpz_mod_mpolyv_init(I->prod_mbetas_coeffs + i*r + j, ctx);
if (i > 0)
{
fmpz_mod_mpoly_to_mpolyv(I->prod_mbetas_coeffs + i*r + j,
I->prod_mbetas + i*r + j, I->xalpha + i, ctx);
}
}
}
for (j = 0; j < r; j++)
fmpz_mod_poly_init(I->inv_prod_dbetas + j, ctx->ffinfo);
for (j = 0; success && j < r; j++)
{
if (fmpz_mod_poly_degree(I->dbetas + j, ctx->ffinfo) !=
fmpz_mod_mpoly_degree_si(betas + j, 0, ctx))
{
success = 0;
}
}
for (j = 0; success && j < r; j++)
{
fmpz_mod_poly_one(pq, ctx->ffinfo);
for (k = 0; k < r; k++)
{
if (k == j)
continue;
fmpz_mod_poly_mul(pq, pq, I->dbetas + k, ctx->ffinfo);
}
fmpz_mod_poly_xgcd(G, S, I->inv_prod_dbetas + j, I->dbetas + j, pq, ctx->ffinfo);
if (!fmpz_mod_poly_is_one(G, ctx->ffinfo))
success = 0;
}
fmpz_mod_poly_clear(p, ctx->ffinfo);
fmpz_mod_poly_clear(G, ctx->ffinfo);
fmpz_mod_poly_clear(S, ctx->ffinfo);
fmpz_mod_poly_clear(pq, ctx->ffinfo);
I->dbetas_mvar = FLINT_ARRAY_ALLOC(r, fmpz_mod_mpoly_struct);
I->inv_prod_dbetas_mvar = FLINT_ARRAY_ALLOC(r, fmpz_mod_mpoly_struct);
for (j = 0; j < r; j++)
{
fmpz_mod_mpoly_init(I->dbetas_mvar + j, ctx);
fmpz_mod_mpoly_init(I->inv_prod_dbetas_mvar + j, ctx);
_fmpz_mod_mpoly_set_fmpz_mod_poly(I->dbetas_mvar + j, I->bits,
I->dbetas[j].coeffs, I->dbetas[j].length, 0, ctx);
_fmpz_mod_mpoly_set_fmpz_mod_poly(I->inv_prod_dbetas_mvar + j, I->bits,
I->inv_prod_dbetas[j].coeffs, I->inv_prod_dbetas[j].length, 0, ctx);
}
fmpz_mod_mpoly_init(I->T, ctx);
fmpz_mod_mpoly_init(I->Q, ctx);
fmpz_mod_mpoly_init(I->R, ctx);
FLINT_ASSERT(success == 1);
return success;
}
void fmpz_mod_mpoly_pfrac_clear(
fmpz_mod_mpoly_pfrac_t I,
const fmpz_mod_mpoly_ctx_t ctx)
{
slong i, j;
for (i = 0; i <= I->w; i++)
{
fmpz_mod_mpoly_clear(I->xalpha + i, ctx);
fmpz_mod_mpoly_clear(I->q + i, ctx);
fmpz_mod_mpoly_geobucket_clear(I->G + i, ctx);
fmpz_mod_mpoly_clear(I->qt + i, ctx);
fmpz_mod_mpoly_clear(I->newt + i, ctx);
for (j = 0; j < I->r; j++)
fmpz_mod_mpolyv_clear(I->delta_coeffs + i*I->r + j, ctx);
}
flint_free(I->xalpha);
flint_free(I->q);
flint_free(I->G);
flint_free(I->qt);
flint_free(I->newt);
flint_free(I->delta_coeffs);
for (j = 0; j < I->r; j++)
{
fmpz_mod_poly_clear(I->inv_prod_dbetas + j, ctx->ffinfo);
fmpz_mod_poly_clear(I->dbetas + j, ctx->ffinfo);
for (i = 0; i <= I->w; i++)
{
fmpz_mod_mpolyv_clear(I->prod_mbetas_coeffs + i*I->r + j, ctx);
fmpz_mod_mpoly_clear(I->prod_mbetas + i*I->r + j, ctx);
fmpz_mod_mpoly_clear(I->mbetas + i*I->r + j, ctx);
fmpz_mod_mpoly_clear(I->deltas + i*I->r + j, ctx);
}
}
flint_free(I->inv_prod_dbetas);
flint_free(I->dbetas);
flint_free(I->prod_mbetas);
flint_free(I->prod_mbetas_coeffs);
flint_free(I->mbetas);
flint_free(I->deltas);
for (j = 0; j < I->r; j++)
{
fmpz_mod_mpoly_clear(I->dbetas_mvar + j, ctx);
fmpz_mod_mpoly_clear(I->inv_prod_dbetas_mvar + j, ctx);
}
flint_free(I->dbetas_mvar);
flint_free(I->inv_prod_dbetas_mvar);
fmpz_mod_mpoly_clear(I->T, ctx);
fmpz_mod_mpoly_clear(I->Q, ctx);
fmpz_mod_mpoly_clear(I->R, ctx);
}
int fmpz_mod_mpoly_pfrac(
slong l,
fmpz_mod_mpoly_t t,
const slong * degs,
fmpz_mod_mpoly_pfrac_t I,
const fmpz_mod_mpoly_ctx_t ctx)
{
int success;
slong i, j, k;
fmpz_mod_mpoly_struct * deltas = I->deltas + l*I->r;
fmpz_mod_mpoly_struct * newdeltas = I->deltas + (l - 1)*I->r;
fmpz_mod_mpoly_struct * q = I->q + l;
fmpz_mod_mpoly_struct * qt = I->qt + l;
fmpz_mod_mpoly_struct * newt = I->newt + l;
fmpz_mod_mpolyv_struct * delta_coeffs = I->delta_coeffs + l*I->r;
fmpz_mod_mpoly_geobucket_struct * G = I->G + l;
FLINT_ASSERT(l >= 0);
if (!fmpz_mod_mpoly_repack_bits_inplace(t, I->bits, ctx))
return -1;
if (l < 1)
{
for (i = 0; i < I->r; i++)
{
fmpz_mod_mpoly_divrem(I->Q, I->R, t, I->dbetas_mvar + i, ctx);
fmpz_mod_mpoly_mul(I->T, I->R, I->inv_prod_dbetas_mvar + i, ctx);
fmpz_mod_mpoly_divrem(I->Q, deltas + i, I->T, I->dbetas_mvar + i, ctx);
}
return 1;
}
for (i = 0; i < I->r; i++)
delta_coeffs[i].length = 0;
for (k = 0; k <= degs[l]; k++)
{
fmpz_mod_mpoly_divrem(q, newt, t, I->xalpha + l, ctx);
fmpz_mod_mpoly_swap(t, q, ctx);
fmpz_mod_mpoly_geobucket_set(G, newt, ctx);
for (j = 0; j < k; j++)
for (i = 0; i < I->r; i++)
{
if (j >= delta_coeffs[i].length)
continue;
if (k - j >= I->prod_mbetas_coeffs[l*I->r + i].length)
continue;
fmpz_mod_mpoly_mul(qt, delta_coeffs[i].coeffs + j,
I->prod_mbetas_coeffs[l*I->r + i].coeffs + k - j, ctx);
fmpz_mod_mpoly_geobucket_sub(G, qt, ctx);
}
fmpz_mod_mpoly_geobucket_empty(newt, G, ctx);
if (fmpz_mod_mpoly_is_zero(newt, ctx))
continue;
success = fmpz_mod_mpoly_pfrac(l - 1, newt, degs, I, ctx);
if (success < 1)
return success;
for (i = 0; i < I->r; i++)
{
if (fmpz_mod_mpoly_is_zero(newdeltas + i, ctx))
continue;
if (k + I->prod_mbetas_coeffs[l*I->r + i].length - 1 > degs[l])
return 0;
fmpz_mod_mpolyv_set_coeff(delta_coeffs + i, k, newdeltas + i, ctx);
}
}
for (i = 0; i < I->r; i++)
fmpz_mod_mpoly_from_mpolyv(deltas + i, I->bits,
delta_coeffs + i, I->xalpha + l, ctx);
return 1;
}