]>
Commit | Line | Data |
---|---|---|
6baa0a5a FW |
1 | #include "../perf.h" |
2 | #include <stdlib.h> | |
3 | #include <stdio.h> | |
4 | #include <string.h> | |
b3165f41 | 5 | #include "session.h" |
6baa0a5a | 6 | #include "thread.h" |
00447ccd | 7 | #include "thread-stack.h" |
6baa0a5a | 8 | #include "util.h" |
6e086437 | 9 | #include "debug.h" |
1902efe7 | 10 | #include "comm.h" |
66f066d8 | 11 | #include "unwind.h" |
6baa0a5a | 12 | |
cddcef60 JO |
13 | int thread__init_map_groups(struct thread *thread, struct machine *machine) |
14 | { | |
15 | struct thread *leader; | |
16 | pid_t pid = thread->pid_; | |
17 | ||
1fcb8768 | 18 | if (pid == thread->tid || pid == -1) { |
11246c70 | 19 | thread->mg = map_groups__new(machine); |
cddcef60 | 20 | } else { |
b91fc39f | 21 | leader = __machine__findnew_thread(machine, pid, pid); |
cddcef60 JO |
22 | if (leader) |
23 | thread->mg = map_groups__get(leader->mg); | |
24 | } | |
25 | ||
26 | return thread->mg ? 0 : -1; | |
27 | } | |
28 | ||
99d725fc | 29 | struct thread *thread__new(pid_t pid, pid_t tid) |
6baa0a5a | 30 | { |
1902efe7 FW |
31 | char *comm_str; |
32 | struct comm *comm; | |
c824c433 | 33 | struct thread *thread = zalloc(sizeof(*thread)); |
6baa0a5a | 34 | |
c824c433 | 35 | if (thread != NULL) { |
c824c433 ACM |
36 | thread->pid_ = pid; |
37 | thread->tid = tid; | |
38 | thread->ppid = -1; | |
bf49c35f | 39 | thread->cpu = -1; |
1902efe7 FW |
40 | INIT_LIST_HEAD(&thread->comm_list); |
41 | ||
66f066d8 NK |
42 | if (unwind__prepare_access(thread) < 0) |
43 | goto err_thread; | |
44 | ||
1902efe7 FW |
45 | comm_str = malloc(32); |
46 | if (!comm_str) | |
47 | goto err_thread; | |
48 | ||
49 | snprintf(comm_str, 32, ":%d", tid); | |
65de51f9 | 50 | comm = comm__new(comm_str, 0, false); |
1902efe7 FW |
51 | free(comm_str); |
52 | if (!comm) | |
53 | goto err_thread; | |
54 | ||
55 | list_add(&comm->list, &thread->comm_list); | |
e1ed3a5b | 56 | atomic_set(&thread->refcnt, 0); |
b91fc39f ACM |
57 | INIT_LIST_HEAD(&thread->node); |
58 | RB_CLEAR_NODE(&thread->rb_node); | |
6baa0a5a FW |
59 | } |
60 | ||
c824c433 | 61 | return thread; |
1902efe7 FW |
62 | |
63 | err_thread: | |
64 | free(thread); | |
65 | return NULL; | |
6baa0a5a FW |
66 | } |
67 | ||
c824c433 | 68 | void thread__delete(struct thread *thread) |
591765fd | 69 | { |
1902efe7 FW |
70 | struct comm *comm, *tmp; |
71 | ||
b91fc39f ACM |
72 | BUG_ON(!RB_EMPTY_NODE(&thread->rb_node)); |
73 | BUG_ON(!list_empty(&thread->node)); | |
74 | ||
00447ccd AH |
75 | thread_stack__free(thread); |
76 | ||
9608b84e AH |
77 | if (thread->mg) { |
78 | map_groups__put(thread->mg); | |
79 | thread->mg = NULL; | |
80 | } | |
1902efe7 FW |
81 | list_for_each_entry_safe(comm, tmp, &thread->comm_list, list) { |
82 | list_del(&comm->list); | |
83 | comm__free(comm); | |
84 | } | |
66f066d8 | 85 | unwind__finish_access(thread); |
1902efe7 | 86 | |
c824c433 | 87 | free(thread); |
591765fd ACM |
88 | } |
89 | ||
f3b623b8 ACM |
90 | struct thread *thread__get(struct thread *thread) |
91 | { | |
b91fc39f ACM |
92 | if (thread) |
93 | atomic_inc(&thread->refcnt); | |
f3b623b8 ACM |
94 | return thread; |
95 | } | |
96 | ||
97 | void thread__put(struct thread *thread) | |
98 | { | |
e1ed3a5b | 99 | if (thread && atomic_dec_and_test(&thread->refcnt)) { |
f3b623b8 ACM |
100 | list_del_init(&thread->node); |
101 | thread__delete(thread); | |
102 | } | |
103 | } | |
104 | ||
4dfced35 | 105 | struct comm *thread__comm(const struct thread *thread) |
6baa0a5a | 106 | { |
1902efe7 FW |
107 | if (list_empty(&thread->comm_list)) |
108 | return NULL; | |
4385d580 | 109 | |
1902efe7 FW |
110 | return list_first_entry(&thread->comm_list, struct comm, list); |
111 | } | |
112 | ||
65de51f9 AH |
113 | struct comm *thread__exec_comm(const struct thread *thread) |
114 | { | |
115 | struct comm *comm, *last = NULL; | |
116 | ||
117 | list_for_each_entry(comm, &thread->comm_list, list) { | |
118 | if (comm->exec) | |
119 | return comm; | |
120 | last = comm; | |
121 | } | |
122 | ||
123 | return last; | |
124 | } | |
125 | ||
65de51f9 AH |
126 | int __thread__set_comm(struct thread *thread, const char *str, u64 timestamp, |
127 | bool exec) | |
1902efe7 FW |
128 | { |
129 | struct comm *new, *curr = thread__comm(thread); | |
3178f58b | 130 | int err; |
1902efe7 | 131 | |
a8480808 AH |
132 | /* Override the default :tid entry */ |
133 | if (!thread->comm_set) { | |
65de51f9 | 134 | err = comm__override(curr, str, timestamp, exec); |
3178f58b FW |
135 | if (err) |
136 | return err; | |
a5285ad9 | 137 | } else { |
65de51f9 | 138 | new = comm__new(str, timestamp, exec); |
a5285ad9 FW |
139 | if (!new) |
140 | return -ENOMEM; | |
141 | list_add(&new->list, &thread->comm_list); | |
380b5143 NK |
142 | |
143 | if (exec) | |
144 | unwind__flush_access(thread); | |
4385d580 | 145 | } |
1902efe7 | 146 | |
1902efe7 FW |
147 | thread->comm_set = true; |
148 | ||
149 | return 0; | |
6baa0a5a FW |
150 | } |
151 | ||
b9c5143a FW |
152 | const char *thread__comm_str(const struct thread *thread) |
153 | { | |
1902efe7 FW |
154 | const struct comm *comm = thread__comm(thread); |
155 | ||
156 | if (!comm) | |
157 | return NULL; | |
158 | ||
159 | return comm__str(comm); | |
b9c5143a FW |
160 | } |
161 | ||
1902efe7 | 162 | /* CHECKME: it should probably better return the max comm len from its comm list */ |
c824c433 | 163 | int thread__comm_len(struct thread *thread) |
a4fb581b | 164 | { |
c824c433 | 165 | if (!thread->comm_len) { |
1902efe7 FW |
166 | const char *comm = thread__comm_str(thread); |
167 | if (!comm) | |
a4fb581b | 168 | return 0; |
1902efe7 | 169 | thread->comm_len = strlen(comm); |
a4fb581b FW |
170 | } |
171 | ||
c824c433 | 172 | return thread->comm_len; |
a4fb581b FW |
173 | } |
174 | ||
3f067dca | 175 | size_t thread__fprintf(struct thread *thread, FILE *fp) |
9958e1f0 | 176 | { |
b9c5143a | 177 | return fprintf(fp, "Thread %d %s\n", thread->tid, thread__comm_str(thread)) + |
acebd408 | 178 | map_groups__fprintf(thread->mg, fp); |
6baa0a5a FW |
179 | } |
180 | ||
c824c433 | 181 | void thread__insert_map(struct thread *thread, struct map *map) |
1b46cddf | 182 | { |
acebd408 | 183 | map_groups__fixup_overlappings(thread->mg, map, stderr); |
93d5731d | 184 | map_groups__insert(thread->mg, map); |
6baa0a5a FW |
185 | } |
186 | ||
cddcef60 JO |
187 | static int thread__clone_map_groups(struct thread *thread, |
188 | struct thread *parent) | |
189 | { | |
190 | int i; | |
191 | ||
192 | /* This is new thread, we share map groups for process. */ | |
193 | if (thread->pid_ == parent->pid_) | |
194 | return 0; | |
195 | ||
196 | /* But this one is new process, copy maps. */ | |
197 | for (i = 0; i < MAP__NR_TYPES; ++i) | |
198 | if (map_groups__clone(thread->mg, parent->mg, i) < 0) | |
199 | return -ENOMEM; | |
200 | ||
201 | return 0; | |
202 | } | |
203 | ||
1902efe7 | 204 | int thread__fork(struct thread *thread, struct thread *parent, u64 timestamp) |
95011c60 | 205 | { |
cddcef60 | 206 | int err; |
6baa0a5a | 207 | |
faa5c5c3 | 208 | if (parent->comm_set) { |
1902efe7 FW |
209 | const char *comm = thread__comm_str(parent); |
210 | if (!comm) | |
faa5c5c3 | 211 | return -ENOMEM; |
1902efe7 | 212 | err = thread__set_comm(thread, comm, timestamp); |
8d00be81 | 213 | if (err) |
1902efe7 | 214 | return err; |
faa5c5c3 | 215 | } |
6baa0a5a | 216 | |
c824c433 | 217 | thread->ppid = parent->tid; |
cddcef60 | 218 | return thread__clone_map_groups(thread, parent); |
6baa0a5a | 219 | } |
52a3cb8c ACM |
220 | |
221 | void thread__find_cpumode_addr_location(struct thread *thread, | |
52a3cb8c ACM |
222 | enum map_type type, u64 addr, |
223 | struct addr_location *al) | |
224 | { | |
225 | size_t i; | |
226 | const u8 const cpumodes[] = { | |
227 | PERF_RECORD_MISC_USER, | |
228 | PERF_RECORD_MISC_KERNEL, | |
229 | PERF_RECORD_MISC_GUEST_USER, | |
230 | PERF_RECORD_MISC_GUEST_KERNEL | |
231 | }; | |
232 | ||
233 | for (i = 0; i < ARRAY_SIZE(cpumodes); i++) { | |
bb871a9c | 234 | thread__find_addr_location(thread, cpumodes[i], type, addr, al); |
52a3cb8c ACM |
235 | if (al->map) |
236 | break; | |
237 | } | |
238 | } |