/*
Copyright (C) 2017 Daniel Schultz
This file is part of FLINT.
FLINT is free software: you can redistribute it and/or modify it under
the terms of the GNU Lesser General Public License (LGPL) as published
by the Free Software Foundation; either version 2.1 of the License, or
(at your option) any later version. See .
*/
#include
#include
#include "flint.h"
#include "fmpz.h"
#include "fmpz_poly.h"
#include "fmpz_mpoly.h"
void fmpz_mpoly_univar_init(fmpz_mpoly_univar_t A, const fmpz_mpoly_ctx_t ctx)
{
A->coeffs = NULL;
A->exps = NULL;
A->alloc = 0;
A->length = 0;
}
void fmpz_mpoly_univar_clear(fmpz_mpoly_univar_t A, const fmpz_mpoly_ctx_t ctx)
{
slong i;
for (i = 0; i < A->alloc; i++)
{
fmpz_mpoly_clear(A->coeffs + i, ctx);
fmpz_clear(A->exps + i);
}
if (A->coeffs)
flint_free(A->coeffs);
if (A->exps)
flint_free(A->exps);
}
void fmpz_mpoly_univar_fit_length(fmpz_mpoly_univar_t A,
slong length, const fmpz_mpoly_ctx_t ctx)
{
slong i;
slong old_alloc = A->alloc;
slong new_alloc = FLINT_MAX(length, 2*A->alloc);
if (length > old_alloc)
{
if (old_alloc == 0)
{
A->exps = (fmpz *) flint_malloc(new_alloc*sizeof(fmpz));
A->coeffs = (fmpz_mpoly_struct *) flint_malloc(
new_alloc*sizeof(fmpz_mpoly_struct));
}
else
{
A->exps = (fmpz *) flint_realloc(A->exps, new_alloc*sizeof(fmpz));
A->coeffs = (fmpz_mpoly_struct *) flint_realloc(A->coeffs,
new_alloc*sizeof(fmpz_mpoly_struct));
}
for (i = old_alloc; i < new_alloc; i++)
{
fmpz_init(A->exps + i);
fmpz_mpoly_init(A->coeffs + i, ctx);
}
A->alloc = new_alloc;
}
}
void fmpz_mpoly_univar_set_coeff_ui(
fmpz_mpoly_univar_t A,
ulong e,
const fmpz_mpoly_t c,
const fmpz_mpoly_ctx_t ctx)
{
slong i, j;
for (i = A->length; i >= 0; i--)
{
int cmp = i > 0 ? fmpz_cmp_ui(A->exps + i - 1, e) : 1;
if (cmp > 0)
{
if (fmpz_mpoly_is_zero(c, ctx))
return;
fmpz_mpoly_univar_fit_length(A, A->length + 1, ctx);
for (j = A->length; j > i; j--)
{
fmpz_mpoly_swap(A->coeffs + j, A->coeffs + j + 1, ctx);
fmpz_swap(A->exps + j, A->exps + j + 1);
}
A->length++;
fmpz_set_ui(A->exps + i, e);
fmpz_mpoly_set(A->coeffs + i, c, ctx);
return;
}
else if (cmp == 0)
{
fmpz_mpoly_set(A->coeffs + i, c, ctx);
if (!fmpz_mpoly_is_zero(A->coeffs + i, ctx))
return;
A->length--;
for (j = i; j < A->length; j++)
{
fmpz_mpoly_swap(A->coeffs + j, A->coeffs + j + 1, ctx);
fmpz_swap(A->exps + j, A->exps + j + 1);
}
}
}
FLINT_ASSERT(0 && "unreachable");
return;
}
void fmpz_mpoly_univar_assert_canonical(fmpz_mpoly_univar_t A, const fmpz_mpoly_ctx_t ctx)
{
slong i;
for (i = 0; i + 1 < A->length; i++)
{
if (fmpz_cmp(A->exps + i, A->exps + i + 1) <= 0
|| fmpz_sgn(A->exps + i) < 0
|| fmpz_sgn(A->exps + i + 1) < 0)
{
flint_throw(FLINT_ERROR, "Univariate polynomial exponents out of order");
}
}
for (i = 0; i < A->length; i++)
fmpz_mpoly_assert_canonical(A->coeffs + i, ctx);
}
void fmpz_mpoly_univar_print_pretty(const fmpz_mpoly_univar_t A,
const char ** x, const fmpz_mpoly_ctx_t ctx)
{
slong i;
if (A->length == 0)
flint_printf("0");
for (i = 0; i < A->length; i++)
{
if (i != 0)
flint_printf("+");
flint_printf("(");
fmpz_mpoly_print_pretty(A->coeffs + i,x,ctx);
flint_printf(")*X^");
fmpz_print(A->exps + i);
}
}
static void _tree_data_clear_sp(
fmpz_mpoly_univar_t A,
mpoly_rbtree_ui_t tree,
slong idx,
const fmpz_mpoly_ctx_t ctx)
{
mpoly_rbnode_ui_struct * nodes = tree->nodes + 2;
fmpz_mpoly_struct * data = (fmpz_mpoly_struct *) tree->data;
if (idx < 0)
return;
_tree_data_clear_sp(A, tree, nodes[idx].right, ctx);
FLINT_ASSERT(A->length < A->alloc);
fmpz_set_ui(A->exps + A->length, nodes[idx].key);
fmpz_mpoly_swap(A->coeffs + A->length, data + idx, ctx);
A->length++;
fmpz_mpoly_clear(data + idx, ctx);
_tree_data_clear_sp(A, tree, nodes[idx].left, ctx);
}
static void _tree_data_clear_mp(
fmpz_mpoly_univar_t A,
mpoly_rbtree_fmpz_t tree,
slong idx,
const fmpz_mpoly_ctx_t ctx)
{
mpoly_rbnode_fmpz_struct * nodes = tree->nodes + 2;
fmpz_mpoly_struct * data = (fmpz_mpoly_struct *) tree->data;
if (idx < 0)
return;
_tree_data_clear_mp(A, tree, nodes[idx].right, ctx);
FLINT_ASSERT(A->length < A->alloc);
fmpz_set(A->exps + A->length, nodes[idx].key);
fmpz_mpoly_swap(A->coeffs + A->length, data + idx, ctx);
A->length++;
fmpz_mpoly_clear(data + idx, ctx);
_tree_data_clear_mp(A, tree, nodes[idx].left, ctx);
}
/* the coefficients of A should be constructed with the same bits as B */
void fmpz_mpoly_to_univar(fmpz_mpoly_univar_t A, const fmpz_mpoly_t B,
slong var, const fmpz_mpoly_ctx_t ctx)
{
flint_bitcnt_t bits = B->bits;
slong N = mpoly_words_per_exp(bits, ctx->minfo);
slong shift, off;
slong Blen = B->length;
const fmpz * Bcoeff = B->coeffs;
const ulong * Bexp = B->exps;
slong i;
int its_new;
ulong * one;
#define LUT_limit (48)
fmpz_mpoly_struct LUT[LUT_limit];
if (B->length == 0)
{
A->length = 0;
return;
}
one = FLINT_ARRAY_ALLOC(N, ulong);
if (bits <= FLINT_BITS)
{
slong Alen;
mpoly_rbtree_ui_t tree;
fmpz_mpoly_struct * d;
ulong mask = (-UWORD(1)) >> (FLINT_BITS - bits);
mpoly_rbtree_ui_init(tree, sizeof(fmpz_mpoly_struct));
mpoly_gen_monomial_offset_shift_sp(one, &off, &shift,
var, bits, ctx->minfo);
for (i = 0; i < LUT_limit; i++)
fmpz_mpoly_init3(LUT + i, 4, bits, ctx);
/* fill in tree/LUT from B */
for (i = 0; i < Blen; i++)
{
ulong k = (Bexp[N*i + off] >> shift) & mask;
if (k < LUT_limit)
{
d = LUT + k;
}
else
{
d = mpoly_rbtree_ui_lookup(tree, &its_new, k);
if (its_new)
fmpz_mpoly_init3(d, 4, bits, ctx);
}
fmpz_mpoly_fit_length(d, d->length + 1, ctx);
fmpz_set(d->coeffs + d->length, Bcoeff + i);
mpoly_monomial_msub(d->exps + N*d->length, Bexp + N*i, k, one, N);
d->length++;
}
/* clear out tree to A */
Alen = tree->length;
for (i = LUT_limit - 1; i >= 0; i--)
Alen += (LUT[i].length > 0);
fmpz_mpoly_univar_fit_length(A, Alen, ctx);
A->length = 0;
_tree_data_clear_sp(A, tree, mpoly_rbtree_ui_head(tree), ctx);
for (i = LUT_limit - 1; i >= 0; i--)
{
d = LUT + i;
if (d->length > 0)
{
FLINT_ASSERT(A->length < A->alloc);
fmpz_set_si(A->exps + A->length, i);
fmpz_mpoly_swap(A->coeffs + A->length, d, ctx);
A->length++;
}
fmpz_mpoly_clear(d, ctx);
}
mpoly_rbtree_ui_clear(tree);
}
else
{
mpoly_rbtree_fmpz_t tree;
fmpz_mpoly_struct * d;
fmpz_t k;
fmpz_init(k);
mpoly_rbtree_fmpz_init(tree, sizeof(fmpz_mpoly_struct));
off = mpoly_gen_monomial_offset_mp(one, var, bits, ctx->minfo);
/* fill in tree from B */
for (i = 0; i < Blen; i++)
{
fmpz_set_ui_array(k, Bexp + N*i + off, bits/FLINT_BITS);
d = mpoly_rbtree_fmpz_lookup(tree, &its_new, k);
if (its_new)
fmpz_mpoly_init3(d, 4, bits, ctx);
fmpz_mpoly_fit_length(d, d->length + 1, ctx);
fmpz_set(d->coeffs + d->length, Bcoeff + i);
mpoly_monomial_msub_ui_array(d->exps + N*d->length, Bexp + N*i,
Bexp + N*i + off, bits/FLINT_BITS, one, N);
d->length++;
}
/* clear out tree to A */
fmpz_mpoly_univar_fit_length(A, tree->length, ctx);
A->length = 0;
_tree_data_clear_mp(A, tree, mpoly_rbtree_fmpz_head(tree), ctx);
fmpz_clear(k);
mpoly_rbtree_fmpz_clear(tree);
}
flint_free(one);
}
/*
Currently this function does not work if the coefficients depend on "var".
The assertion x->next == NULL would need to be replaced by a loop.
Other asserts would need to be removed as well.
*/
void _fmpz_mpoly_from_univar(fmpz_mpoly_t A, flint_bitcnt_t Abits,
const fmpz_mpoly_univar_t B, slong var, const fmpz_mpoly_ctx_t ctx)
{
slong N = mpoly_words_per_exp(Abits, ctx->minfo);
slong i;
slong next_loc, heap_len = 1;
ulong * cmpmask;
slong total_len;
mpoly_heap_s * heap;
slong Alen;
ulong ** Btexp;
ulong * exp;
ulong * one;
mpoly_heap_t * chain, * x;
TMP_INIT;
if (B->length == 0)
{
fmpz_mpoly_fit_bits(A, Abits, ctx);
A->bits = Abits;
_fmpz_mpoly_set_length(A, 0, ctx);
return;
}
TMP_START;
/* pack everything into Abits */
one = (ulong*) TMP_ALLOC(N*sizeof(ulong));
cmpmask = (ulong*) TMP_ALLOC(N*sizeof(ulong));
Btexp = (ulong **) TMP_ALLOC(B->length*sizeof(ulong*));
total_len = 0;
for (i = 0; i < B->length; i++)
{
fmpz_mpoly_struct * Bi = B->coeffs + i;
total_len += Bi->length;
Btexp[i] = Bi->exps;
if (Abits != Bi->bits)
{
Btexp[i] = (ulong *) flint_malloc(N*Bi->length*sizeof(ulong));
if (!mpoly_repack_monomials(Btexp[i], Abits,
Bi->exps, Bi->bits, Bi->length, ctx->minfo))
{
FLINT_ASSERT(0 && "repack does not fit");
}
}
}
fmpz_mpoly_fit_length(A, total_len, ctx);
fmpz_mpoly_fit_bits(A, Abits, ctx);
A->bits = Abits;
next_loc = B->length + 2;
heap = (mpoly_heap_s *) TMP_ALLOC((B->length + 1)*sizeof(mpoly_heap_s));
exp = (ulong *) TMP_ALLOC(B->length*N*sizeof(ulong));
chain = (mpoly_heap_t *) TMP_ALLOC(B->length*sizeof(mpoly_heap_t));
mpoly_get_cmpmask(cmpmask, N, Abits, ctx->minfo);
Alen = 0;
if (Abits <= FLINT_BITS)
{
mpoly_gen_monomial_sp(one, var, Abits, ctx->minfo);
for (i = 0; i < B->length; i++)
{
FLINT_ASSERT(fmpz_fits_si(B->exps + i));
x = chain + i;
x->i = i;
x->j = 0;
x->next = NULL;
mpoly_monomial_madd(exp + N*x->i, Btexp[x->i] + N*x->j,
fmpz_get_si(B->exps + i), one, N);
_mpoly_heap_insert(heap, exp + N*i, x, &next_loc, &heap_len,
N, cmpmask);
}
while (heap_len > 1)
{
FLINT_ASSERT(Alen < A->alloc);
mpoly_monomial_set(A->exps + N*Alen, heap[1].exp, N);
x = _mpoly_heap_pop(heap, &heap_len, N, cmpmask);
fmpz_set(A->coeffs + Alen, (B->coeffs + x->i)->coeffs + x->j);
Alen++;
FLINT_ASSERT(x->next == NULL);
if (x->j + 1 < (B->coeffs + x->i)->length)
{
FLINT_ASSERT(fmpz_fits_si(B->exps + x->i));
x->j = x->j + 1;
x->next = NULL;
mpoly_monomial_madd(exp + N*x->i, Btexp[x->i] + N*x->j,
fmpz_get_si(B->exps + x->i), one, N);
_mpoly_heap_insert(heap, exp + N*x->i, x, &next_loc, &heap_len,
N, cmpmask);
}
}
}
else
{
mpoly_gen_monomial_offset_mp(one, var, Abits, ctx->minfo);
for (i = 0; i < B->length; i++)
{
x = chain + i;
x->i = i;
x->j = 0;
x->next = NULL;
mpoly_monomial_madd_fmpz(exp + N*x->i, Btexp[x->i] + N*x->j,
B->exps + i, one, N);
_mpoly_heap_insert(heap, exp + N*i, x, &next_loc, &heap_len,
N, cmpmask);
}
while (heap_len > 1)
{
FLINT_ASSERT(Alen < A->alloc);
mpoly_monomial_set(A->exps + N*Alen, heap[1].exp, N);
x = _mpoly_heap_pop(heap, &heap_len, N, cmpmask);
fmpz_set(A->coeffs + Alen, (B->coeffs + x->i)->coeffs + x->j);
Alen++;
FLINT_ASSERT(x->next == NULL);
if (x->j + 1 < (B->coeffs + x->i)->length)
{
x->j = x->j + 1;
x->next = NULL;
mpoly_monomial_madd_fmpz(exp + N*x->i, Btexp[x->i] + N*x->j,
B->exps + x->i, one, N);
_mpoly_heap_insert(heap, exp + N*x->i, x, &next_loc, &heap_len,
N, cmpmask);
}
}
}
_fmpz_mpoly_set_length(A, Alen, ctx);
for (i = 0; i < B->length; i++)
{
if (Btexp[i] != (B->coeffs + i)->exps)
flint_free(Btexp[i]);
}
TMP_END;
}
void fmpz_mpoly_from_univar(fmpz_mpoly_t A, const fmpz_mpoly_univar_t B,
slong var, const fmpz_mpoly_ctx_t ctx)
{
slong n = ctx->minfo->nfields;
flint_bitcnt_t bits;
slong i;
fmpz * gen_fields, * tmp_fields, * max_fields;
TMP_INIT;
if (B->length == 0)
{
fmpz_mpoly_zero(A, ctx);
return;
}
TMP_START;
/* find bits required to represent result */
gen_fields = (fmpz *) TMP_ALLOC(ctx->minfo->nfields*sizeof(fmpz));
tmp_fields = (fmpz *) TMP_ALLOC(ctx->minfo->nfields*sizeof(fmpz));
max_fields = (fmpz *) TMP_ALLOC(ctx->minfo->nfields*sizeof(fmpz));
for (i = 0; i < ctx->minfo->nfields; i++)
{
fmpz_init(gen_fields + i);
fmpz_init(tmp_fields + i);
fmpz_init(max_fields + i);
}
mpoly_gen_fields_fmpz(gen_fields, var, ctx->minfo);
for (i = 0; i < B->length; i++)
{
fmpz_mpoly_struct * Bi = B->coeffs + i;
mpoly_max_fields_fmpz(tmp_fields, Bi->exps, Bi->length, Bi->bits, ctx->minfo);
_fmpz_vec_scalar_addmul_fmpz(tmp_fields, gen_fields, n, B->exps + i);
_fmpz_vec_max_inplace(max_fields, tmp_fields, n);
}
bits = _fmpz_vec_max_bits(max_fields, n);
bits = FLINT_MAX(MPOLY_MIN_BITS, bits + 1);
bits = mpoly_fix_bits(bits, ctx->minfo);
for (i = 0; i < ctx->minfo->nfields; i++)
{
fmpz_clear(gen_fields + i);
fmpz_clear(tmp_fields + i);
fmpz_clear(max_fields + i);
}
TMP_END;
_fmpz_mpoly_from_univar(A, bits, B, var, ctx);
}
#define COEFF(A, i) ((void*)(A->coeffs + (i)*R->elem_size))
static void mpoly_univar_set_fmpz_mpoly_univar(
mpoly_univar_t A,
mpoly_void_ring_t R,
const fmpz_mpoly_univar_t B,
const fmpz_mpoly_ctx_t ctx)
{
slong i;
mpoly_univar_fit_length(A, B->length, R);
A->length = B->length;
for (i = B->length - 1; i >= 0; i--)
{
fmpz_set(A->exps + i, B->exps + i);
fmpz_mpoly_set(COEFF(A, i), B->coeffs + i, ctx);
}
}
static void mpoly_univar_swap_fmpz_mpoly_univar(
mpoly_univar_t A,
mpoly_void_ring_t R,
fmpz_mpoly_univar_t B,
const fmpz_mpoly_ctx_t ctx)
{
slong i;
mpoly_univar_fit_length(A, B->length, R);
fmpz_mpoly_univar_fit_length(B, A->length, ctx);
for (i = FLINT_MAX(A->length, B->length) - 1; i >= 0; i--)
{
fmpz_swap(A->exps + i, B->exps + i);
fmpz_mpoly_swap(COEFF(A, i), B->coeffs + i, ctx);
}
SLONG_SWAP(A->length, B->length);
}
int fmpz_mpoly_univar_pseudo_gcd(
fmpz_mpoly_univar_t gx,
const fmpz_mpoly_univar_t ax,
const fmpz_mpoly_univar_t bx,
const fmpz_mpoly_ctx_t ctx)
{
int success;
mpoly_void_ring_t R;
mpoly_univar_t Ax, Bx, Gx;
mpoly_void_ring_init_fmpz_mpoly_ctx(R, ctx);
mpoly_univar_init(Ax, R);
mpoly_univar_init(Bx, R);
mpoly_univar_init(Gx, R);
mpoly_univar_set_fmpz_mpoly_univar(Ax, R, ax, ctx);
mpoly_univar_set_fmpz_mpoly_univar(Bx, R, bx, ctx);
success = mpoly_univar_pseudo_gcd_ducos(Gx, Ax, Bx, R);
if (success)
mpoly_univar_swap_fmpz_mpoly_univar(Gx, R, gx, ctx);
mpoly_univar_clear(Ax, R);
mpoly_univar_clear(Bx, R);
mpoly_univar_clear(Gx, R);
return success;
}
int fmpz_mpoly_univar_resultant(
fmpz_mpoly_t d,
const fmpz_mpoly_univar_t ax,
const fmpz_mpoly_univar_t bx,
const fmpz_mpoly_ctx_t ctx)
{
int success;
mpoly_void_ring_t R;
mpoly_univar_t Ax, Bx;
mpoly_void_ring_init_fmpz_mpoly_ctx(R, ctx);
mpoly_univar_init(Ax, R);
mpoly_univar_init(Bx, R);
mpoly_univar_set_fmpz_mpoly_univar(Ax, R, ax, ctx);
mpoly_univar_set_fmpz_mpoly_univar(Bx, R, bx, ctx);
success = mpoly_univar_resultant(d, Ax, Bx, R);
mpoly_univar_clear(Ax, R);
mpoly_univar_clear(Bx, R);
return success;
}
int fmpz_mpoly_univar_discriminant(
fmpz_mpoly_t d,
const fmpz_mpoly_univar_t fx,
const fmpz_mpoly_ctx_t ctx)
{
int success;
mpoly_void_ring_t R;
mpoly_univar_t Fx;
mpoly_void_ring_init_fmpz_mpoly_ctx(R, ctx);
mpoly_univar_init(Fx, R);
mpoly_univar_set_fmpz_mpoly_univar(Fx, R, fx, ctx);
success = mpoly_univar_discriminant(d, Fx, R);
mpoly_univar_clear(Fx, R);
return success;
}