test.cpp 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. // Copyright Antony Polukhin, 2016-2019.
  2. //
  3. // Distributed under the Boost Software License, Version 1.0. (See
  4. // accompanying file LICENSE_1_0.txt or copy at
  5. // http://www.boost.org/LICENSE_1_0.txt)
  6. #include <boost/stacktrace/stacktrace_fwd.hpp>
  7. #include <boost/stacktrace.hpp>
  8. #include <stdexcept>
  9. #include <iostream>
  10. #include <sstream>
  11. #include <cctype>
  12. #include <boost/core/lightweight_test.hpp>
  13. #include <boost/functional/hash.hpp>
  14. #include "test_impl.hpp"
  15. using boost::stacktrace::stacktrace;
  16. using boost::stacktrace::frame;
  17. #if (defined(BOOST_GCC) && defined(BOOST_WINDOWS) && !defined(BOOST_STACKTRACE_USE_BACKTRACE) && !defined(BOOST_STACKTRACE_USE_ADDR2LINE)) \
  18. || defined(BOOST_STACKTRACE_TEST_NO_DEBUG_AT_ALL)
  19. # define BOOST_STACKTRACE_TEST_SHOULD_OUTPUT_READABLE_NAMES 0
  20. #else
  21. # define BOOST_STACKTRACE_TEST_SHOULD_OUTPUT_READABLE_NAMES 1
  22. #endif
  23. void test_deeply_nested_namespaces() {
  24. std::stringstream ss;
  25. ss << return_from_nested_namespaces();
  26. std::cout << ss.str() << '\n';
  27. #if BOOST_STACKTRACE_TEST_SHOULD_OUTPUT_READABLE_NAMES
  28. BOOST_TEST(ss.str().find("main") != std::string::npos);
  29. BOOST_TEST(ss.str().find("get_backtrace_from_nested_namespaces") != std::string::npos
  30. || ss.str().find("1# return_from_nested_namespaces") != std::string::npos); // GCC with -O1 has strange inlining, so this line is true while the prev one is false.
  31. BOOST_TEST(ss.str().find("return_from_nested_namespaces") != std::string::npos);
  32. #endif
  33. stacktrace ns1 = return_from_nested_namespaces();
  34. BOOST_TEST(ns1 != return_from_nested_namespaces()); // Different addresses in test_deeply_nested_namespaces() function
  35. }
  36. std::size_t count_unprintable_chars(const std::string& s) {
  37. std::size_t result = 0;
  38. for (std::size_t i = 0; i < s.size(); ++i) {
  39. result += (std::isprint(s[i]) ? 0 : 1);
  40. }
  41. return result;
  42. }
  43. void test_frames_string_data_validity() {
  44. stacktrace trace = return_from_nested_namespaces();
  45. for (std::size_t i = 0; i < trace.size(); ++i) {
  46. BOOST_TEST_EQ(count_unprintable_chars(trace[i].source_file()), 0);
  47. BOOST_TEST_EQ(count_unprintable_chars(trace[i].name()), 0);
  48. }
  49. BOOST_TEST(to_string(trace).find('\0') == std::string::npos);
  50. }
  51. // Template parameter Depth is to produce different functions on each Depth. This simplifies debugging when one of the tests catches error
  52. template <std::size_t Depth>
  53. void test_nested(bool print = true) {
  54. std::pair<stacktrace, stacktrace> res = function_from_library(Depth, function_from_main_translation_unit);
  55. std::stringstream ss1, ss2;
  56. ss1 << res.first;
  57. ss2 << res.second;
  58. if (print) {
  59. std::cout << "'" << ss1.str() << "'\n\n" << ss2.str() << std::endl;
  60. }
  61. BOOST_TEST(!ss1.str().empty());
  62. BOOST_TEST(!ss2.str().empty());
  63. BOOST_TEST(ss1.str().find(" 0# ") != std::string::npos);
  64. BOOST_TEST(ss2.str().find(" 0# ") != std::string::npos);
  65. BOOST_TEST(ss1.str().find(" 1# ") != std::string::npos);
  66. BOOST_TEST(ss2.str().find(" 1# ") != std::string::npos);
  67. BOOST_TEST(ss1.str().find(" in ") != std::string::npos);
  68. BOOST_TEST(ss2.str().find(" in ") != std::string::npos);
  69. #if BOOST_STACKTRACE_TEST_SHOULD_OUTPUT_READABLE_NAMES
  70. BOOST_TEST(ss1.str().find("main") != std::string::npos);
  71. BOOST_TEST(ss2.str().find("main") != std::string::npos);
  72. BOOST_TEST(ss1.str().find("function_from_library") != std::string::npos);
  73. BOOST_TEST(ss2.str().find("function_from_library") != std::string::npos);
  74. BOOST_TEST(ss1.str().find("function_from_main_translation_unit") != std::string::npos);
  75. BOOST_TEST(ss2.str().find("function_from_main_translation_unit") != std::string::npos);
  76. #endif
  77. }
  78. template <class Bt>
  79. void test_comparisons_base(Bt nst, Bt st) {
  80. Bt cst(st);
  81. st = st;
  82. cst = cst;
  83. BOOST_TEST(nst);
  84. BOOST_TEST(st);
  85. #if !defined(BOOST_MSVC) && !defined(BOOST_STACKTRACE_USE_WINDBG)
  86. // This is very dependent on compiler and link flags. No sane way to make it work, because
  87. // BOOST_NOINLINE could be ignored by MSVC compiler if link-time optimization is enabled.
  88. BOOST_TEST(nst[0] != st[0]);
  89. #endif
  90. BOOST_TEST(nst != st);
  91. BOOST_TEST(st != nst);
  92. BOOST_TEST(st == st);
  93. BOOST_TEST(nst == nst);
  94. BOOST_TEST(nst != cst);
  95. BOOST_TEST(cst != nst);
  96. BOOST_TEST(cst == st);
  97. BOOST_TEST(cst == cst);
  98. BOOST_TEST(nst < st || nst > st);
  99. BOOST_TEST(st < nst || nst < st);
  100. BOOST_TEST(st <= st);
  101. BOOST_TEST(nst <= nst);
  102. BOOST_TEST(st >= st);
  103. BOOST_TEST(nst >= nst);
  104. BOOST_TEST(nst < cst || cst < nst);
  105. BOOST_TEST(nst > cst || cst > nst);
  106. BOOST_TEST(hash_value(nst) == hash_value(nst));
  107. BOOST_TEST(hash_value(cst) == hash_value(st));
  108. BOOST_TEST(hash_value(nst) != hash_value(cst));
  109. BOOST_TEST(hash_value(st) != hash_value(nst));
  110. }
  111. void test_comparisons() {
  112. stacktrace nst = return_from_nested_namespaces();
  113. stacktrace st;
  114. test_comparisons_base(nst, st);
  115. }
  116. void test_iterators() {
  117. stacktrace st;
  118. BOOST_TEST(st.begin() == st.begin());
  119. BOOST_TEST(st.cbegin() == st.cbegin());
  120. BOOST_TEST(st.crbegin() == st.crbegin());
  121. BOOST_TEST(st.rbegin() == st.rbegin());
  122. BOOST_TEST(st.begin() + 1 == st.begin() + 1);
  123. BOOST_TEST(st.cbegin() + 1 == st.cbegin() + 1);
  124. BOOST_TEST(st.crbegin() + 1 == st.crbegin() + 1);
  125. BOOST_TEST(st.rbegin() + 1 == st.rbegin() + 1);
  126. BOOST_TEST(st.end() == st.end());
  127. BOOST_TEST(st.cend() == st.cend());
  128. BOOST_TEST(st.crend() == st.crend());
  129. BOOST_TEST(st.rend() == st.rend());
  130. BOOST_TEST(st.end() > st.begin());
  131. BOOST_TEST(st.end() > st.cbegin());
  132. BOOST_TEST(st.cend() > st.cbegin());
  133. BOOST_TEST(st.cend() > st.begin());
  134. BOOST_TEST(st.size() == static_cast<std::size_t>(st.end() - st.begin()));
  135. BOOST_TEST(st.size() == static_cast<std::size_t>(st.end() - st.cbegin()));
  136. BOOST_TEST(st.size() == static_cast<std::size_t>(st.cend() - st.cbegin()));
  137. BOOST_TEST(st.size() == static_cast<std::size_t>(st.cend() - st.begin()));
  138. BOOST_TEST(st.size() == static_cast<std::size_t>(std::distance(st.rbegin(), st.rend())));
  139. BOOST_TEST(st.size() == static_cast<std::size_t>(std::distance(st.crbegin(), st.rend())));
  140. BOOST_TEST(st.size() == static_cast<std::size_t>(std::distance(st.crbegin(), st.crend())));
  141. BOOST_TEST(st.size() == static_cast<std::size_t>(std::distance(st.rbegin(), st.crend())));
  142. boost::stacktrace::stacktrace::iterator it = st.begin();
  143. ++ it;
  144. BOOST_TEST(it == st.begin() + 1);
  145. }
  146. void test_frame() {
  147. stacktrace nst = return_from_nested_namespaces();
  148. stacktrace st = make_some_stacktrace1();
  149. const std::size_t min_size = (nst.size() < st.size() ? nst.size() : st.size());
  150. BOOST_TEST(min_size > 2);
  151. for (std::size_t i = 0; i < min_size; ++i) {
  152. BOOST_TEST(st[i] == st[i]);
  153. BOOST_TEST(st[i].source_file() == st[i].source_file());
  154. BOOST_TEST(st[i].source_line() == st[i].source_line());
  155. BOOST_TEST(st[i] <= st[i]);
  156. BOOST_TEST(st[i] >= st[i]);
  157. frame fv = nst[i];
  158. BOOST_TEST(fv);
  159. if (i > 1 && i < min_size - 3) { // Begin ...and end of the trace may match, skipping
  160. BOOST_TEST(st[i] != fv);
  161. #if !(defined(BOOST_STACKTRACE_TEST_NO_DEBUG_AT_ALL) && defined(BOOST_MSVC))
  162. // MSVC can not get function name withhout debug symbols even if it is exported
  163. BOOST_TEST(st[i].name() != fv.name());
  164. BOOST_TEST(st[i] != fv);
  165. BOOST_TEST(st[i] < fv || st[i] > fv);
  166. BOOST_TEST(hash_value(st[i]) != hash_value(fv));
  167. #endif
  168. if (st[i].source_line()) {
  169. BOOST_TEST(st[i].source_file() != fv.source_file() || st[i].source_line() != fv.source_line());
  170. }
  171. BOOST_TEST(st[i]);
  172. }
  173. fv = st[i];
  174. BOOST_TEST(hash_value(st[i]) == hash_value(fv));
  175. }
  176. boost::stacktrace::frame empty_frame;
  177. BOOST_TEST(!empty_frame);
  178. BOOST_TEST_EQ(empty_frame.source_file(), "");
  179. BOOST_TEST_EQ(empty_frame.name(), "");
  180. BOOST_TEST_EQ(empty_frame.source_line(), 0);
  181. }
  182. // Template parameter bool BySkip is to produce different functions on each BySkip. This simplifies debugging when one of the tests catches error
  183. template <bool BySkip>
  184. void test_empty_basic_stacktrace() {
  185. typedef boost::stacktrace::stacktrace st_t;
  186. st_t st = BySkip ? st_t(100500, 1024) : st_t(0, 0);
  187. BOOST_TEST(!st);
  188. BOOST_TEST(st.empty());
  189. BOOST_TEST(st.size() == 0);
  190. BOOST_TEST(st.begin() == st.end());
  191. BOOST_TEST(st.cbegin() == st.end());
  192. BOOST_TEST(st.cbegin() == st.cend());
  193. BOOST_TEST(st.begin() == st.cend());
  194. BOOST_TEST(st.rbegin() == st.rend());
  195. BOOST_TEST(st.crbegin() == st.rend());
  196. BOOST_TEST(st.crbegin() == st.crend());
  197. BOOST_TEST(st.rbegin() == st.crend());
  198. BOOST_TEST(hash_value(st) == hash_value(st_t(0, 0)));
  199. BOOST_TEST(st == st_t(0, 0));
  200. BOOST_TEST(!(st < st_t(0, 0)));
  201. BOOST_TEST(!(st > st_t(0, 0)));
  202. }
  203. int main() {
  204. test_deeply_nested_namespaces();
  205. test_frames_string_data_validity();
  206. test_nested<15>();
  207. test_comparisons();
  208. test_iterators();
  209. test_frame();
  210. test_empty_basic_stacktrace<true>();
  211. test_empty_basic_stacktrace<false>();
  212. BOOST_TEST(&make_some_stacktrace1 != &make_some_stacktrace2);
  213. boost::stacktrace::stacktrace b1 = make_some_stacktrace1();
  214. BOOST_TEST(b1.size() == 4);
  215. boost::stacktrace::stacktrace b2 = make_some_stacktrace2();
  216. BOOST_TEST(b2.size() == 4);
  217. test_comparisons_base(make_some_stacktrace1(), make_some_stacktrace2());
  218. test_nested<260>(false);
  219. BOOST_TEST(boost::stacktrace::stacktrace(0, 1).size() == 1);
  220. BOOST_TEST(boost::stacktrace::stacktrace(1, 1).size() == 1);
  221. return boost::report_errors();
  222. }