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
|
/* This testcase is part of GDB, the GNU debugger.
Copyright 2015 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 <pthread.h>
#ifdef SYMBOL_PREFIX
#define SYMBOL(str) SYMBOL_PREFIX #str
#else
#define SYMBOL(str) #str
#endif
/* Called if the testcase failed. */
static void
fail (void)
{
}
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
/* This function overrides gdb_collect in the in-process agent library.
See gdbserver/tracepoint.c (gdb_collect). We want this function to
be ran instead of the one from the library to easily check that only
one thread is tracing at a time.
This works as expected because GDBserver will ask GDB about symbols
present in the inferior with the 'qSymbol' packet. And GDB will
reply with the address of this function instead of the one from the
in-process agent library. */
void
gdb_agent_gdb_collect (void *tpoint, unsigned char *regs)
{
/* If we cannot acquire a lock, then this means another thread is
tracing and the lock implemented by the jump pad is not working! */
if (pthread_mutex_trylock (&mutex) != 0)
{
fail ();
return;
}
sleep (1);
if (pthread_mutex_unlock (&mutex) != 0)
{
fail ();
return;
}
}
/* Called from asm. */
static void __attribute__((used))
func (void)
{
}
static void *
thread_function (void *arg)
{
/* `set_point' is the label at which to set a fast tracepoint. The
insn at the label must be large enough to fit a fast tracepoint
jump. */
asm (" .global " SYMBOL (set_point) "\n"
SYMBOL (set_point) ":\n"
#if (defined __x86_64__ || defined __i386__)
" call " SYMBOL (func) "\n"
#elif (defined __aarch64__)
" nop\n"
#endif
);
}
static void
end (void)
{
}
int
main (int argc, char *argv[], char *envp[])
{
pthread_t threads[NUM_THREADS];
int i;
for (i = 0; i < NUM_THREADS; i++)
pthread_create (&threads[i], NULL, thread_function, NULL);
for (i = 0; i < NUM_THREADS; i++)
pthread_join (threads[i], NULL);
end ();
return 0;
}
|