File: backtrace_test.cpp

package info (click to toggle)
passenger 2.2.11debian-2
  • links: PTS, VCS
  • area: main
  • in suites: squeeze
  • size: 11,576 kB
  • ctags: 28,138
  • sloc: cpp: 66,323; ruby: 9,646; ansic: 2,425; python: 141; sh: 56; makefile: 29
file content (128 lines) | stat: -rw-r--r-- 2,589 bytes parent folder | download | duplicates (2)
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
117
118
119
120
121
122
123
124
125
126
127
128
#include "tut.h"
#include <oxt/backtrace.hpp>
#include <oxt/tracable_exception.hpp>
#include <oxt/thread.hpp>

using namespace oxt;
using namespace std;

namespace tut {
	struct backtrace_test {
	};
	
	DEFINE_TEST_GROUP(backtrace_test);

	TEST_METHOD(1) {
		// Test TRACE_POINT() and tracable_exception.
		struct {
			void foo() {
				TRACE_POINT();
				bar();
			}
			
			void bar() {
				TRACE_POINT();
				baz();
			}
			
			void baz() {
				TRACE_POINT();
				throw tracable_exception();
			}
		} object;
		
		try {
			object.foo();
			fail("tracable_exception expected.");
		} catch (const tracable_exception &e) {
			ensure("Backtrace contains foo()",
				e.backtrace().find("foo()") != string::npos);
			ensure("Backtrace contains bar()",
				e.backtrace().find("bar()") != string::npos);
			ensure("Backtrace contains baz()",
				e.backtrace().find("baz()") != string::npos);
		}
	}
	
	struct QuitEvent {
		bool is_done;
		boost::mutex mutex;
		boost::condition_variable cond;
		
		QuitEvent() {
			is_done = false;
		}
		
		void wait() {
			boost::unique_lock<boost::mutex> l(mutex);
			while (!is_done) {
				cond.wait(l);
			}
		}
		
		void done() {
			is_done = true;
			cond.notify_all();
		}
	};
	
	struct FooCaller {
		QuitEvent *quit_event;
		
		static void foo(QuitEvent *quit_event) {
			TRACE_POINT();
			quit_event->wait();
		}
		
		void operator()() {
			foo(quit_event);
		}
	};
	
	struct BarCaller {
		QuitEvent *quit_event;
		
		static void bar(QuitEvent *quit_event) {
			TRACE_POINT();
			quit_event->wait();
		}
		
		void operator()() {
			bar(quit_event);
		}
	};
	
	TEST_METHOD(2) {
		// Test whether oxt::thread's backtrace support works.
		FooCaller foo;
		QuitEvent foo_quit;
		foo.quit_event = &foo_quit;
		
		BarCaller bar;
		QuitEvent bar_quit;
		bar.quit_event = &bar_quit;
		
		oxt::thread foo_thread(foo);
		oxt::thread bar_thread(bar);
		usleep(20000);
		
		ensure("Foo thread's backtrace contains foo()",
			foo_thread.backtrace().find("foo") != string::npos);
		ensure("Foo thread's backtrace doesn't contain bar()",
			foo_thread.backtrace().find("bar") == string::npos);
		ensure("Bar thread's backtrace contains bar()",
			bar_thread.backtrace().find("bar") != string::npos);
		ensure("Bar thread's backtrace doesn't contain foo()",
			bar_thread.backtrace().find("foo") == string::npos);
		
		string all_backtraces(oxt::thread::all_backtraces());
		ensure(all_backtraces.find("foo") != string::npos);
		ensure(all_backtraces.find("bar") != string::npos);
		
		foo_quit.done();
		bar_quit.done();
		foo_thread.join();
		bar_thread.join();
	}
}