#include "test_helpers.h"
#include "fq_nmod_mpoly.h"
void univar_divides_check(
const fq_nmod_mpoly_t A,
const fq_nmod_mpoly_t B,
const fq_nmod_mpoly_ctx_t ctx,
slong ii,
slong jj,
const char * name,
flint_rand_t randstate)
{
int divides, udivides;
fq_nmod_mpoly_ctx_t uctx;
fq_nmod_mpolyu_t Au, Bu, Qu;
fq_nmod_mpoly_t Q, Qcheck;
flint_bitcnt_t ABbits;
ulong * shift, * stride;
slong * perm;
slong i, j, k;
if ( A->bits > FLINT_BITS
|| B->bits > FLINT_BITS
|| A->length == 0
|| B->length == 0
|| ctx->minfo->nvars < 2)
{
return;
}
perm = (slong *) flint_malloc((ctx->minfo->nvars)*sizeof(slong));
shift = (ulong *) flint_malloc((ctx->minfo->nvars)*sizeof(ulong));
stride = (ulong *) flint_malloc((ctx->minfo->nvars)*sizeof(ulong));
fq_nmod_mpoly_init(Q, ctx);
fq_nmod_mpoly_init(Qcheck, ctx);
fq_nmod_mpoly_ctx_init(uctx, ctx->minfo->nvars - 1, ORD_LEX, ctx->fqctx);
ABbits = FLINT_MAX(A->bits, B->bits);
fq_nmod_mpolyu_init(Au, ABbits, uctx);
fq_nmod_mpolyu_init(Bu, ABbits, uctx);
fq_nmod_mpolyu_init(Qu, ABbits, uctx);
for (i = 0; i < ctx->minfo->nvars; i++)
{
perm[i] = i;
shift[i] = 0;
stride[i] = 1;
}
for (k = 0; k < ctx->minfo->nvars; k++)
{
slong t1, t2;
i = n_randint(randstate, ctx->minfo->nvars - 1);
j = i + n_randint(randstate, ctx->minfo->nvars - i);
t1 = perm[i];
t2 = perm[j];
perm[i] = t2;
perm[j] = t1;
}
fq_nmod_mpoly_to_mpolyu_perm_deflate(Au, uctx, A, ctx, perm, shift, stride);
fq_nmod_mpoly_to_mpolyu_perm_deflate(Bu, uctx, B, ctx, perm, shift, stride);
udivides = fq_nmod_mpolyuu_divides(Qu, Au, Bu, 1, uctx);
divides = fq_nmod_mpoly_divides(Q, A, B, ctx);
if (divides != udivides)
{
flint_printf("check univariate divisibility\n"
"i = %wd, j = %wd, %s\n", ii, jj, name);
fflush(stdout);
flint_abort();
}
if (!divides)
{
goto cleanup;
}
fq_nmod_mpoly_from_mpolyu_perm_inflate(Qcheck, ABbits, ctx, Qu, uctx,
perm, shift, stride);
if (!fq_nmod_mpoly_equal(Q, Qcheck, ctx))
{
flint_printf("check univariate quotient\n"
"i = %wd, j = %wd, %s\n", ii, jj, name);
fflush(stdout);
flint_abort();
}
cleanup:
fq_nmod_mpolyu_clear(Au, uctx);
fq_nmod_mpolyu_clear(Bu, uctx);
fq_nmod_mpolyu_clear(Qu, uctx);
fq_nmod_mpoly_ctx_clear(uctx);
fq_nmod_mpoly_clear(Q, ctx);
fq_nmod_mpoly_clear(Qcheck, ctx);
flint_free(perm);
flint_free(shift);
flint_free(stride);
}
TEST_FUNCTION_START(fq_nmod_mpoly_mpolyuu_divides, state)
{
slong i, j, tmul = 50;
for (i = 0; i < tmul * flint_test_multiplier(); i++)
{
fq_nmod_mpoly_ctx_t ctx;
fq_nmod_mpoly_t a, b;
slong len1, len2;
flint_bitcnt_t exp_bits;
flint_bitcnt_t pbits;
slong deg;
pbits = 1 + n_randint(state, FLINT_BITS);
pbits = 1 + n_randint(state, pbits);
deg = 1 + n_randint(state, 4);
fq_nmod_mpoly_ctx_init_rand(ctx, state, 15, pbits, deg);
fq_nmod_mpoly_init(a, ctx);
fq_nmod_mpoly_init(b, ctx);
len1 = n_randint(state, 20);
len2 = n_randint(state, 20);
exp_bits = n_randint(state, FLINT_BITS/2 - 2) + 2;
for (j = 0; j < 4; j++)
{
fq_nmod_mpoly_randtest_bits(a, state, len1, exp_bits, ctx);
fq_nmod_mpoly_randtest_bits(b, state, len2, exp_bits, ctx);
fq_nmod_mpoly_mul(a, a, b, ctx);
univar_divides_check(a, b, ctx, i, j, "univar Check (a*b)/b = a", state);
}
fq_nmod_mpoly_clear(a, ctx);
fq_nmod_mpoly_clear(b, ctx);
fq_nmod_mpoly_ctx_clear(ctx);
}
for (i = 0; i < tmul * flint_test_multiplier(); i++)
{
fq_nmod_mpoly_ctx_t ctx;
fq_nmod_mpoly_t a, b, c;
slong len1, len2, len3;
flint_bitcnt_t exp_bits;
flint_bitcnt_t pbits;
slong deg;
pbits = 1 + n_randint(state, FLINT_BITS);
pbits = 1 + n_randint(state, pbits);
deg = 1 + n_randint(state, 4);
fq_nmod_mpoly_ctx_init_rand(ctx, state, 12, pbits, deg);
fq_nmod_mpoly_init(a, ctx);
fq_nmod_mpoly_init(b, ctx);
fq_nmod_mpoly_init(c, ctx);
len1 = n_randint(state, 20);
len2 = n_randint(state, 20);
len3 = n_randint(state, 30);
exp_bits = n_randint(state, 8) + 3;
for (j = 0; j < 4; j++)
{
fq_nmod_mpoly_randtest_bits(a, state, len1, exp_bits, ctx);
fq_nmod_mpoly_randtest_bits(b, state, len2, exp_bits, ctx);
fq_nmod_mpoly_randtest_bits(c, state, len3, exp_bits, ctx);
fq_nmod_mpoly_mul(a, a, b, ctx);
fq_nmod_mpoly_add(a, a, c, ctx);
univar_divides_check(a, b, ctx, i, j, "univar Check (a*b + c)/b", state);
}
fq_nmod_mpoly_clear(a, ctx);
fq_nmod_mpoly_clear(b, ctx);
fq_nmod_mpoly_clear(c, ctx);
fq_nmod_mpoly_ctx_clear(ctx);
}
TEST_FUNCTION_END(state);
}