#ifndef APRCL_H
#define APRCL_H
#include "limb_types.h"
#include "fmpz_types.h"
#include "fmpz_mod_types.h"
#ifdef __cplusplus
extern "C" {
#endif
#define SQUARING_SPACE 70
typedef struct
{
ulong R;
fmpz_t s;
n_factor_t rs;
fmpz_factor_t qs;
int * qs_used;
} _aprcl_config;
typedef _aprcl_config aprcl_config[1];
typedef struct
{
fmpz_mod_poly_t *polys;
ulong p;
ulong q;
fmpz_mod_ctx_t ctx;
} _unity_zpq;
typedef _unity_zpq unity_zpq[1];
typedef struct
{
fmpz_mod_poly_t poly;
ulong p;
ulong exp;
fmpz_mod_ctx_t ctx;
} _unity_zp;
typedef _unity_zp unity_zp[1];
typedef enum
{
UNKNOWN,
PRIME,
COMPOSITE,
PROBABPRIME
} primality_test_status;
int _aprcl_p_ind(const aprcl_config conf, ulong p);
ulong aprcl_p_power_in_q(ulong q, ulong p);
int aprcl_is_mul_coprime_ui_ui(ulong x, ulong y, const fmpz_t n);
int aprcl_is_mul_coprime_ui_fmpz(ulong x, const fmpz_t y, const fmpz_t n);
int aprcl_is_prime(const fmpz_t n);
void aprcl_config_gauss_init(aprcl_config conf, const fmpz_t n);
void aprcl_config_gauss_init_min_R(aprcl_config conf, const fmpz_t n, ulong R);
void aprcl_config_gauss_clear(aprcl_config conf);
ulong aprcl_R_value(const fmpz_t n);
void aprcl_config_jacobi_init(aprcl_config conf, const fmpz_t n);
void aprcl_config_jacobi_clear(aprcl_config conf);
int aprcl_is_prime_gauss(const fmpz_t n);
int aprcl_is_prime_gauss_min_R(const fmpz_t n, ulong R);
primality_test_status _aprcl_is_prime_gauss(const fmpz_t n, const aprcl_config config);
int _aprcl_is_gausspower_2q_equal_first(ulong q, const fmpz_t n);
int _aprcl_is_gausspower_2q_equal_second(ulong q, const fmpz_t n);
slong _aprcl_is_gausspower_from_unity_p(ulong q, ulong r, const fmpz_t n);
int aprcl_is_prime_jacobi(const fmpz_t n);
primality_test_status _aprcl_is_prime_jacobi(const fmpz_t n, const aprcl_config config);
slong _aprcl_is_prime_jacobi_check_pk(const unity_zp j, const fmpz_t u, ulong v);
slong _aprcl_is_prime_jacobi_check_21(ulong q, const fmpz_t n);
slong _aprcl_is_prime_jacobi_check_22(const unity_zp j, const fmpz_t u, ulong v, ulong q);
slong _aprcl_is_prime_jacobi_check_2k(const unity_zp j, const unity_zp j2_1, const unity_zp j2_2, const fmpz_t u, ulong v);
int _aprcl_is_prime_jacobi_additional_test(const fmpz_t n, ulong p);
int aprcl_is_prime_final_division(const fmpz_t n, const fmpz_t s, ulong r);
void unity_zp_init(unity_zp f, ulong p, ulong exp, const fmpz_t n);
void unity_zp_clear(unity_zp f);
void unity_zp_copy(unity_zp f, const unity_zp g);
void unity_zp_swap(unity_zp f, unity_zp g);
void unity_zp_set_zero(unity_zp f);
slong unity_zp_is_unity(unity_zp f);
int unity_zp_equal(unity_zp f, unity_zp g);
void unity_zp_coeff_set_fmpz(unity_zp f, ulong ind, const fmpz_t x);
void unity_zp_coeff_set_ui(unity_zp f, ulong ind, ulong x);
void unity_zp_coeff_add_fmpz(unity_zp f, ulong ind, const fmpz_t x);
void unity_zp_coeff_add_ui(unity_zp f, ulong ind, ulong x);
void unity_zp_coeff_inc(unity_zp f, ulong ind);
void unity_zp_coeff_dec(unity_zp f, ulong ind);
void unity_zp_mul_scalar_ui(unity_zp f, const unity_zp g, ulong s);
void unity_zp_add(unity_zp f, const unity_zp g, const unity_zp h);
void unity_zp_mul(unity_zp f, const unity_zp g, const unity_zp h);
void unity_zp_sqr(unity_zp f, const unity_zp g);
void unity_zp_mul_inplace(unity_zp f, const unity_zp g, const unity_zp h, fmpz_t * t);
void unity_zp_sqr_inplace(unity_zp f, const unity_zp g, fmpz_t * t);
void unity_zp_ar1(fmpz_t * t);
void unity_zp_ar2(fmpz_t * t);
void unity_zp_ar3(fmpz_t * t);
void unity_zp_ar4(fmpz_t * t);
void unity_zp_mul3(unity_zp f, const unity_zp g, const unity_zp h, fmpz_t * t);
void unity_zp_mul4(unity_zp f, const unity_zp g, const unity_zp h, fmpz_t * t);
void unity_zp_mul5(unity_zp f, const unity_zp g, const unity_zp h, fmpz_t * t);
void unity_zp_mul7(unity_zp f, const unity_zp g, const unity_zp h, fmpz_t * t);
void unity_zp_mul8(unity_zp f, const unity_zp g, const unity_zp h, fmpz_t * t);
void unity_zp_mul9(unity_zp f, const unity_zp g, const unity_zp h, fmpz_t * t);
void unity_zp_mul11(unity_zp f, const unity_zp g, const unity_zp h, fmpz_t * t);
void unity_zp_mul16(unity_zp f, const unity_zp g, const unity_zp h, fmpz_t * t);
void unity_zp_sqr3(unity_zp f, const unity_zp g, fmpz_t * t);
void unity_zp_sqr4(unity_zp f, const unity_zp g, fmpz_t * t);
void unity_zp_sqr5(unity_zp f, const unity_zp g, fmpz_t * t);
void unity_zp_sqr7(unity_zp f, const unity_zp g, fmpz_t * t);
void unity_zp_sqr8(unity_zp f, const unity_zp g, fmpz_t * t);
void unity_zp_sqr9(unity_zp f, const unity_zp g, fmpz_t * t);
void unity_zp_sqr11(unity_zp f, const unity_zp g, fmpz_t * t);
void unity_zp_sqr16(unity_zp f, const unity_zp g, fmpz_t * t);
void unity_zp_pow_fmpz(unity_zp f, const unity_zp g, const fmpz_t pow);
void unity_zp_pow_ui(unity_zp f, const unity_zp g, ulong pow);
ulong _unity_zp_pow_select_k(const fmpz_t n);
void unity_zp_pow_2k_fmpz(unity_zp f, const unity_zp g, const fmpz_t pow);
void unity_zp_pow_2k_ui(unity_zp f, const unity_zp g, ulong pow);
void unity_zp_pow_sliding_fmpz(unity_zp f, unity_zp g, const fmpz_t pow);
void _unity_zp_reduce_cyclotomic_divmod(unity_zp f);
void _unity_zp_reduce_cyclotomic(unity_zp f);
void unity_zp_reduce_cyclotomic(unity_zp f, const unity_zp g);
void unity_zp_aut(unity_zp f, const unity_zp g, ulong x);
void unity_zp_aut_inv(unity_zp f, const unity_zp g, ulong x);
nn_ptr aprcl_f_table(const ulong q);
void _unity_zp_jacobi_sum_pq_general(unity_zp f, const nn_ptr table, ulong p, ulong q, ulong k, ulong a, ulong b);
void unity_zp_jacobi_sum_pq(unity_zp f, ulong q, ulong p);
void unity_zp_jacobi_sum_2q_one(unity_zp f, ulong q);
void unity_zp_jacobi_sum_2q_two(unity_zp f, ulong q);
void unity_zpq_init(unity_zpq f, ulong q, ulong p, const fmpz_t n);
void unity_zpq_clear(unity_zpq f);
void unity_zpq_copy(unity_zpq f, const unity_zpq g);
void unity_zpq_swap(unity_zpq f, unity_zpq g);
int unity_zpq_equal(const unity_zpq f, const unity_zpq g);
void unity_zpq_coeff_set_fmpz(unity_zpq f, slong i, slong j, const fmpz_t x);
void unity_zpq_coeff_set_ui(unity_zpq f, slong i, slong j, ulong x);
void unity_zpq_coeff_add(unity_zpq f, slong i, slong j, const fmpz_t x);
void unity_zpq_coeff_add_ui(unity_zpq f, slong i, slong j, ulong x);
void unity_zpq_add(unity_zpq f, const unity_zpq g, const unity_zpq h);
void unity_zpq_mul(unity_zpq f, const unity_zpq g, const unity_zpq h);
void _unity_zpq_mul_unity_p(unity_zpq f);
void unity_zpq_mul_unity_p_pow(unity_zpq f, const unity_zpq g, slong p);
void unity_zpq_pow(unity_zpq f, const unity_zpq g, const fmpz_t p);
void unity_zpq_pow_ui(unity_zpq f, const unity_zpq g, ulong pow);
void unity_zpq_gauss_sum(unity_zpq f, ulong q, ulong p);
void unity_zpq_gauss_sum_character_pow(unity_zpq f, ulong q, ulong p, ulong pow);
void unity_zpq_gauss_sum_sigma_pow(unity_zpq f, ulong q, ulong p);
#ifdef __cplusplus
}
#endif
#endif