]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blob - gdb/testsuite/gdb.threads/hand-call-in-threads.exp
2010-06-08 Michael Snyder <msnyder@vmware.com>
[thirdparty/binutils-gdb.git] / gdb / testsuite / gdb.threads / hand-call-in-threads.exp
1 # Copyright (C) 2004, 2007, 2008, 2010 Free Software Foundation, Inc.
2
3 # This program is free software; you can redistribute it and/or modify
4 # it under the terms of the GNU General Public License as published by
5 # the Free Software Foundation; either version 3 of the License, or
6 # (at your option) any later version.
7 #
8 # This program is distributed in the hope that it will be useful,
9 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # GNU General Public License for more details.
12 #
13 # You should have received a copy of the GNU General Public License
14 # along with this program. If not, see <http://www.gnu.org/licenses/>.
15
16 # Test making hand function calls in multiple threads.
17
18 set NR_THREADS 4
19
20 if $tracelevel then {
21 strace $tracelevel
22 }
23
24 set testfile "hand-call-in-threads"
25 set srcfile ${testfile}.c
26 set binfile ${objdir}/${subdir}/${testfile}
27
28 if {[gdb_compile_pthreads "${srcdir}/${subdir}/${srcfile}" "${binfile}" executable [list debug "incdir=${objdir}" "additional_flags=-DNR_THREADS=$NR_THREADS"]] != "" } {
29 return -1
30 }
31
32 # Some targets can't do function calls, so don't even bother with this
33 # test.
34 if [target_info exists gdb,cannot_call_functions] {
35 setup_xfail "*-*-*" 2416
36 fail "This target can not call functions"
37 continue
38 }
39
40 proc get_dummy_frame_number { } {
41 global gdb_prompt
42
43 gdb_test_multiple "bt" "" {
44 -re "#(\[0-9\]*) *<function called from gdb>.*$gdb_prompt $" {
45 return $expect_out(1,string)
46 }
47 -re "$gdb_prompt $" {
48 return ""
49 }
50 timeout {
51 return ""
52 }
53 }
54 return ""
55 }
56
57 gdb_exit
58 gdb_start
59 gdb_reinitialize_dir $srcdir/$subdir
60 gdb_load ${binfile}
61
62 if { ![runto_main] } {
63 fail "Can't run to main"
64 return 0
65 }
66
67 gdb_test "break all_threads_running" \
68 "Breakpoint 2 at .*: file .*${srcfile}, line .*" \
69 "breakpoint on all_threads_running"
70
71 gdb_test "break hand_call" \
72 "Breakpoint 3 at .*: file .*${srcfile}, line .*" \
73 "breakpoint on hand_call"
74
75 # Run the program and make sure GDB reports that we stopped after
76 # hitting breakpoint 2 in all_threads_running().
77
78 gdb_test "continue" \
79 ".*Breakpoint 2, all_threads_running ().*" \
80 "run to all_threads_running"
81
82 # Before we start making hand function calls, turn on scheduler locking.
83
84 gdb_test_no_output "set scheduler-locking on" "enable scheduler locking"
85 gdb_test "show scheduler-locking" ".* locking scheduler .* is \"on\"." "show scheduler locking on"
86
87 # Now hand-call a function in each thread, having the function
88 # stop without returning.
89
90 # Add one for the main thread.
91 set total_nr_threads [expr $NR_THREADS + 1]
92
93 # Thread numbering in gdb is origin-1, so begin numbering at 1.
94 for { set i 1 } { $i <= $total_nr_threads } { incr i } {
95 set thread_nr $i
96 gdb_test "thread $thread_nr" ".*" \
97 "prepare to make hand call, thread $thread_nr"
98 gdb_test "call hand_call()" "Breakpoint 3, .*" \
99 "hand call, thread $thread_nr"
100 }
101
102 # Now have each hand-called function return.
103
104 # Turn confirmation off for the "return" command.
105 gdb_test_no_output "set confirm off"
106
107 clear_xfail "*-*-*"
108
109 for { set i 1 } { $i <= $total_nr_threads } { incr i } {
110 set thread_nr $i
111 gdb_test "thread $thread_nr" ".*" \
112 "prepare to discard hand call, thread $thread_nr"
113 set frame_number [get_dummy_frame_number]
114 if { "$frame_number" == "" } {
115 fail "dummy stack frame number, thread $thread_nr"
116 # Need something.
117 set frame_number 0
118 } else {
119 pass "dummy stack frame number, thread $thread_nr"
120 }
121 # Pop the dummy frame.
122 gdb_test "frame $frame_number" ".*" "setting frame, thread $thread_nr"
123 gdb_test "return" ".*" "discard hand call, thread $thread_nr"
124 # In case getting the dummy frame number failed, re-enable for next iter.
125 clear_xfail "*-*-*"
126 }
127
128 # Make sure all dummy frames got popped.
129
130 gdb_test_multiple "maint print dummy-frames" "all dummies popped" {
131 -re ".*stack=.*$gdb_prompt $" {
132 fail "all dummies popped"
133 }
134 -re ".*$gdb_prompt $" {
135 pass "all dummies popped"
136 }
137 }
138
139 # Before we resume the full program, turn off scheduler locking.
140 gdb_test_no_output "set scheduler-locking off" "disable scheduler locking"
141 gdb_test "show scheduler-locking" ".* locking scheduler .* is \"off\"." "show scheduler locking off"
142
143 # Continue one last time, the program should exit normally.
144 #
145 # ??? This currently doesn't work because gdb doesn't know how to singlestep
146 # over reported breakpoints that weren't in the last thread to run.
147 # Commented out until then.
148 #
149 # For reference sake ...
150 # An alternative is to manually work around the issue by manually setting
151 # the thread back to the first thread: the program is still at the
152 # all_threads_running breakpoint, which wasn't the last thread to run,
153 # and gdb doesn't know how to singlestep over reported breakpoints that
154 # weren't in the last thread to run.
155 #gdb_test "thread 1" ".*" "set thread to 1, prepare to resume"
156 #
157 #gdb_continue_to_end "hand-call-in-threads"
158
159 return 0