/* This file is public domain. Author: Fredrik Johansson. */ #include #include "acb.h" #include "acb_poly.h" #include "arith.h" #include "profiler.h" int check_accuracy(acb_ptr vec, slong len, slong prec) { slong i; for (i = 0; i < len; i++) { if (mag_cmp_2exp_si(arb_radref(acb_realref(vec + i)), -prec) >= 0 || mag_cmp_2exp_si(arb_radref(acb_imagref(vec + i)), -prec) >= 0) return 0; } return 1; } void poly_roots(const fmpz_poly_t poly, slong initial_prec, slong target_prec, slong print_digits) { slong i, prec, deg, isolated, maxiter; acb_poly_t cpoly; acb_ptr roots; deg = poly->length - 1; acb_poly_init(cpoly); roots = _acb_vec_init(deg); for (prec = initial_prec; ; prec *= 2) { acb_poly_set_fmpz_poly(cpoly, poly, prec); maxiter = FLINT_MIN(FLINT_MAX(deg, 32), prec); TIMEIT_ONCE_START flint_printf("prec=%wd: ", prec); isolated = acb_poly_find_roots(roots, cpoly, prec == initial_prec ? NULL : roots, maxiter, prec); flint_printf("%wd isolated roots | ", isolated); TIMEIT_ONCE_STOP if (isolated == deg && check_accuracy(roots, deg, target_prec)) { flint_printf("done!\n"); break; } } if (print_digits != 0) { _acb_vec_sort_pretty(roots, deg); for (i = 0; i < deg; i++) { acb_printd(roots + i, print_digits); flint_printf("\n"); } } acb_poly_clear(cpoly); _acb_vec_clear(roots, deg); } int main(int argc, char *argv[]) { fmpz_poly_t f; fmpz_t t; slong compd, printd, i, j; if (argc < 2) { flint_printf("poly_roots2 [-refine d] [-print d] \n\n"); flint_printf("Isolates all the complex roots of a polynomial with\n"); flint_printf("integer coefficients. For convergence, the input polynomial\n"); flint_printf("is required to be squarefree.\n\n"); flint_printf("If -refine d is passed, the roots are refined to an absolute\n"); flint_printf("tolerance better than 10^(-d). By default, the roots are only\n"); flint_printf("computed to sufficient accuracy to isolate them.\n"); flint_printf("The refinement is not currently done efficiently.\n\n"); flint_printf("If -print d is passed, the computed roots are printed to\n"); flint_printf("d decimals. By default, the roots are not printed.\n\n"); flint_printf("The polynomial can be specified by passing the following as :\n\n"); flint_printf("a Easy polynomial 1 + 2x + ... + (n+1)x^n\n"); flint_printf("t Chebyshev polynomial T_n\n"); flint_printf("u Chebyshev polynomial U_n\n"); flint_printf("p Legendre polynomial P_n\n"); flint_printf("c Cyclotomic polynomial Phi_n\n"); flint_printf("s Swinnerton-Dyer polynomial S_n\n"); flint_printf("b Bernoulli polynomial B_n\n"); flint_printf("w Wilkinson polynomial W_n\n"); flint_printf("e Taylor series of exp(x) truncated to degree n\n"); flint_printf("m The Mignotte-like polynomial x^n + (100x+1)^m, n > m\n"); flint_printf("c0 c1 ... cn c0 + c1 x + ... + cn x^n where all c:s are specified integers\n"); return 1; } compd = 0; printd = 0; j = 0; fmpz_poly_init(f); fmpz_init(t); for (i = 1; i < argc; i++) { if (!strcmp(argv[i], "-refine")) { compd = atol(argv[i+1]); i++; } else if (!strcmp(argv[i], "-print")) { printd = atol(argv[i+1]); i++; } else if (!strcmp(argv[i], "a")) { slong n = atol(argv[i+1]); for (j = 0; j <= n; j++) fmpz_poly_set_coeff_ui(f, j, j+1); break; } else if (!strcmp(argv[i], "t")) { arith_chebyshev_t_polynomial(f, atol(argv[i+1])); break; } else if (!strcmp(argv[i], "u")) { arith_chebyshev_u_polynomial(f, atol(argv[i+1])); break; } else if (!strcmp(argv[i], "p")) { fmpq_poly_t g; fmpq_poly_init(g); arith_legendre_polynomial(g, atol(argv[i+1])); fmpq_poly_get_numerator(f, g); fmpq_poly_clear(g); break; } else if (!strcmp(argv[i], "c")) { arith_cyclotomic_polynomial(f, atol(argv[i+1])); break; } else if (!strcmp(argv[i], "s")) { arith_swinnerton_dyer_polynomial(f, atol(argv[i+1])); break; } else if (!strcmp(argv[i], "b")) { fmpq_poly_t g; fmpq_poly_init(g); arith_bernoulli_polynomial(g, atol(argv[i+1])); fmpq_poly_get_numerator(f, g); fmpq_poly_clear(g); break; } else if (!strcmp(argv[i], "w")) { slong n = atol(argv[i+1]); fmpz_poly_fit_length(f, n+2); arith_stirling_number_1_vec(f->coeffs, n+1, n+2); _fmpz_poly_set_length(f, n+2); fmpz_poly_shift_right(f, f, 1); break; } else if (!strcmp(argv[i], "e")) { fmpq_poly_t g; fmpq_poly_init(g); fmpq_poly_set_coeff_si(g, 0, 0); fmpq_poly_set_coeff_si(g, 1, 1); fmpq_poly_exp_series(g, g, atol(argv[i+1]) + 1); fmpq_poly_get_numerator(f, g); fmpq_poly_clear(g); break; } else if (!strcmp(argv[i], "m")) { fmpz_poly_set_coeff_ui(f, 0, 1); fmpz_poly_set_coeff_ui(f, 1, 100); fmpz_poly_pow(f, f, atol(argv[i+2])); fmpz_poly_set_coeff_ui(f, atol(argv[i+1]), 1); break; } else { fmpz_set_str(t, argv[i], 10); fmpz_poly_set_coeff_fmpz(f, j, t); j++; } } TIMEIT_ONCE_START poly_roots(f, 32, compd * 3.32193 + 2, printd); TIMEIT_ONCE_STOP fmpz_poly_clear(f); fmpz_clear(t); flint_cleanup(); return EXIT_SUCCESS; }