/* * (C) Copyright Nick Thompson 2018. * Use, modification and distribution are subject to the * Boost Software License, Version 1.0. (See accompanying file * LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) */ #if (defined(BOOST_MSVC) && (BOOST_MSVC < 1500)) || \ (defined(__clang_major__) && (__clang_major__ == 3) && (__clang_minor__ < 2)) || \ (defined(BOOST_GCC) && defined(BOOST_GCC_CXX11) && BOOST_GCC < 40800) #define DISABLE_MP_TESTS #endif #ifndef DISABLE_MP_TESTS #include #include #include #include using boost::multiprecision::int128_t; using boost::multiprecision::int256_t; using boost::integer::mod_inverse; using boost::integer::gcd; template void test_mod_inverse() { //Z max_arg = std::numeric_limits::max(); Z max_arg = 500; for (Z modulus = 2; modulus < max_arg; ++modulus) { if (modulus % 1000 == 0) { std::cout << "Testing all inverses modulo " << modulus << std::endl; } for (Z a = 1; a < modulus; ++a) { Z gcdam = gcd(a, modulus); boost::optional inv_a = mod_inverse(a, modulus); // Should fail if gcd(a, mod) != 1: if (gcdam > 1) { BOOST_TEST(!inv_a); } else { BOOST_TEST(inv_a.value() > 0); // Cast to a bigger type so the multiplication won't overflow. int256_t a_inv = inv_a.value(); int256_t big_a = a; int256_t m = modulus; int256_t outta_be_one = (a_inv*big_a) % m; BOOST_TEST_EQ(outta_be_one, 1); } } } } int main() { test_mod_inverse(); test_mod_inverse(); test_mod_inverse(); test_mod_inverse(); return boost::report_errors(); } #else int main() { return 0; } #endif