File: isqrt_constexpr.cpp

package info (click to toggle)
primesieve 12.12%2Bds-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 1,952 kB
  • sloc: cpp: 16,515; ansic: 723; sh: 531; makefile: 91
file content (116 lines) | stat: -rw-r--r-- 5,124 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
///
/// @file   isqrt_constexpr.cpp
/// @brief  Test compile time square root function.
///
/// Copyright (C) 2022 Kim Walisch, <kim.walisch@gmail.com>
///
/// This file is distributed under the BSD License. See the COPYING
/// file in the top level directory.
///

#include <primesieve/pmath.hpp>

#include <stdint.h>
#include <limits>
#include <iostream>

#if defined(BAD_ISQRT)

/// The following compile time integer square root function
/// has a recursion depth of O(sqrt(n)). This is very bad, the
/// stack will explose if you try to compute the square root
/// of a number > 10^9. Furthermore constexpr recursion depth
/// is limited by the compiler even more e.g. both GCC and
/// Clang currently limit constexpr recursion depth to 512.
///
/// The nasty thing is that GCC and Clang start calculating
/// constexpr functions at compile time but once the function
/// reaches a recursion depth > 512, GCC and Clang will stop
/// the compile time calculation and instead compute the
/// function at run-time.
///
/// Because of this you cannot easily tell if your constexpr
/// function was evaluated at compile time or at run-time.
/// But you can use static_assert to test whether your constexpr
/// function is calculated at compile time or at runtime.
/// The code below will not compile with the compiler error
/// message telling you the maximum recursion depth has been
/// exceeded.

template <typename T>
constexpr T bad_isqrt(T sq, T dlt, T value)
{
  return sq > value ? (dlt >> 1) - 1
      : bad_isqrt<T>(sq + dlt, dlt + 2, value);
}

template <typename T>
constexpr T bad_isqrt(T value)
{
  return bad_isqrt<T>(1, 3, value);
}

static_assert(bad_isqrt(100000000) == 10000, "bad_isqrt(10^8) failed!");

#endif

int main()
{
  static_assert(ctSqrt(0) == 0, "ctSqrt(0) failed!");
  static_assert(ctSqrt(1) == 1, "ctSqrt(1) failed!");
  static_assert(ctSqrt(2) == 1, "ctSqrt(2) failed!");
  static_assert(ctSqrt(3) == 1, "ctSqrt(3) failed!");
  static_assert(ctSqrt(4) == 2, "ctSqrt(4) failed!");
  static_assert(ctSqrt(5) == 2, "ctSqrt(5) failed!");
  static_assert(ctSqrt(6) == 2, "ctSqrt(6) failed!");
  static_assert(ctSqrt(7) == 2, "ctSqrt(7) failed!");
  static_assert(ctSqrt(8) == 2, "ctSqrt(8) failed!");
  static_assert(ctSqrt(9) == 3, "ctSqrt(9) failed!");
  static_assert(ctSqrt(10) == 3, "ctSqrt(10) failed!");
  static_assert(ctSqrt(11) == 3, "ctSqrt(11) failed!");
  static_assert(ctSqrt(12) == 3, "ctSqrt(12) failed!");
  static_assert(ctSqrt(13) == 3, "ctSqrt(13) failed!");
  static_assert(ctSqrt(14) == 3, "ctSqrt(14) failed!");
  static_assert(ctSqrt(15) == 3, "ctSqrt(15) failed!");
  static_assert(ctSqrt(16) == 4, "ctSqrt(16) failed!");
  static_assert(ctSqrt(17) == 4, "ctSqrt(17) failed!");
  static_assert(ctSqrt(18) == 4, "ctSqrt(18) failed!");
  static_assert(ctSqrt(19) == 4, "ctSqrt(19) failed!");
  static_assert(ctSqrt(20) == 4, "ctSqrt(20) failed!");
  static_assert(ctSqrt(21) == 4, "ctSqrt(21) failed!");
  static_assert(ctSqrt(22) == 4, "ctSqrt(22) failed!");
  static_assert(ctSqrt(23) == 4, "ctSqrt(23) failed!");
  static_assert(ctSqrt(24) == 4, "ctSqrt(24) failed!");
  static_assert(ctSqrt(25) == 5, "ctSqrt(25) failed!");
  static_assert(ctSqrt(26) == 5, "ctSqrt(26) failed!");
  static_assert(ctSqrt(27) == 5, "ctSqrt(27) failed!");
  static_assert(ctSqrt(28) == 5, "ctSqrt(28) failed!");
  static_assert(ctSqrt(29) == 5, "ctSqrt(29) failed!");
  static_assert(ctSqrt(30) == 5, "ctSqrt(30) failed!");
  static_assert(ctSqrt(31) == 5, "ctSqrt(31) failed!");
  static_assert(ctSqrt(32) == 5, "ctSqrt(32) failed!");
  static_assert(ctSqrt(33) == 5, "ctSqrt(33) failed!");
  static_assert(ctSqrt(34) == 5, "ctSqrt(34) failed!");
  static_assert(ctSqrt(35) == 5, "ctSqrt(35) failed!");
  static_assert(ctSqrt(36) == 6, "ctSqrt(36) failed!");
  static_assert(ctSqrt(37) == 6, "ctSqrt(37) failed!");
  static_assert(ctSqrt(38) == 6, "ctSqrt(38) failed!");
  static_assert(ctSqrt(39) == 6, "ctSqrt(39) failed!");

  static_assert(ctSqrt(9223372037000249999ull) == 3037000499ull, "ctSqrt(3037000500^2-1) failed!");
  static_assert(ctSqrt(9223372037000250000ull) == 3037000500ull, "ctSqrt(3037000500^2) failed!");
  static_assert(ctSqrt(9223372037000250001ull) == 3037000500ull, "ctSqrt(3037000500^2+1) failed!");

  static_assert(ctSqrt(std::numeric_limits<int8_t>::max()) == 11, "ctSqrt(2^7-1) failed!");
  static_assert(ctSqrt(std::numeric_limits<uint8_t>::max()) == 15, "ctSqrt(2^8-1) failed!");
  static_assert(ctSqrt(std::numeric_limits<int16_t>::max()) == 181, "ctSqrt(2^15-1) failed!");
  static_assert(ctSqrt(std::numeric_limits<uint16_t>::max()) == 255, "ctSqrt(2^16-1) failed!");
  static_assert(ctSqrt(std::numeric_limits<int32_t>::max()) == 46340, "ctSqrt(2^31-1) failed!");
  static_assert(ctSqrt(std::numeric_limits<uint32_t>::max()) == 65535, "ctSqrt(2^32-1) failed!");
  static_assert(ctSqrt(std::numeric_limits<int64_t>::max()) == 3037000499ll, "ctSqrt(2^63-1) failed!");
  static_assert(ctSqrt(std::numeric_limits<uint64_t>::max()) == 4294967295ull, "ctSqrt(2^64-1) failed!");

  std::cout << "All tests passed successfully!" << std::endl;

  return 0;
}