/* 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 . */ #include "nmod_mpoly.h" void nmod_mpolyn_init(nmod_mpolyn_t A, flint_bitcnt_t bits, const nmod_mpoly_ctx_t ctx) { A->coeffs = NULL; A->exps = NULL; A->alloc = 0; A->length = 0; A->bits = bits; } void nmod_mpolyn_clear(nmod_mpolyn_t A, const nmod_mpoly_ctx_t ctx) { slong i; for (i = 0; i < A->alloc; i++) n_poly_clear(A->coeffs + i); flint_free(A->coeffs); flint_free(A->exps); } void nmod_mpolyn_swap(nmod_mpolyn_t A, nmod_mpolyn_t B) { nmod_mpolyn_struct t = *A; *A = *B; *B = t; } void nmod_mpolyn_zero(nmod_mpolyn_t A, const nmod_mpoly_ctx_t ctx) { slong i; for (i = 0; i < A->alloc; i++) { n_poly_clear(A->coeffs + i); n_poly_init(A->coeffs + i); } A->length = 0; } int nmod_mpolyn_is_zero(nmod_mpolyn_t A, const nmod_mpoly_ctx_t ctx) { return A->length == 0; } void nmod_mpolyn_print_pretty(const nmod_mpolyn_t A, const char ** x_in, const nmod_mpoly_ctx_t ctx) { n_poly_struct * coeff = A->coeffs; slong len = A->length; ulong * exp = A->exps; slong bits = A->bits; slong i, j, N; fmpz * exponents; char ** x = (char **) x_in; TMP_INIT; if (len == 0) { flint_printf("0"); return; } N = mpoly_words_per_exp(bits, ctx->minfo); TMP_START; if (x == NULL) { x = (char **) TMP_ALLOC(ctx->minfo->nvars*sizeof(char *)); for (i = 0; i < ctx->minfo->nvars; i++) { x[i] = (char *) TMP_ALLOC(((FLINT_BITS+4)/3)*sizeof(char)); flint_sprintf(x[i], "x%wd", i+1); } } exponents = (fmpz *) TMP_ALLOC(ctx->minfo->nvars*sizeof(fmpz)); for (i = 0; i < ctx->minfo->nvars; i++) fmpz_init(exponents + i); for (i = 0; i < len; i++) { if (i > 0) { printf(" + "); } printf("("); n_poly_print_pretty(coeff + i, "v"); printf(")"); mpoly_get_monomial_ffmpz(exponents, exp + N*i, bits, ctx->minfo); for (j = 0; j < ctx->minfo->nvars; j++) { int cmp = fmpz_cmp_ui(exponents + j, WORD(1)); if (cmp > 0) { printf("*%s^", x[j]); fmpz_print(exponents + j); } else if (cmp == 0) { printf("*%s", x[j]); } } } for (i = 0; i < ctx->minfo->nvars; i++) fmpz_clear(exponents + i); TMP_END; } void nmod_mpolyn_fit_length(nmod_mpolyn_t A, slong length, const nmod_mpoly_ctx_t ctx) { slong i; slong old_alloc = A->alloc; slong new_alloc = FLINT_MAX(length, 2*A->alloc); if (length > old_alloc) { slong N = mpoly_words_per_exp(A->bits, ctx->minfo); if (old_alloc == 0) { A->exps = (ulong *) flint_malloc(new_alloc*N*sizeof(ulong)); A->coeffs = (n_poly_struct *) flint_malloc(new_alloc*sizeof(n_poly_struct)); } else { A->exps = (ulong *) flint_realloc(A->exps, new_alloc*N*sizeof(ulong)); A->coeffs = (n_poly_struct *) flint_realloc(A->coeffs, new_alloc*sizeof(n_poly_struct)); } for (i = old_alloc; i < new_alloc; i++) { n_poly_init(A->coeffs + i); } A->alloc = new_alloc; } } void nmod_mpolyn_set_length(nmod_mpolyn_t A, slong newlen, const nmod_mpoly_ctx_t ctx) { if (A->length > newlen) { slong i; for (i = newlen; i < A->length; i++) { n_poly_clear(A->coeffs + i); n_poly_init(A->coeffs + i); } } A->length = newlen; } void nmod_mpolyn_fit_bits(nmod_mpolyn_t A, slong bits, const nmod_mpoly_ctx_t ctx) { slong N; ulong * t; if (A->bits < bits) { if (A->alloc != 0) { N = mpoly_words_per_exp(bits, ctx->minfo); t = flint_malloc(N*A->alloc*sizeof(ulong)); mpoly_repack_monomials(t, bits, A->exps, A->bits, A->length, ctx->minfo); flint_free(A->exps); A->exps = t; } A->bits = bits; } } int nmod_mpolyn_is_canonical(const nmod_mpolyn_t A, const nmod_mpoly_ctx_t ctx) { slong i; if (!mpoly_monomials_valid_test(A->exps, A->length, A->bits, ctx->minfo)) { return 0; } if (mpoly_monomials_overflow_test(A->exps, A->length, A->bits, ctx->minfo)) return 0; if (!mpoly_monomials_inorder_test(A->exps, A->length, A->bits, ctx->minfo)) return 0; for (i = 0; i < A->length; i++) { slong l = (A->coeffs + i)->length; if (l == 0) { return 0; } if ((A->coeffs + i)->coeffs[l - 1] == 0) { return 0; } } return 1; } void nmod_mpolyn_set(nmod_mpolyn_t A, const nmod_mpolyn_t B, const nmod_mpoly_ctx_t ctx) { slong i; n_poly_struct * Acoeff, * Bcoeff; ulong * Aexp, * Bexp; slong Blen; slong N; nmod_mpolyn_fit_bits(A, B->bits, ctx); A->bits = B->bits; Blen = B->length; nmod_mpolyn_fit_length(A, Blen, ctx); Acoeff = A->coeffs; Bcoeff = B->coeffs; Aexp = A->exps; Bexp = B->exps; N = mpoly_words_per_exp(B->bits, ctx->minfo); for (i = 0; i < Blen; i++) { n_poly_set(Acoeff + i, Bcoeff + i); mpoly_monomial_set(Aexp + N*i, Bexp + N*i, N); } /* demote remaining coefficients */ for (i = Blen; i < A->length; i++) { n_poly_clear(Acoeff + i); n_poly_init(Acoeff + i); } A->length = Blen; } void nmod_mpolyn_mul_poly( nmod_mpolyn_t A, const nmod_mpolyn_t B, const n_poly_t c, const nmod_mpoly_ctx_t ctx) { slong i; n_poly_struct * Acoeff, * Bcoeff; ulong * Aexp, * Bexp; slong Blen; slong N; nmod_mpolyn_fit_bits(A, B->bits, ctx); A->bits = B->bits; Blen = B->length; nmod_mpolyn_fit_length(A, Blen, ctx); Acoeff = A->coeffs; Bcoeff = B->coeffs; Aexp = A->exps; Bexp = B->exps; N = mpoly_words_per_exp(B->bits, ctx->minfo); FLINT_ASSERT(!n_poly_is_zero(c)); for (i = 0; i < Blen; i++) { n_poly_mod_mul(Acoeff + i, Bcoeff + i, c, ctx->mod); mpoly_monomial_set(Aexp + N*i, Bexp + N*i, N); } /* demote remaining coefficients */ for (i = Blen; i < A->length; i++) { n_poly_clear(Acoeff + i); n_poly_init(Acoeff + i); } A->length = Blen; }