/*
Copyright (C) 2018 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 .
*/
#ifndef FMPQ_MPOLY_H
#define FMPQ_MPOLY_H
#ifdef FMPQ_MPOLY_INLINES_C
#define FMPQ_MPOLY_INLINE FLINT_DLL
#else
#define FMPQ_MPOLY_INLINE static __inline__
#endif
#undef ulong
#define ulong ulongxx /* interferes with system includes */
#include
#undef ulong
#include
#define ulong mp_limb_t
#include "fmpq_poly.h"
#include "fmpz_mpoly.h"
#ifdef __cplusplus
extern "C" {
#endif
/* Context object ************************************************************/
typedef struct
{
fmpz_mpoly_ctx_t zctx;
} fmpq_mpoly_ctx_struct;
typedef fmpq_mpoly_ctx_struct fmpq_mpoly_ctx_t[1];
FMPQ_MPOLY_INLINE
void fmpq_mpoly_ctx_init(fmpq_mpoly_ctx_t ctx,
slong nvars, const ordering_t ord)
{
fmpz_mpoly_ctx_init(ctx->zctx, nvars, ord);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_ctx_init_rand(fmpq_mpoly_ctx_t ctx,
flint_rand_t state, slong max_nvars)
{
fmpz_mpoly_ctx_init_rand(ctx->zctx, state, max_nvars);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_ctx_clear(fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_ctx_clear(ctx->zctx);
}
FMPQ_MPOLY_INLINE
slong fmpq_mpoly_ctx_nvars(const fmpq_mpoly_ctx_t ctx)
{
return ctx->zctx->minfo->nvars;
}
FMPQ_MPOLY_INLINE
ordering_t fmpq_mpoly_ctx_ord(const fmpq_mpoly_ctx_t ctx)
{
return ctx->zctx->minfo->ord;
}
/* Polynomials over Q ********************************************************/
/*
A polynomial f is represented as
content * zpoly,
where zpoly should have positive leading coefficient and trivial content.
If f is zero, then the representation should have
content = 0 and zpoly = 0
*/
typedef struct
{ /* non zero case: | zero case: */
fmpq_t content; /* positive or negative content | zero */
fmpz_mpoly_t zpoly; /* contentless poly, lc is positive | zero */
} fmpq_mpoly_struct;
typedef fmpq_mpoly_struct fmpq_mpoly_t[1];
FMPQ_MPOLY_INLINE
fmpq * fmpq_mpoly_content_ref(fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
return A->content;
}
FMPQ_MPOLY_INLINE
fmpz_mpoly_struct * fmpq_mpoly_zpoly_ref(fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
return A->zpoly;
}
FMPQ_MPOLY_INLINE
fmpz * fmpq_mpoly_zpoly_term_coeff_ref(fmpq_mpoly_t A, slong i,
const fmpq_mpoly_ctx_t ctx)
{
FLINT_ASSERT(i < A->zpoly->length);
return A->zpoly->coeffs + i;
}
/* Internal type definitions *************************************************/
/*
fmpq_mpoly_univar_t
sparse univariates with multivariate coefficients
*/
typedef struct
{
fmpq_mpoly_struct * coeffs; /* multivariate coefficients */
fmpz * exps;
slong alloc;
slong length;
} fmpq_mpoly_univar_struct;
typedef fmpq_mpoly_univar_struct fmpq_mpoly_univar_t[1];
/* Memory management ********************************************************/
FMPQ_MPOLY_INLINE
void fmpq_mpoly_init(fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
fmpq_init(A->content);
fmpz_mpoly_init(A->zpoly, ctx->zctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_init2(fmpq_mpoly_t A, slong alloc,
const fmpq_mpoly_ctx_t ctx)
{
fmpq_init(A->content);
fmpz_mpoly_init2(A->zpoly, alloc, ctx->zctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_init3(fmpq_mpoly_t A, slong alloc, flint_bitcnt_t bits,
const fmpq_mpoly_ctx_t ctx)
{
fmpq_init(A->content);
fmpz_mpoly_init3(A->zpoly, alloc, bits, ctx->zctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_realloc(fmpq_mpoly_t A, slong alloc,
const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_realloc(A->zpoly, alloc, ctx->zctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_fit_length(fmpq_mpoly_t A, slong len,
const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_fit_length(A->zpoly, len, ctx->zctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_clear(fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_clear(A->zpoly, ctx->zctx);
fmpq_clear(A->content);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_fit_bits(fmpq_mpoly_t A,
flint_bitcnt_t bits, const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_fit_bits(A->zpoly, bits, ctx->zctx);
}
/* Input/output **************************************************************/
FLINT_DLL int fmpq_mpoly_set_str_pretty(fmpq_mpoly_t A, const char * str,
const char ** x, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL char * fmpq_mpoly_get_str_pretty(const fmpq_mpoly_t A,
const char ** x, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_fprint_pretty(FILE * file,
const fmpq_mpoly_t A, const char ** x, const fmpq_mpoly_ctx_t ctx);
FMPQ_MPOLY_INLINE
int fmpq_mpoly_print_pretty(const fmpq_mpoly_t A,
const char ** x, const fmpq_mpoly_ctx_t ctx)
{
return fmpq_mpoly_fprint_pretty(stdout, A, x, ctx);
}
/* Basic manipulation *******************************************************/
FMPQ_MPOLY_INLINE
void fmpq_mpoly_gen(fmpq_mpoly_t A, slong var, const fmpq_mpoly_ctx_t ctx)
{
fmpq_one(A->content);
fmpz_mpoly_gen(A->zpoly, var, ctx->zctx);
}
FMPQ_MPOLY_INLINE
int fmpq_mpoly_is_gen(const fmpq_mpoly_t A,
slong var, const fmpq_mpoly_ctx_t ctx)
{
return fmpq_is_one(A->content)
&& fmpz_mpoly_is_gen(A->zpoly, var, ctx->zctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_set(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_mpoly_ctx_t ctx)
{
fmpq_set(A->content, B->content);
fmpz_mpoly_set(A->zpoly, B->zpoly, ctx->zctx);
}
FMPQ_MPOLY_INLINE
int fmpq_mpoly_equal(const fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_mpoly_ctx_t ctx)
{
return fmpq_equal(A->content, B->content)
&& fmpz_mpoly_equal(A->zpoly, B->zpoly, ctx->zctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_swap(fmpq_mpoly_t A,
fmpq_mpoly_t B, const fmpq_mpoly_ctx_t ctx)
{
fmpq_mpoly_struct t = *A;
*A = *B;
*B = t;
}
/* Constants *****************************************************************/
FMPQ_MPOLY_INLINE
int fmpq_mpoly_is_fmpq(const fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
return fmpz_mpoly_is_fmpz(A->zpoly, ctx->zctx);
}
FLINT_DLL void fmpq_mpoly_get_fmpq(fmpq_t c, const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_set_fmpq(fmpq_mpoly_t A,
const fmpq_t c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_set_fmpz(fmpq_mpoly_t A,
const fmpz_t c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_set_ui(fmpq_mpoly_t A,
ulong c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_set_si(fmpq_mpoly_t A,
slong c, const fmpq_mpoly_ctx_t ctx);
FMPQ_MPOLY_INLINE
void fmpq_mpoly_zero(fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
fmpq_zero(A->content);
fmpz_mpoly_zero(A->zpoly, ctx->zctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_one(fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
fmpq_one(A->content);
fmpz_mpoly_one(A->zpoly, ctx->zctx);
}
FLINT_DLL int fmpq_mpoly_equal_fmpq(const fmpq_mpoly_t A, const fmpq_t c,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_equal_fmpz(const fmpq_mpoly_t A, const fmpz_t c,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_equal_ui(const fmpq_mpoly_t A, ulong c,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_equal_si(const fmpq_mpoly_t A, slong c,
const fmpq_mpoly_ctx_t ctx);
FMPQ_MPOLY_INLINE
int fmpq_mpoly_is_zero(const fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
return A->zpoly->length == WORD(0);
}
FMPQ_MPOLY_INLINE
int fmpq_mpoly_is_one(const fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
return fmpq_is_one(A->content)
&& fmpz_mpoly_is_one(A->zpoly, ctx->zctx);
}
/* Degrees *******************************************************************/
FMPQ_MPOLY_INLINE
int fmpq_mpoly_degrees_fit_si(const fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
return A->zpoly->bits <= FLINT_BITS ? 1
: mpoly_degrees_fit_si(A->zpoly->exps,
A->zpoly->length, A->zpoly->bits,
ctx->zctx->minfo);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_degrees_fmpz(fmpz ** degs, const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx)
{
mpoly_degrees_pfmpz(degs, A->zpoly->exps, A->zpoly->length,
A->zpoly->bits, ctx->zctx->minfo);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_degrees_si(slong * degs, const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx)
{
mpoly_degrees_si(degs, A->zpoly->exps, A->zpoly->length,
A->zpoly->bits, ctx->zctx->minfo);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_degree_fmpz(fmpz_t deg, const fmpq_mpoly_t A, slong var,
const fmpq_mpoly_ctx_t ctx)
{
mpoly_degree_fmpz(deg, A->zpoly->exps, A->zpoly->length,
A->zpoly->bits, var, ctx->zctx->minfo);
}
FMPQ_MPOLY_INLINE
slong fmpq_mpoly_degree_si(const fmpq_mpoly_t A, slong var,
const fmpq_mpoly_ctx_t ctx)
{
return mpoly_degree_si(A->zpoly->exps, A->zpoly->length,
A->zpoly->bits, var, ctx->zctx->minfo);
}
FMPQ_MPOLY_INLINE
int fmpq_mpoly_total_degree_fits_si(const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx)
{
return mpoly_total_degree_fits_si(A->zpoly->exps, A->zpoly->length,
A->zpoly->bits, ctx->zctx->minfo);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_total_degree_fmpz(fmpz_t tdeg, const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx)
{
mpoly_total_degree_fmpz(tdeg, A->zpoly->exps, A->zpoly->length,
A->zpoly->bits, ctx->zctx->minfo);
}
FMPQ_MPOLY_INLINE
slong fmpq_mpoly_total_degree_si(const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx)
{
return mpoly_total_degree_si(A->zpoly->exps, A->zpoly->length,
A->zpoly->bits, ctx->zctx->minfo);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_used_vars(int * used, const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_used_vars(used, A->zpoly, ctx->zctx);
}
/* Coefficients **************************************************************/
FMPQ_MPOLY_INLINE
void fmpq_mpoly_get_denominator(fmpz_t d, const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx)
{
fmpz_set(d, fmpq_denref(A->content));
}
FLINT_DLL int fmpq_mpoly_is_monic(const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_get_coeff_fmpq_monomial(fmpq_t c,
const fmpq_mpoly_t A, const fmpq_mpoly_t M,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_set_coeff_fmpq_monomial(fmpq_mpoly_t A,
const fmpq_t c, const fmpq_mpoly_t M,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void _fmpq_mpoly_set_coeff_fmpq_fmpz(fmpq_mpoly_t A,
const fmpq_t c, const fmpz * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_set_coeff_fmpq_fmpz(fmpq_mpoly_t A,
const fmpq_t c, fmpz * const * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_set_coeff_fmpq_ui(fmpq_mpoly_t A,
const fmpq_t c, const ulong * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void _fmpq_mpoly_get_coeff_fmpq_fmpz(fmpq_t c, const fmpq_mpoly_t A,
const fmpz * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_get_coeff_fmpq_fmpz(fmpq_t c, const fmpq_mpoly_t A,
fmpz * const * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_get_coeff_fmpq_ui(fmpq_t c, const fmpq_mpoly_t A,
const ulong * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_get_coeff_vars_ui(fmpq_mpoly_t C,
const fmpq_mpoly_t A, const slong * vars, const ulong * exps,
slong length, const fmpq_mpoly_ctx_t ctx);
/* conversion ****************************************************************/
FLINT_DLL int fmpq_mpoly_is_fmpq_poly(const fmpq_mpoly_t A,
slong var, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_get_fmpq_poly(fmpq_poly_t A, const fmpq_mpoly_t B,
slong var, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_set_fmpq_poly(fmpq_mpoly_t A, const fmpq_poly_t B,
slong var, const fmpq_mpoly_ctx_t ctx);
/* comparison ****************************************************************/
FLINT_DLL int fmpq_mpoly_cmp(const fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_mpoly_ctx_t ctx);
/* container operations ******************************************************/
FLINT_DLL int fmpq_mpoly_is_canonical(const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx);
FMPQ_MPOLY_INLINE
slong fmpq_mpoly_length(const fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
return A->zpoly->length;
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_resize(fmpq_mpoly_t A, slong new_length,
const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_resize(A->zpoly, new_length, ctx->zctx);
}
FLINT_DLL void fmpq_mpoly_get_term_coeff_fmpq(fmpq_t c, const fmpq_mpoly_t A,
slong i, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_set_term_coeff_fmpq(fmpq_mpoly_t A,
slong i, const fmpq_t c, const fmpq_mpoly_ctx_t ctx);
FMPQ_MPOLY_INLINE
int fmpq_mpoly_term_exp_fits_ui(const fmpq_mpoly_t A,
slong i, const fmpq_mpoly_ctx_t ctx)
{
return A->zpoly->bits <= FLINT_BITS ? 1
: mpoly_term_exp_fits_ui(A->zpoly->exps,
A->zpoly->bits, i, ctx->zctx->minfo);
}
FMPQ_MPOLY_INLINE
int fmpq_mpoly_term_exp_fits_si(const fmpq_mpoly_t A,
slong i, const fmpq_mpoly_ctx_t ctx)
{
return A->zpoly->bits <= FLINT_BITS ? 1
: mpoly_term_exp_fits_si(A->zpoly->exps,
A->zpoly->bits, i, ctx->zctx->minfo);
}
FLINT_DLL void fmpq_mpoly_get_term_exp_fmpz(fmpz ** exps, const fmpq_mpoly_t A,
slong i, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_get_term_exp_ui(ulong * exps, const fmpq_mpoly_t A,
slong i, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_get_term_exp_si(slong * exps, const fmpq_mpoly_t A,
slong i, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL ulong fmpq_mpoly_get_term_var_exp_ui(const fmpq_mpoly_t A,
slong i, slong var, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL slong fmpq_mpoly_get_term_var_exp_si(const fmpq_mpoly_t A,
slong i, slong var, const fmpq_mpoly_ctx_t ctx);
FMPQ_MPOLY_INLINE
void fmpq_mpoly_set_term_exp_fmpz(fmpq_mpoly_t A,
slong i, fmpz * const * exps, const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_set_term_exp_fmpz(A->zpoly, i, exps, ctx->zctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_set_term_exp_ui(fmpq_mpoly_t A,
slong i, const ulong * exps, const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_set_term_exp_ui(A->zpoly, i, exps, ctx->zctx);
}
FLINT_DLL void fmpq_mpoly_get_term(fmpq_mpoly_t M, const fmpq_mpoly_t A,
slong i, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_get_term_monomial(fmpq_mpoly_t M, const fmpq_mpoly_t A,
slong i, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_push_term_fmpq_fmpz(fmpq_mpoly_t A,
const fmpq_t c, fmpz * const * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_push_term_fmpz_fmpz(fmpq_mpoly_t A,
const fmpz_t c, fmpz * const * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_push_term_ui_fmpz(fmpq_mpoly_t A,
ulong c, fmpz * const * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_push_term_si_fmpz(fmpq_mpoly_t A,
slong c, fmpz * const * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_push_term_fmpq_ui(fmpq_mpoly_t A,
const fmpq_t c, const ulong * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_push_term_fmpz_ui(fmpq_mpoly_t A,
const fmpz_t c, const ulong * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_push_term_ui_ui(fmpq_mpoly_t A,
ulong c, const ulong * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_push_term_si_ui(fmpq_mpoly_t A,
slong c, const ulong * exp, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_reduce(fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_reduce_easy(fmpq_mpoly_t A, slong easy_length,
const fmpq_mpoly_ctx_t);
FMPQ_MPOLY_INLINE
void fmpq_mpoly_sort_terms(fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_sort_terms(A->zpoly, ctx->zctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_combine_like_terms(fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_combine_like_terms(A->zpoly, ctx->zctx);
fmpq_mpoly_reduce(A, ctx);
}
FLINT_DLL void fmpq_mpoly_reverse(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_assert_canonical(const fmpq_mpoly_t poly,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void _fmpq_mpoly_push_rescale(fmpq_mpoly_t A,
fmpq_t C, const fmpq_mpoly_ctx_t ctx);
/* Random generation *********************************************************/
FMPQ_MPOLY_INLINE
void fmpq_mpoly_randtest_bounds(fmpq_mpoly_t A, flint_rand_t state,
slong length, flint_bitcnt_t coeff_bits, ulong * exp_bounds,
const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_randtest_bounds(A->zpoly, state,
length, coeff_bits, exp_bounds, ctx->zctx);
fmpq_randtest_not_zero(A->content, state, coeff_bits + 1);
fmpq_mpoly_reduce(A, ctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_randtest_bound(fmpq_mpoly_t A, flint_rand_t state,
slong length, flint_bitcnt_t coeff_bits, ulong exp_bound,
const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_randtest_bound(A->zpoly, state,
length, coeff_bits, exp_bound, ctx->zctx);
fmpq_randtest_not_zero(A->content, state, coeff_bits + 1);
fmpq_mpoly_reduce(A, ctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_randtest_bits(fmpq_mpoly_t A, flint_rand_t state,
slong length, flint_bitcnt_t coeff_bits, flint_bitcnt_t exp_bits,
const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_randtest_bits(A->zpoly, state,
length, coeff_bits, exp_bits, ctx->zctx);
fmpq_randtest_not_zero(A->content, state, coeff_bits + 1);
fmpq_mpoly_reduce(A, ctx);
}
/* Basic arithmetic **********************************************************/
FLINT_DLL void fmpq_mpoly_add_fmpq(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_t c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_add_fmpz(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpz_t c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_add_ui(fmpq_mpoly_t A, const fmpq_mpoly_t B,
ulong c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_add_si(fmpq_mpoly_t A, const fmpq_mpoly_t B,
slong c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_sub_fmpq(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_t c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_sub_fmpz(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpz_t c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_sub_ui(fmpq_mpoly_t A, const fmpq_mpoly_t B,
ulong c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_sub_si(fmpq_mpoly_t A, const fmpq_mpoly_t B,
slong c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_add(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_mpoly_t C, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_sub(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_mpoly_t C, const fmpq_mpoly_ctx_t ctx);
/* Scalar operations *********************************************************/
FMPQ_MPOLY_INLINE
void fmpq_mpoly_neg(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_mpoly_ctx_t ctx)
{
fmpq_neg(A->content, B->content);
fmpz_mpoly_set(A->zpoly, B->zpoly, ctx->zctx);
}
FLINT_DLL void fmpq_mpoly_scalar_mul_fmpq(fmpq_mpoly_t A,
const fmpq_mpoly_t B, const fmpq_t c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_scalar_mul_fmpz(fmpq_mpoly_t A,
const fmpq_mpoly_t B, const fmpz_t c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_scalar_mul_ui(fmpq_mpoly_t A,
const fmpq_mpoly_t B, ulong c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_scalar_mul_si(fmpq_mpoly_t A,
const fmpq_mpoly_t B, slong c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_scalar_div_fmpq(fmpq_mpoly_t A,
const fmpq_mpoly_t B, const fmpq_t c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_scalar_div_fmpz(fmpq_mpoly_t A,
const fmpq_mpoly_t B, const fmpz_t c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_scalar_div_ui(fmpq_mpoly_t A,
const fmpq_mpoly_t B, ulong c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_scalar_div_si(fmpq_mpoly_t A,
const fmpq_mpoly_t B, slong c, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_make_monic(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void _fmpq_mpoly_make_monic_inplace(fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx);
/* Differentiation/Integration ***********************************************/
FLINT_DLL void fmpq_mpoly_derivative(fmpq_mpoly_t A,
const fmpq_mpoly_t B, slong var, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_integral(fmpq_mpoly_t A,
const fmpq_mpoly_t B, slong var, const fmpq_mpoly_ctx_t ctx);
/* Evaluation ****************************************************************/
FLINT_DLL int _fmpq_mpoly_rescale(fmpq_t Acontent, fmpz * Acoeff,
const fmpq_mpoly_t B, const fmpq * scales, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_evaluate_all_fmpq(fmpq_t ev, const fmpq_mpoly_t A,
fmpq * const * vals, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_evaluate_one_fmpq(fmpq_mpoly_t A,
const fmpq_mpoly_t B, slong var, const fmpq_t val,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_compose_fmpq_poly(fmpq_poly_t A,
const fmpq_mpoly_t B, fmpq_poly_struct * const * C,
const fmpq_mpoly_ctx_t ctxB);
FLINT_DLL int fmpq_mpoly_compose_fmpq_mpoly(fmpq_mpoly_t A,
const fmpq_mpoly_t B, fmpq_mpoly_struct * const * C,
const fmpq_mpoly_ctx_t ctxB, const fmpq_mpoly_ctx_t ctxAC);
FLINT_DLL void fmpq_mpoly_compose_fmpq_mpoly_gen(fmpq_mpoly_t A,
const fmpq_mpoly_t B, const slong * c,
const fmpq_mpoly_ctx_t ctxB, const fmpq_mpoly_ctx_t ctxAC);
/* Multiplication ************************************************************/
FLINT_DLL void fmpq_mpoly_mul(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_mpoly_t C, const fmpq_mpoly_ctx_t ctx);
/* Powering ******************************************************************/
FLINT_DLL int fmpq_mpoly_pow_fmpz(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpz_t k, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_pow_ui(fmpq_mpoly_t A, const fmpq_mpoly_t B,
ulong k, const fmpq_mpoly_ctx_t ctx);
/* Division ******************************************************************/
FLINT_DLL int fmpq_mpoly_divides(fmpq_mpoly_t poly1,
const fmpq_mpoly_t poly2, const fmpq_mpoly_t poly3,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_div(fmpq_mpoly_t q,
const fmpq_mpoly_t poly2, const fmpq_mpoly_t poly3,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_divrem(fmpq_mpoly_t q, fmpq_mpoly_t r,
const fmpq_mpoly_t poly2, const fmpq_mpoly_t poly3,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_divrem_ideal(fmpq_mpoly_struct ** q, fmpq_mpoly_t r,
const fmpq_mpoly_t poly2, fmpq_mpoly_struct * const * poly3, slong len,
const fmpq_mpoly_ctx_t ctx);
/* Square root ***************************************************************/
FLINT_DLL int fmpq_mpoly_sqrt(fmpq_mpoly_t Q, const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx);
FMPQ_MPOLY_INLINE
int fmpq_mpoly_is_square(const fmpq_mpoly_t A, const fmpq_mpoly_ctx_t ctx)
{
return fmpz_is_square(fmpq_numref(A->content)) &&
fmpz_is_square(fmpq_denref(A->content)) &&
fmpz_mpoly_is_square(A->zpoly, ctx->zctx);
}
/* GCD ***********************************************************************/
FMPQ_MPOLY_INLINE
void fmpq_mpoly_content(fmpq_t g, const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx)
{
fmpq_abs(g, A->content);
}
FLINT_DLL void fmpq_mpoly_term_content(fmpq_mpoly_t M, const fmpq_mpoly_t A,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_content_vars(fmpq_mpoly_t g, const fmpq_mpoly_t A,
slong * vars, slong vars_length, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_gcd(fmpq_mpoly_t G, const fmpq_mpoly_t A,
const fmpq_mpoly_t B, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_inflate(fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpz * shift, const fmpz * stride, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_gcd_cofactors(fmpq_mpoly_t G, fmpq_mpoly_t Abar,
fmpq_mpoly_t Bbar, const fmpq_mpoly_t A, const fmpq_mpoly_t B,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_gcd_hensel(fmpq_mpoly_t G,
const fmpq_mpoly_t A, const fmpq_mpoly_t B, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_gcd_brown(fmpq_mpoly_t G,
const fmpq_mpoly_t A, const fmpq_mpoly_t B, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_gcd_subresultant(fmpq_mpoly_t G,
const fmpq_mpoly_t A, const fmpq_mpoly_t B, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_gcd_zippel(fmpq_mpoly_t G,
const fmpq_mpoly_t A, const fmpq_mpoly_t B, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_gcd_zippel2(fmpq_mpoly_t G,
const fmpq_mpoly_t A, const fmpq_mpoly_t B, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_resultant(fmpq_mpoly_t R, const fmpq_mpoly_t A,
const fmpq_mpoly_t B, slong var, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_discriminant(fmpq_mpoly_t R, const fmpq_mpoly_t A,
slong var, const fmpq_mpoly_ctx_t ctx);
/******************************************************************************
Internal functions (guaranteed to change without notice)
******************************************************************************/
FLINT_DLL void mpoly_void_ring_init_fmpq_mpoly_ctx(mpoly_void_ring_t R,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL int fmpq_mpoly_repack_bits(fmpq_mpoly_t A, const fmpq_mpoly_t B,
flint_bitcnt_t Abits, const fmpq_mpoly_ctx_t ctx);
/* Univariates ***************************************************************/
FLINT_DLL void fmpq_mpoly_univar_init(fmpq_mpoly_univar_t A,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_univar_clear(fmpq_mpoly_univar_t A,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_univar_fit_length(fmpq_mpoly_univar_t A,
slong length, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_univar_print_pretty(const fmpq_mpoly_univar_t A,
const char ** x, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_univar_assert_canonical(fmpq_mpoly_univar_t A,
const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_to_univar(fmpq_mpoly_univar_t A,
const fmpq_mpoly_t B, slong var, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_from_univar_bits(fmpq_mpoly_t A, flint_bitcnt_t Abits,
const fmpq_mpoly_univar_t B, slong var, const fmpq_mpoly_ctx_t ctx);
FLINT_DLL void fmpq_mpoly_from_univar(fmpq_mpoly_t A,
const fmpq_mpoly_univar_t B, slong var, const fmpq_mpoly_ctx_t ctx);
FMPQ_MPOLY_INLINE
void fmpq_mpoly_univar_swap(fmpq_mpoly_univar_t A, fmpq_mpoly_univar_t B,
const fmpq_mpoly_ctx_t ctx)
{
fmpq_mpoly_univar_struct t = *A;
*A = *B;
*B = t;
}
FMPQ_MPOLY_INLINE
int fmpq_mpoly_univar_degree_fits_si(const fmpq_mpoly_univar_t A,
const fmpq_mpoly_ctx_t ctx)
{
return A->length == 0 || fmpz_fits_si(A->exps + 0);
}
FMPQ_MPOLY_INLINE
slong fmpq_mpoly_univar_length(const fmpq_mpoly_univar_t A,
const fmpq_mpoly_ctx_t ctx)
{
return A->length;
}
FMPQ_MPOLY_INLINE
slong fmpq_mpoly_univar_get_term_exp_si(fmpq_mpoly_univar_t A, slong i,
const fmpq_mpoly_ctx_t ctx)
{
FLINT_ASSERT((ulong)i < (ulong)A->length);
return fmpz_get_si(A->exps + i);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_univar_get_term_coeff(fmpq_mpoly_t c,
const fmpq_mpoly_univar_t A, slong i, const fmpq_mpoly_ctx_t ctx)
{
FLINT_ASSERT((ulong)i < (ulong)A->length);
fmpq_mpoly_set(c, A->coeffs + i, ctx);
}
FMPQ_MPOLY_INLINE
void fmpq_mpoly_univar_swap_term_coeff(fmpq_mpoly_t c,
fmpq_mpoly_univar_t A, slong i, const fmpq_mpoly_ctx_t ctx)
{
FLINT_ASSERT((ulong)i < (ulong)A->length);
fmpq_mpoly_swap(c, A->coeffs + i, ctx);
}
/******************************************************************************
Internal consistency checks
******************************************************************************/
/*
test that r is a valid remainder upon division by g over Q
this means that no term of r is divisible by lt(g)
*/
FMPQ_MPOLY_INLINE
void fmpq_mpoly_remainder_test(const fmpq_mpoly_t r, const fmpq_mpoly_t g,
const fmpq_mpoly_ctx_t ctx)
{
fmpz_mpoly_remainder_strongtest(r->zpoly, g->zpoly, ctx->zctx);
}
#ifdef __cplusplus
}
#endif
#endif