Add translations for various sub-directories
[binutils-gdb.git] / gdb / testsuite / gdb.threads / attach-many-short-lived-threads.exp
blob8883485004294689e601d4124813fb8027e4210a
1 # Copyright 2008-2024 Free Software Foundation, Inc.
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.
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.
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/>.
16 # Test attaching to a program that is constantly spawning short-lived
17 # threads.  The stresses the edge cases of attaching to threads that
18 # have just been created or are in process of dying.  In addition, the
19 # test attaches, debugs, detaches, reattaches in a loop a few times,
20 # to stress the behavior of the debug API around detach (some systems
21 # end up leaving stale state behind that confuse the following
22 # attach).
24 # Return true if the running version of DejaGnu is known to not be
25 # able to run this test.
26 proc bad_dejagnu {} {
27     set dj_ver [dejagnu_version]
28     set dj_ver_major [lindex $dj_ver 0]
29     set dj_ver_minor [lindex $dj_ver 1]
31     # DejaGnu versions prior to 1.6 manage to kill the wrong process
32     # due to PID-reuse races.  Since this test spawns many threads, it
33     # widens the race window a whole lot, enough that the inferior is
34     # often killed, and thus the test randomly fails.  See:
35     # http://lists.gnu.org/archive/html/dejagnu/2015-07/msg00005.html
36     # The fix added a close_wait_program procedure.  If that procedure
37     # is defined, and DejaGnu is older than 1.6, assume that means the
38     # fix was backported.
39     if {$dj_ver_major == 1
40         && ($dj_ver_minor < 6 && [info procs close_wait_program] == "")} {
41         return 1
42     }
44     return 0
47 if {[bad_dejagnu]} {
48     unsupported "broken DejaGnu"
49     return 0
52 require can_spawn_for_attach
54 standard_testfile
56 # The test proper.  See description above.
58 proc test {} {
59     global binfile
60     global gdb_prompt
61     global decimal
63     set test_spawn_id [spawn_wait_for_attach $binfile]
64     set testpid [spawn_id_get_pid $test_spawn_id]
66     set attempts 10
67     for {set attempt 1} { $attempt <= $attempts } { incr attempt } {
68         with_test_prefix "iter $attempt" {
69             set attached 0
70             set eperm 0
71             set test "attach"
72             gdb_test_multiple "attach $testpid" $test {
73                 -re "new threads in iteration" {
74                     # Seen when "set debug libthread_db" is on.
75                     exp_continue
76                 }
77                 -re "Cannot attach to lwp $decimal: Operation not permitted" {
78                     # On Linux, PTRACE_ATTACH sometimes fails with
79                     # EPERM, even though /proc/PID/status indicates
80                     # the thread is running.
81                     set eperm 1
82                     exp_continue
83                 }
84                 -re "debugger service failed.*$gdb_prompt $" {
85                     fail $test
86                 }
87                 -re "$gdb_prompt $" {
88                     if {$eperm} {
89                         xfail "$test (EPERM)"
90                     } else {
91                         pass $test
92                     }
93                 }
94                 -re "Attaching to program.*process $testpid.*$gdb_prompt $" {
95                     pass $test
96                 }
97             }
99             if { $eperm } {
100                 continue
101             }
103             # Sleep a bit and try updating the thread list.  We should
104             # know about all threads already at this point.  If we see
105             # "New Thread" or similar being output, then "attach" is
106             # failing to actually attach to all threads in the process,
107             # which would be a bug.
108             sleep 1
110             set test "no new threads"
111             set status 1
112             gdb_test_multiple "info threads" $test -lbl {
113                 -re "\r\n\[^\r\n\]*New " {
114                     set status 0
115                     exp_continue
116                 }
117                 -re -wrap "" {
118                     if { $status == 1 } {
119                         pass $gdb_test_name
120                     } else {
121                         fail $gdb_test_name
122                     }
123                 }
124             }
126             # Force breakpoints always inserted, so that threads we might
127             # have failed to attach to hit them even when threads we do
128             # know about are stopped.
129             gdb_test_no_output "set breakpoint always-inserted on"
131             # Run to a breakpoint a few times.  A few threads should spawn
132             # and die meanwhile.  This checks that thread creation/death
133             # events carry on correctly after attaching.  Also, be
134             # detaching from the program and reattaching, we check that
135             # the program doesn't die due to gdb leaving a pending
136             # breakpoint hit on a new thread unprocessed.
137             gdb_test "break break_fn" "Breakpoint.*"
139             # Wait a bit, to give time for most threads to hit the
140             # breakpoint, including threads we might have failed to
141             # attach.
142             sleep 2
144             set re_pr26286 \
145                 [multi_line \
146                      [string_to_regexp \
147                           "Program terminated with signal SIGTRAP, Trace/breakpoint trap."] \
148                      [string_to_regexp \
149                           "The program no longer exists."]]
151             set bps 3
152             set exited 0
153             for {set bp 1} { $bp <= $bps } { incr bp } {
154                 gdb_test_multiple "continue" "break at break_fn: $bp" {
155                     -re -wrap "$re_pr26286" {
156                         kfail threads/26286 $gdb_test_name
157                         set exited 1
158                     }
159                     -re -wrap "Breakpoint.*" {
160                         pass $gdb_test_name
161                     }
162                 }
163                 if { $exited } {
164                     break
165                 }
166             }
168             if { $exited } {
169                 return
170             }
172             if {$attempt < $attempts} {
173                 # Kick the time out timer for another round.
174                 gdb_test "print again = 1" " = 1" "reset timer in the inferior"
175                 # Show the time we had left in the logs, in case
176                 # something goes wrong.
177                 gdb_test "print seconds_left" " = .*"
179                 gdb_test "detach" "Detaching from.*"
180             } else {
181                 gdb_test "kill" "" "kill process" "Kill the program being debugged.*y or n. $" "y"
182             }
184             gdb_test_no_output "set breakpoint always-inserted off"
185             delete_breakpoints
186         }
187     }
188     kill_wait_spawned_process $test_spawn_id
191 # The test program exits after a while, in case GDB crashes.  Make it
192 # wait at least as long as we may wait before declaring a time out
193 # failure.
194 set options { "additional_flags=-DTIMEOUT=$timeout" debug pthreads }
196 if {[prepare_for_testing "failed to prepare" $testfile $srcfile \
197          $options] == -1} {
198     return -1
201 test