gcd_lcm.h 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. // -*- C++ -*-
  2. //===----------------------------------------------------------------------===//
  3. //
  4. // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
  5. // See https://llvm.org/LICENSE.txt for license information.
  6. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  7. //
  8. //===----------------------------------------------------------------------===//
  9. #ifndef _LIBCPP___NUMERIC_GCD_LCM_H
  10. #define _LIBCPP___NUMERIC_GCD_LCM_H
  11. #include <__assert>
  12. #include <__config>
  13. #include <__type_traits/common_type.h>
  14. #include <__type_traits/is_integral.h>
  15. #include <__type_traits/is_same.h>
  16. #include <__type_traits/is_signed.h>
  17. #include <__type_traits/make_unsigned.h>
  18. #include <limits>
  19. #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
  20. # pragma GCC system_header
  21. #endif
  22. _LIBCPP_PUSH_MACROS
  23. #include <__undef_macros>
  24. _LIBCPP_BEGIN_NAMESPACE_STD
  25. #if _LIBCPP_STD_VER >= 17
  26. template <typename _Result, typename _Source, bool _IsSigned = is_signed<_Source>::value>
  27. struct __ct_abs;
  28. template <typename _Result, typename _Source>
  29. struct __ct_abs<_Result, _Source, true> {
  30. _LIBCPP_CONSTEXPR _LIBCPP_HIDE_FROM_ABI _Result operator()(_Source __t) const noexcept {
  31. if (__t >= 0)
  32. return __t;
  33. if (__t == numeric_limits<_Source>::min())
  34. return -static_cast<_Result>(__t);
  35. return -__t;
  36. }
  37. };
  38. template <typename _Result, typename _Source>
  39. struct __ct_abs<_Result, _Source, false> {
  40. _LIBCPP_CONSTEXPR _LIBCPP_HIDE_FROM_ABI _Result operator()(_Source __t) const noexcept { return __t; }
  41. };
  42. template <class _Tp>
  43. _LIBCPP_CONSTEXPR _LIBCPP_HIDDEN _Tp __gcd(_Tp __m, _Tp __n) {
  44. static_assert((!is_signed<_Tp>::value), "");
  45. return __n == 0 ? __m : std::__gcd<_Tp>(__n, __m % __n);
  46. }
  47. template <class _Tp, class _Up>
  48. _LIBCPP_CONSTEXPR _LIBCPP_HIDE_FROM_ABI common_type_t<_Tp, _Up> gcd(_Tp __m, _Up __n) {
  49. static_assert((is_integral<_Tp>::value && is_integral<_Up>::value), "Arguments to gcd must be integer types");
  50. static_assert((!is_same<__remove_cv_t<_Tp>, bool>::value), "First argument to gcd cannot be bool");
  51. static_assert((!is_same<__remove_cv_t<_Up>, bool>::value), "Second argument to gcd cannot be bool");
  52. using _Rp = common_type_t<_Tp, _Up>;
  53. using _Wp = make_unsigned_t<_Rp>;
  54. return static_cast<_Rp>(
  55. std::__gcd(static_cast<_Wp>(__ct_abs<_Rp, _Tp>()(__m)), static_cast<_Wp>(__ct_abs<_Rp, _Up>()(__n))));
  56. }
  57. template <class _Tp, class _Up>
  58. _LIBCPP_CONSTEXPR _LIBCPP_HIDE_FROM_ABI common_type_t<_Tp, _Up> lcm(_Tp __m, _Up __n) {
  59. static_assert((is_integral<_Tp>::value && is_integral<_Up>::value), "Arguments to lcm must be integer types");
  60. static_assert((!is_same<__remove_cv_t<_Tp>, bool>::value), "First argument to lcm cannot be bool");
  61. static_assert((!is_same<__remove_cv_t<_Up>, bool>::value), "Second argument to lcm cannot be bool");
  62. if (__m == 0 || __n == 0)
  63. return 0;
  64. using _Rp = common_type_t<_Tp, _Up>;
  65. _Rp __val1 = __ct_abs<_Rp, _Tp>()(__m) / std::gcd(__m, __n);
  66. _Rp __val2 = __ct_abs<_Rp, _Up>()(__n);
  67. _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN((numeric_limits<_Rp>::max() / __val1 > __val2), "Overflow in lcm");
  68. return __val1 * __val2;
  69. }
  70. #endif // _LIBCPP_STD_VER
  71. _LIBCPP_END_NAMESPACE_STD
  72. _LIBCPP_POP_MACROS
  73. #endif // _LIBCPP___NUMERIC_GCD_LCM_H