aboutsummaryrefslogtreecommitdiff
path: root/gdb/testsuite/gdb.threads/infcall-thread-announce.c
blob: 90bf95e374dfe0b699d281f4e63108384e2b4f9e (plain)
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
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
/* This testcase is part of GDB, the GNU debugger.

   Copyright 2023-2024 Free Software Foundation, Inc.

   This program is free software; you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation; either version 3 of the License, or
   (at your option) any later version.

   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.

   You should have received a copy of the GNU General Public License
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */

#include <assert.h>
#include <pthread.h>

/* Test can create at most this number of extra threads.  */
#define MAX_THREADS 3

/* For convenience.  */
#define FALSE 0
#define TRUE (!FALSE)

/* Controls a thread created by this test.  */
struct thread_descriptor
{
  /* The pthread handle.  Not valid unless STARTED is true.  */
  pthread_t thr;

  /* This field is set to TRUE when a thread has been created, otherwise,
     is false.  */
  int started;

  /* A condition variable and mutex, used for synchronising between the
     worker thread and the main thread.  */
  pthread_cond_t cond;
  pthread_mutex_t mutex;
};

/* Keep track of worker threads.  */
struct thread_descriptor threads[MAX_THREADS];

/* Worker thread function.  Doesn't do much.  Synchronise with the main
   thread, mark the thread as started, and then block waiting for the main
   thread.  Once the main thread wakes us, this thread exits.

   ARG is a thread_descriptor shared with the main thread.  */

void *
thread_function (void *arg)
{
  int res;
  struct thread_descriptor *thread = (struct thread_descriptor *) arg;

  /* Acquire the thread's lock.  Initially the main thread holds this lock,
     but releases it when the main thread enters a pthread_cond_wait.  */
  res = pthread_mutex_lock (&thread->mutex);
  assert (res == 0);

  /* Mark the thread as started.  */
  thread->started = TRUE;

  /* Signal the main thread to tell it we are started.  The main thread
     will still be blocked though as we hold the thread's lock.  */
  res = pthread_cond_signal (&thread->cond);
  assert (res == 0);

  /* Now wait until the main thread tells us to exit.  By entering this
     pthread_cond_wait we release the lock, which allows the main thread to
     resume.  */
  res = pthread_cond_wait (&thread->cond, &thread->mutex);
  assert (res == 0);

  /* The main thread woke us up.  We reacquired the thread lock as we left
     the pthread_cond_wait, so release the lock now.  */
  res = pthread_mutex_unlock (&thread->mutex);
  assert (res == 0);

  return NULL;
}

/* Start a new thread within the global THREADS array.  Return true if a
   new thread was started, otherwise return false.  */

int
start_thread ()
{
  int idx, res;

  for (idx = 0; idx < MAX_THREADS; ++idx)
    if (!threads[idx].started)
      break;

  if (idx == MAX_THREADS)
    return FALSE;

  /* Acquire the thread lock before starting the new thread.  */
  res = pthread_mutex_lock (&threads[idx].mutex);
  assert (res == 0);

  /* Start the new thread.  */
  res = pthread_create (&threads[idx].thr, NULL,
			thread_function, &threads[idx]);
  assert (res == 0);

  /* Unlock and wait.  The thread signals us once it is ready.  */
  res = pthread_cond_wait (&threads[idx].cond, &threads[idx].mutex);
  assert (res == 0);

  /* The worker thread is now blocked in a pthread_cond_wait and we
     reacquired the lock as we left our own pthread_cond_wait above.  */
  res = pthread_mutex_unlock (&threads[idx].mutex);
  assert (res == 0);

  return TRUE;
}

/* Stop a thread from within the global THREADS array.  Return true if a
   thread was stopped, otherwise return false.  */
int
stop_thread ()
{
  /* Look for a thread that is started.  */
  for (int idx = 0; idx < MAX_THREADS; ++idx)
    if (threads[idx].started)
      {
	int res;

	/* Grab the thread lock.  */
	res = pthread_mutex_lock (&threads[idx].mutex);
	assert (res == 0);

	/* Signal the worker thread, this wakes it up, but it can't exit
	   until it acquires the thread lock, which we currently hold.  */
	res = pthread_cond_signal (&threads[idx].cond);
	assert (res == 0);

	/* Release the thread lock, this allows the worker thread to exit.  */
	res = pthread_mutex_unlock (&threads[idx].mutex);
	assert (res == 0);

	/* Now wait for the thread to exit.  */
	void *retval;
	res = pthread_join (threads[idx].thr, &retval);
	assert (res == 0);
	assert (retval == NULL);

	/* Now the thread has exited, mark it as no longer started.  */
	assert (threads[idx].started);
	threads[idx].started = FALSE;

	return TRUE;
      }

  return FALSE;
}

void
init_descriptor_array ()
{
  for (int i = 0; i < MAX_THREADS; ++i)
    {
      int res;

      threads[i].started = FALSE;
      res = pthread_cond_init (&threads[i].cond, NULL);
      assert (res == 0);
      res = pthread_mutex_init (&threads[i].mutex, NULL);
      assert (res == 0);
    }
}

void
breakpt ()
{
  /* Nothing.  */
}

int
main ()
{
  init_descriptor_array ();
  breakpt ();
  start_thread ();
  stop_thread ();
  breakpt ();
  return 0;
}