blob: 8d9d4cbff76d17d5f54d78c0aa8eb75011ac3175 [file] [log] [blame]
Greg Kroah-Hartmanb2441312017-11-01 15:07:57 +01001// SPDX-License-Identifier: GPL-2.0
Arnaldo Carvalho de Melofd20e812017-04-17 15:23:08 -03002#include <inttypes.h>
Jiri Olsa4e85edf2014-03-05 17:20:31 +01003#include <unistd.h>
4#include <sys/syscall.h>
5#include <sys/types.h>
6#include <sys/mman.h>
7#include <pthread.h>
8#include <stdlib.h>
9#include <stdio.h>
10#include "debug.h"
Arnaldo Carvalho de Meloea49e012019-09-18 11:36:13 -030011#include "event.h"
Jiri Olsa4e85edf2014-03-05 17:20:31 +010012#include "tests.h"
13#include "machine.h"
14#include "thread_map.h"
Arnaldo Carvalho de Melo1101f692019-01-27 13:42:37 +010015#include "map.h"
Jiri Olsa4e85edf2014-03-05 17:20:31 +010016#include "symbol.h"
Arnaldo Carvalho de Meloea49e012019-09-18 11:36:13 -030017#include "util/synthetic-events.h"
Jiri Olsa4e85edf2014-03-05 17:20:31 +010018#include "thread.h"
Jiri Olsa20f2be12019-08-06 15:25:25 +020019#include <internal/lib.h> // page_size
Jiri Olsa4e85edf2014-03-05 17:20:31 +010020
21#define THREADS 4
22
23static int go_away;
24
25struct thread_data {
26 pthread_t pt;
27 pid_t tid;
28 void *map;
29 int ready[2];
30};
31
32static struct thread_data threads[THREADS];
33
34static int thread_init(struct thread_data *td)
35{
36 void *map;
37
38 map = mmap(NULL, page_size,
39 PROT_READ|PROT_WRITE|PROT_EXEC,
40 MAP_SHARED|MAP_ANONYMOUS, -1, 0);
41
42 if (map == MAP_FAILED) {
43 perror("mmap failed");
44 return -1;
45 }
46
47 td->map = map;
48 td->tid = syscall(SYS_gettid);
49
50 pr_debug("tid = %d, map = %p\n", td->tid, map);
51 return 0;
52}
53
54static void *thread_fn(void *arg)
55{
56 struct thread_data *td = arg;
57 ssize_t ret;
Numfor Mbiziwo-Tiapo4e4cf622019-07-02 10:37:15 -070058 int go = 0;
Jiri Olsa4e85edf2014-03-05 17:20:31 +010059
60 if (thread_init(td))
61 return NULL;
62
63 /* Signal thread_create thread is initialized. */
64 ret = write(td->ready[1], &go, sizeof(int));
65 if (ret != sizeof(int)) {
66 pr_err("failed to notify\n");
67 return NULL;
68 }
69
70 while (!go_away) {
71 /* Waiting for main thread to kill us. */
72 usleep(100);
73 }
74
75 munmap(td->map, page_size);
76 return NULL;
77}
78
79static int thread_create(int i)
80{
81 struct thread_data *td = &threads[i];
82 int err, go;
83
84 if (pipe(td->ready))
85 return -1;
86
87 err = pthread_create(&td->pt, NULL, thread_fn, td);
88 if (!err) {
89 /* Wait for thread initialization. */
90 ssize_t ret = read(td->ready[0], &go, sizeof(int));
91 err = ret != sizeof(int);
92 }
93
94 close(td->ready[0]);
95 close(td->ready[1]);
96 return err;
97}
98
99static int threads_create(void)
100{
101 struct thread_data *td0 = &threads[0];
102 int i, err = 0;
103
104 go_away = 0;
105
106 /* 0 is main thread */
107 if (thread_init(td0))
108 return -1;
109
110 for (i = 1; !err && i < THREADS; i++)
111 err = thread_create(i);
112
113 return err;
114}
115
116static int threads_destroy(void)
117{
118 struct thread_data *td0 = &threads[0];
119 int i, err = 0;
120
121 /* cleanup the main thread */
122 munmap(td0->map, page_size);
123
124 go_away = 1;
125
126 for (i = 1; !err && i < THREADS; i++)
127 err = pthread_join(threads[i].pt, NULL);
128
129 return err;
130}
131
132typedef int (*synth_cb)(struct machine *machine);
133
134static int synth_all(struct machine *machine)
135{
136 return perf_event__synthesize_threads(NULL,
137 perf_event__process,
Mark Drayton3fcb10e2018-12-04 12:34:20 -0800138 machine, 0, 1);
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100139}
140
141static int synth_process(struct machine *machine)
142{
Jiri Olsa9749b902019-07-21 13:23:50 +0200143 struct perf_thread_map *map;
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100144 int err;
145
146 map = thread_map__new_by_pid(getpid());
147
148 err = perf_event__synthesize_thread_map(NULL, map,
149 perf_event__process,
Mark Drayton3fcb10e2018-12-04 12:34:20 -0800150 machine, 0);
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100151
Jiri Olsa7836e522019-07-21 13:24:20 +0200152 perf_thread_map__put(map);
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100153 return err;
154}
155
156static int mmap_events(synth_cb synth)
157{
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100158 struct machine *machine;
159 int err, i;
160
161 /*
162 * The threads_create will not return before all threads
163 * are spawned and all created memory map.
164 *
165 * They will loop until threads_destroy is called, so we
166 * can safely run synthesizing function.
167 */
168 TEST_ASSERT_VAL("failed to create threads", !threads_create());
169
Jiri Olsa04684792015-12-03 09:34:13 +0100170 machine = machine__new_host();
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100171
172 dump_trace = verbose > 1 ? 1 : 0;
173
174 err = synth(machine);
175
176 dump_trace = 0;
177
178 TEST_ASSERT_VAL("failed to destroy threads", !threads_destroy());
179 TEST_ASSERT_VAL("failed to synthesize maps", !err);
180
181 /*
182 * All data is synthesized, try to find map for each
183 * thread object.
184 */
185 for (i = 0; i < THREADS; i++) {
186 struct thread_data *td = &threads[i];
187 struct addr_location al;
188 struct thread *thread;
189
190 thread = machine__findnew_thread(machine, getpid(), td->tid);
191
192 pr_debug("looking for map %p\n", td->map);
193
Arnaldo Carvalho de Melof07a2d32018-04-24 10:49:50 -0300194 thread__find_map(thread, PERF_RECORD_MISC_USER,
195 (unsigned long) (td->map + 1), &al);
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100196
Arnaldo Carvalho de Melob91fc392015-04-06 20:43:22 -0300197 thread__put(thread);
198
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100199 if (!al.map) {
200 pr_debug("failed, couldn't find map\n");
201 err = -1;
202 break;
203 }
204
205 pr_debug("map %p, addr %" PRIx64 "\n", al.map, al.map->start);
206 }
207
208 machine__delete_threads(machine);
Jiri Olsa04684792015-12-03 09:34:13 +0100209 machine__delete(machine);
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100210 return err;
211}
212
213/*
214 * This test creates 'THREADS' number of threads (including
215 * main thread) and each thread creates memory map.
216 *
217 * When threads are created, we synthesize them with both
218 * (separate tests):
219 * perf_event__synthesize_thread_map (process based)
220 * perf_event__synthesize_threads (global)
221 *
222 * We test we can find all memory maps via:
Arnaldo Carvalho de Melof07a2d32018-04-24 10:49:50 -0300223 * thread__find_map
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100224 *
225 * by using all thread objects.
226 */
Arnaldo Carvalho de Melo81f17c92017-08-03 15:16:31 -0300227int test__mmap_thread_lookup(struct test *test __maybe_unused, int subtest __maybe_unused)
Jiri Olsa4e85edf2014-03-05 17:20:31 +0100228{
229 /* perf_event__synthesize_threads synthesize */
230 TEST_ASSERT_VAL("failed with sythesizing all",
231 !mmap_events(synth_all));
232
233 /* perf_event__synthesize_thread_map synthesize */
234 TEST_ASSERT_VAL("failed with sythesizing process",
235 !mmap_events(synth_process));
236
237 return 0;
238}