xref: /linux/kernel/trace/trace_event_perf.c (revision 5d029b035bf112466541b844ee1b86197936db65)
1bcea3f96SSteven Rostedt (VMware) // SPDX-License-Identifier: GPL-2.0
297d5a220SFrederic Weisbecker /*
397d5a220SFrederic Weisbecker  * trace event based perf event profiling/tracing
497d5a220SFrederic Weisbecker  *
590eec103SPeter Zijlstra  * Copyright (C) 2009 Red Hat Inc, Peter Zijlstra
697d5a220SFrederic Weisbecker  * Copyright (C) 2009-2010 Frederic Weisbecker <fweisbec@gmail.com>
797d5a220SFrederic Weisbecker  */
897d5a220SFrederic Weisbecker 
997d5a220SFrederic Weisbecker #include <linux/module.h>
1097d5a220SFrederic Weisbecker #include <linux/kprobes.h>
11da97e184SJoel Fernandes (Google) #include <linux/security.h>
1297d5a220SFrederic Weisbecker #include "trace.h"
13e12f03d7SSong Liu #include "trace_probe.h"
1497d5a220SFrederic Weisbecker 
156016ee13SNamhyung Kim static char __percpu *perf_trace_buf[PERF_NR_CONTEXTS];
1697d5a220SFrederic Weisbecker 
17eb1e7961SFrederic Weisbecker /*
18eb1e7961SFrederic Weisbecker  * Force it to be aligned to unsigned long to avoid misaligned accesses
19eb1e7961SFrederic Weisbecker  * suprises
20eb1e7961SFrederic Weisbecker  */
21eb1e7961SFrederic Weisbecker typedef typeof(unsigned long [PERF_MAX_TRACE_SIZE / sizeof(unsigned long)])
22eb1e7961SFrederic Weisbecker 	perf_trace_t;
2397d5a220SFrederic Weisbecker 
2497d5a220SFrederic Weisbecker /* Count the events in use (per event id, not per instance) */
2597d5a220SFrederic Weisbecker static int	total_ref_count;
2697d5a220SFrederic Weisbecker 
272425bcb9SSteven Rostedt (Red Hat) static int perf_trace_event_perm(struct trace_event_call *tp_event,
2861c32659SFrederic Weisbecker 				 struct perf_event *p_event)
2961c32659SFrederic Weisbecker {
30da97e184SJoel Fernandes (Google) 	int ret;
31da97e184SJoel Fernandes (Google) 
32d5b5f391SPeter Zijlstra 	if (tp_event->perf_perm) {
33da97e184SJoel Fernandes (Google) 		ret = tp_event->perf_perm(tp_event, p_event);
34d5b5f391SPeter Zijlstra 		if (ret)
35d5b5f391SPeter Zijlstra 			return ret;
36d5b5f391SPeter Zijlstra 	}
37d5b5f391SPeter Zijlstra 
38f4be073dSJiri Olsa 	/*
39f4be073dSJiri Olsa 	 * We checked and allowed to create parent,
40f4be073dSJiri Olsa 	 * allow children without checking.
41f4be073dSJiri Olsa 	 */
42f4be073dSJiri Olsa 	if (p_event->parent)
43f4be073dSJiri Olsa 		return 0;
44f4be073dSJiri Olsa 
45f4be073dSJiri Olsa 	/*
46f4be073dSJiri Olsa 	 * It's ok to check current process (owner) permissions in here,
47f4be073dSJiri Olsa 	 * because code below is called only via perf_event_open syscall.
48f4be073dSJiri Olsa 	 */
49f4be073dSJiri Olsa 
50ced39002SJiri Olsa 	/* The ftrace function trace is allowed only for root. */
51cfa77bc4SJiri Olsa 	if (ftrace_event_is_function(tp_event)) {
52da97e184SJoel Fernandes (Google) 		ret = perf_allow_tracepoint(&p_event->attr);
53da97e184SJoel Fernandes (Google) 		if (ret)
54da97e184SJoel Fernandes (Google) 			return ret;
55ced39002SJiri Olsa 
560a74c5b3SJiri Olsa 		if (!is_sampling_event(p_event))
570a74c5b3SJiri Olsa 			return 0;
580a74c5b3SJiri Olsa 
59cfa77bc4SJiri Olsa 		/*
60cfa77bc4SJiri Olsa 		 * We don't allow user space callchains for  function trace
61cfa77bc4SJiri Olsa 		 * event, due to issues with page faults while tracing page
62cfa77bc4SJiri Olsa 		 * fault handler and its overall trickiness nature.
63cfa77bc4SJiri Olsa 		 */
64cfa77bc4SJiri Olsa 		if (!p_event->attr.exclude_callchain_user)
65cfa77bc4SJiri Olsa 			return -EINVAL;
6663c45f4bSJiri Olsa 
6763c45f4bSJiri Olsa 		/*
6863c45f4bSJiri Olsa 		 * Same reason to disable user stack dump as for user space
6963c45f4bSJiri Olsa 		 * callchains above.
7063c45f4bSJiri Olsa 		 */
7163c45f4bSJiri Olsa 		if (p_event->attr.sample_type & PERF_SAMPLE_STACK_USER)
7263c45f4bSJiri Olsa 			return -EINVAL;
73cfa77bc4SJiri Olsa 	}
74cfa77bc4SJiri Olsa 
7561c32659SFrederic Weisbecker 	/* No tracing, just counting, so no obvious leak */
7661c32659SFrederic Weisbecker 	if (!(p_event->attr.sample_type & PERF_SAMPLE_RAW))
7761c32659SFrederic Weisbecker 		return 0;
7861c32659SFrederic Weisbecker 
7961c32659SFrederic Weisbecker 	/* Some events are ok to be traced by non-root users... */
8061c32659SFrederic Weisbecker 	if (p_event->attach_state == PERF_ATTACH_TASK) {
8161c32659SFrederic Weisbecker 		if (tp_event->flags & TRACE_EVENT_FL_CAP_ANY)
8261c32659SFrederic Weisbecker 			return 0;
8361c32659SFrederic Weisbecker 	}
8461c32659SFrederic Weisbecker 
8561c32659SFrederic Weisbecker 	/*
8661c32659SFrederic Weisbecker 	 * ...otherwise raw tracepoint data can be a severe data leak,
8761c32659SFrederic Weisbecker 	 * only allow root to have these.
8861c32659SFrederic Weisbecker 	 */
89da97e184SJoel Fernandes (Google) 	ret = perf_allow_tracepoint(&p_event->attr);
90da97e184SJoel Fernandes (Google) 	if (ret)
91da97e184SJoel Fernandes (Google) 		return ret;
9261c32659SFrederic Weisbecker 
9361c32659SFrederic Weisbecker 	return 0;
9461c32659SFrederic Weisbecker }
9561c32659SFrederic Weisbecker 
962425bcb9SSteven Rostedt (Red Hat) static int perf_trace_event_reg(struct trace_event_call *tp_event,
971c024ecaSPeter Zijlstra 				struct perf_event *p_event)
9897d5a220SFrederic Weisbecker {
996016ee13SNamhyung Kim 	struct hlist_head __percpu *list;
100ceec0b6fSJiri Olsa 	int ret = -ENOMEM;
1011c024ecaSPeter Zijlstra 	int cpu;
10297d5a220SFrederic Weisbecker 
1031c024ecaSPeter Zijlstra 	p_event->tp_event = tp_event;
1041c024ecaSPeter Zijlstra 	if (tp_event->perf_refcount++ > 0)
10597d5a220SFrederic Weisbecker 		return 0;
10697d5a220SFrederic Weisbecker 
1071c024ecaSPeter Zijlstra 	list = alloc_percpu(struct hlist_head);
1081c024ecaSPeter Zijlstra 	if (!list)
1091c024ecaSPeter Zijlstra 		goto fail;
1101c024ecaSPeter Zijlstra 
1111c024ecaSPeter Zijlstra 	for_each_possible_cpu(cpu)
1121c024ecaSPeter Zijlstra 		INIT_HLIST_HEAD(per_cpu_ptr(list, cpu));
1131c024ecaSPeter Zijlstra 
1141c024ecaSPeter Zijlstra 	tp_event->perf_events = list;
11597d5a220SFrederic Weisbecker 
11697d5a220SFrederic Weisbecker 	if (!total_ref_count) {
1176016ee13SNamhyung Kim 		char __percpu *buf;
118b7e2ecefSPeter Zijlstra 		int i;
119b7e2ecefSPeter Zijlstra 
1207ae07ea3SFrederic Weisbecker 		for (i = 0; i < PERF_NR_CONTEXTS; i++) {
1216016ee13SNamhyung Kim 			buf = (char __percpu *)alloc_percpu(perf_trace_t);
12297d5a220SFrederic Weisbecker 			if (!buf)
1231c024ecaSPeter Zijlstra 				goto fail;
12497d5a220SFrederic Weisbecker 
1251c024ecaSPeter Zijlstra 			perf_trace_buf[i] = buf;
126b7e2ecefSPeter Zijlstra 		}
12797d5a220SFrederic Weisbecker 	}
12897d5a220SFrederic Weisbecker 
129ceec0b6fSJiri Olsa 	ret = tp_event->class->reg(tp_event, TRACE_REG_PERF_REGISTER, NULL);
1301c024ecaSPeter Zijlstra 	if (ret)
1311c024ecaSPeter Zijlstra 		goto fail;
1321c024ecaSPeter Zijlstra 
13397d5a220SFrederic Weisbecker 	total_ref_count++;
13497d5a220SFrederic Weisbecker 	return 0;
13597d5a220SFrederic Weisbecker 
1361c024ecaSPeter Zijlstra fail:
137b7e2ecefSPeter Zijlstra 	if (!total_ref_count) {
138b7e2ecefSPeter Zijlstra 		int i;
139b7e2ecefSPeter Zijlstra 
1407ae07ea3SFrederic Weisbecker 		for (i = 0; i < PERF_NR_CONTEXTS; i++) {
141b7e2ecefSPeter Zijlstra 			free_percpu(perf_trace_buf[i]);
142b7e2ecefSPeter Zijlstra 			perf_trace_buf[i] = NULL;
143b7e2ecefSPeter Zijlstra 		}
14497d5a220SFrederic Weisbecker 	}
14597d5a220SFrederic Weisbecker 
1461c024ecaSPeter Zijlstra 	if (!--tp_event->perf_refcount) {
1471c024ecaSPeter Zijlstra 		free_percpu(tp_event->perf_events);
1481c024ecaSPeter Zijlstra 		tp_event->perf_events = NULL;
14997d5a220SFrederic Weisbecker 	}
15097d5a220SFrederic Weisbecker 
15197d5a220SFrederic Weisbecker 	return ret;
15297d5a220SFrederic Weisbecker }
15397d5a220SFrederic Weisbecker 
154ceec0b6fSJiri Olsa static void perf_trace_event_unreg(struct perf_event *p_event)
155ceec0b6fSJiri Olsa {
1562425bcb9SSteven Rostedt (Red Hat) 	struct trace_event_call *tp_event = p_event->tp_event;
157ceec0b6fSJiri Olsa 	int i;
158ceec0b6fSJiri Olsa 
159ceec0b6fSJiri Olsa 	if (--tp_event->perf_refcount > 0)
160ceec0b6fSJiri Olsa 		goto out;
161ceec0b6fSJiri Olsa 
162ceec0b6fSJiri Olsa 	tp_event->class->reg(tp_event, TRACE_REG_PERF_UNREGISTER, NULL);
163ceec0b6fSJiri Olsa 
164ceec0b6fSJiri Olsa 	/*
165ceec0b6fSJiri Olsa 	 * Ensure our callback won't be called anymore. The buffers
166ceec0b6fSJiri Olsa 	 * will be freed after that.
167ceec0b6fSJiri Olsa 	 */
168ceec0b6fSJiri Olsa 	tracepoint_synchronize_unregister();
169ceec0b6fSJiri Olsa 
170ceec0b6fSJiri Olsa 	free_percpu(tp_event->perf_events);
171ceec0b6fSJiri Olsa 	tp_event->perf_events = NULL;
172ceec0b6fSJiri Olsa 
173ceec0b6fSJiri Olsa 	if (!--total_ref_count) {
174ceec0b6fSJiri Olsa 		for (i = 0; i < PERF_NR_CONTEXTS; i++) {
175ceec0b6fSJiri Olsa 			free_percpu(perf_trace_buf[i]);
176ceec0b6fSJiri Olsa 			perf_trace_buf[i] = NULL;
177ceec0b6fSJiri Olsa 		}
178ceec0b6fSJiri Olsa 	}
179ceec0b6fSJiri Olsa out:
180ceec0b6fSJiri Olsa 	module_put(tp_event->mod);
181ceec0b6fSJiri Olsa }
182ceec0b6fSJiri Olsa 
183ceec0b6fSJiri Olsa static int perf_trace_event_open(struct perf_event *p_event)
184ceec0b6fSJiri Olsa {
1852425bcb9SSteven Rostedt (Red Hat) 	struct trace_event_call *tp_event = p_event->tp_event;
186ceec0b6fSJiri Olsa 	return tp_event->class->reg(tp_event, TRACE_REG_PERF_OPEN, p_event);
187ceec0b6fSJiri Olsa }
188ceec0b6fSJiri Olsa 
189ceec0b6fSJiri Olsa static void perf_trace_event_close(struct perf_event *p_event)
190ceec0b6fSJiri Olsa {
1912425bcb9SSteven Rostedt (Red Hat) 	struct trace_event_call *tp_event = p_event->tp_event;
192ceec0b6fSJiri Olsa 	tp_event->class->reg(tp_event, TRACE_REG_PERF_CLOSE, p_event);
193ceec0b6fSJiri Olsa }
194ceec0b6fSJiri Olsa 
1952425bcb9SSteven Rostedt (Red Hat) static int perf_trace_event_init(struct trace_event_call *tp_event,
196ceec0b6fSJiri Olsa 				 struct perf_event *p_event)
197ceec0b6fSJiri Olsa {
198ceec0b6fSJiri Olsa 	int ret;
199ceec0b6fSJiri Olsa 
200ceec0b6fSJiri Olsa 	ret = perf_trace_event_perm(tp_event, p_event);
201ceec0b6fSJiri Olsa 	if (ret)
202ceec0b6fSJiri Olsa 		return ret;
203ceec0b6fSJiri Olsa 
204ceec0b6fSJiri Olsa 	ret = perf_trace_event_reg(tp_event, p_event);
205ceec0b6fSJiri Olsa 	if (ret)
206ceec0b6fSJiri Olsa 		return ret;
207ceec0b6fSJiri Olsa 
208ceec0b6fSJiri Olsa 	ret = perf_trace_event_open(p_event);
209ceec0b6fSJiri Olsa 	if (ret) {
210ceec0b6fSJiri Olsa 		perf_trace_event_unreg(p_event);
211ceec0b6fSJiri Olsa 		return ret;
212ceec0b6fSJiri Olsa 	}
213ceec0b6fSJiri Olsa 
214ceec0b6fSJiri Olsa 	return 0;
215ceec0b6fSJiri Olsa }
216ceec0b6fSJiri Olsa 
2171c024ecaSPeter Zijlstra int perf_trace_init(struct perf_event *p_event)
21897d5a220SFrederic Weisbecker {
2192425bcb9SSteven Rostedt (Red Hat) 	struct trace_event_call *tp_event;
2200022ceddSVince Weaver 	u64 event_id = p_event->attr.config;
22197d5a220SFrederic Weisbecker 	int ret = -EINVAL;
22297d5a220SFrederic Weisbecker 
22397d5a220SFrederic Weisbecker 	mutex_lock(&event_mutex);
2241c024ecaSPeter Zijlstra 	list_for_each_entry(tp_event, &ftrace_events, list) {
225ff5f149bSSteven Rostedt 		if (tp_event->event.type == event_id &&
226a1d0ce82SSteven Rostedt 		    tp_event->class && tp_event->class->reg &&
2271c024ecaSPeter Zijlstra 		    try_module_get(tp_event->mod)) {
2281c024ecaSPeter Zijlstra 			ret = perf_trace_event_init(tp_event, p_event);
2299cb627d5SLi Zefan 			if (ret)
2309cb627d5SLi Zefan 				module_put(tp_event->mod);
23197d5a220SFrederic Weisbecker 			break;
23297d5a220SFrederic Weisbecker 		}
23397d5a220SFrederic Weisbecker 	}
23497d5a220SFrederic Weisbecker 	mutex_unlock(&event_mutex);
23597d5a220SFrederic Weisbecker 
23697d5a220SFrederic Weisbecker 	return ret;
23797d5a220SFrederic Weisbecker }
23897d5a220SFrederic Weisbecker 
239ceec0b6fSJiri Olsa void perf_trace_destroy(struct perf_event *p_event)
240ceec0b6fSJiri Olsa {
241ceec0b6fSJiri Olsa 	mutex_lock(&event_mutex);
242ceec0b6fSJiri Olsa 	perf_trace_event_close(p_event);
243ceec0b6fSJiri Olsa 	perf_trace_event_unreg(p_event);
244ceec0b6fSJiri Olsa 	mutex_unlock(&event_mutex);
245ceec0b6fSJiri Olsa }
246ceec0b6fSJiri Olsa 
247e12f03d7SSong Liu #ifdef CONFIG_KPROBE_EVENTS
248e12f03d7SSong Liu int perf_kprobe_init(struct perf_event *p_event, bool is_retprobe)
249e12f03d7SSong Liu {
250e12f03d7SSong Liu 	int ret;
251e12f03d7SSong Liu 	char *func = NULL;
252e12f03d7SSong Liu 	struct trace_event_call *tp_event;
253e12f03d7SSong Liu 
254e12f03d7SSong Liu 	if (p_event->attr.kprobe_func) {
255e12f03d7SSong Liu 		func = kzalloc(KSYM_NAME_LEN, GFP_KERNEL);
256e12f03d7SSong Liu 		if (!func)
257e12f03d7SSong Liu 			return -ENOMEM;
258e12f03d7SSong Liu 		ret = strncpy_from_user(
259e12f03d7SSong Liu 			func, u64_to_user_ptr(p_event->attr.kprobe_func),
260e12f03d7SSong Liu 			KSYM_NAME_LEN);
2615da13ab8SMasami Hiramatsu 		if (ret == KSYM_NAME_LEN)
2625da13ab8SMasami Hiramatsu 			ret = -E2BIG;
263e12f03d7SSong Liu 		if (ret < 0)
264e12f03d7SSong Liu 			goto out;
265e12f03d7SSong Liu 
266e12f03d7SSong Liu 		if (func[0] == '\0') {
267e12f03d7SSong Liu 			kfree(func);
268e12f03d7SSong Liu 			func = NULL;
269e12f03d7SSong Liu 		}
270e12f03d7SSong Liu 	}
271e12f03d7SSong Liu 
272e12f03d7SSong Liu 	tp_event = create_local_trace_kprobe(
273e12f03d7SSong Liu 		func, (void *)(unsigned long)(p_event->attr.kprobe_addr),
274e12f03d7SSong Liu 		p_event->attr.probe_offset, is_retprobe);
275e12f03d7SSong Liu 	if (IS_ERR(tp_event)) {
276e12f03d7SSong Liu 		ret = PTR_ERR(tp_event);
277e12f03d7SSong Liu 		goto out;
278e12f03d7SSong Liu 	}
279e12f03d7SSong Liu 
2806b1340ccSPrateek Sood 	mutex_lock(&event_mutex);
281e12f03d7SSong Liu 	ret = perf_trace_event_init(tp_event, p_event);
282e12f03d7SSong Liu 	if (ret)
283e12f03d7SSong Liu 		destroy_local_trace_kprobe(tp_event);
2846b1340ccSPrateek Sood 	mutex_unlock(&event_mutex);
285e12f03d7SSong Liu out:
286e12f03d7SSong Liu 	kfree(func);
287e12f03d7SSong Liu 	return ret;
288e12f03d7SSong Liu }
289e12f03d7SSong Liu 
290e12f03d7SSong Liu void perf_kprobe_destroy(struct perf_event *p_event)
291e12f03d7SSong Liu {
2926b1340ccSPrateek Sood 	mutex_lock(&event_mutex);
293e12f03d7SSong Liu 	perf_trace_event_close(p_event);
294e12f03d7SSong Liu 	perf_trace_event_unreg(p_event);
2956b1340ccSPrateek Sood 	mutex_unlock(&event_mutex);
296e12f03d7SSong Liu 
297e12f03d7SSong Liu 	destroy_local_trace_kprobe(p_event->tp_event);
298e12f03d7SSong Liu }
299e12f03d7SSong Liu #endif /* CONFIG_KPROBE_EVENTS */
300e12f03d7SSong Liu 
30133ea4b24SSong Liu #ifdef CONFIG_UPROBE_EVENTS
302a6ca88b2SSong Liu int perf_uprobe_init(struct perf_event *p_event,
303a6ca88b2SSong Liu 		     unsigned long ref_ctr_offset, bool is_retprobe)
30433ea4b24SSong Liu {
30533ea4b24SSong Liu 	int ret;
30633ea4b24SSong Liu 	char *path = NULL;
30733ea4b24SSong Liu 	struct trace_event_call *tp_event;
30833ea4b24SSong Liu 
30933ea4b24SSong Liu 	if (!p_event->attr.uprobe_path)
31033ea4b24SSong Liu 		return -EINVAL;
31183540fbcSJann Horn 
31283540fbcSJann Horn 	path = strndup_user(u64_to_user_ptr(p_event->attr.uprobe_path),
31383540fbcSJann Horn 			    PATH_MAX);
31483540fbcSJann Horn 	if (IS_ERR(path)) {
31583540fbcSJann Horn 		ret = PTR_ERR(path);
31683540fbcSJann Horn 		return (ret == -EINVAL) ? -E2BIG : ret;
31783540fbcSJann Horn 	}
31833ea4b24SSong Liu 	if (path[0] == '\0') {
31933ea4b24SSong Liu 		ret = -EINVAL;
32033ea4b24SSong Liu 		goto out;
32133ea4b24SSong Liu 	}
32233ea4b24SSong Liu 
323a6ca88b2SSong Liu 	tp_event = create_local_trace_uprobe(path, p_event->attr.probe_offset,
324a6ca88b2SSong Liu 					     ref_ctr_offset, is_retprobe);
32533ea4b24SSong Liu 	if (IS_ERR(tp_event)) {
32633ea4b24SSong Liu 		ret = PTR_ERR(tp_event);
32733ea4b24SSong Liu 		goto out;
32833ea4b24SSong Liu 	}
32933ea4b24SSong Liu 
33033ea4b24SSong Liu 	/*
33133ea4b24SSong Liu 	 * local trace_uprobe need to hold event_mutex to call
33233ea4b24SSong Liu 	 * uprobe_buffer_enable() and uprobe_buffer_disable().
33333ea4b24SSong Liu 	 * event_mutex is not required for local trace_kprobes.
33433ea4b24SSong Liu 	 */
33533ea4b24SSong Liu 	mutex_lock(&event_mutex);
33633ea4b24SSong Liu 	ret = perf_trace_event_init(tp_event, p_event);
33733ea4b24SSong Liu 	if (ret)
33833ea4b24SSong Liu 		destroy_local_trace_uprobe(tp_event);
33933ea4b24SSong Liu 	mutex_unlock(&event_mutex);
34033ea4b24SSong Liu out:
34133ea4b24SSong Liu 	kfree(path);
34233ea4b24SSong Liu 	return ret;
34333ea4b24SSong Liu }
34433ea4b24SSong Liu 
34533ea4b24SSong Liu void perf_uprobe_destroy(struct perf_event *p_event)
34633ea4b24SSong Liu {
34733ea4b24SSong Liu 	mutex_lock(&event_mutex);
34833ea4b24SSong Liu 	perf_trace_event_close(p_event);
34933ea4b24SSong Liu 	perf_trace_event_unreg(p_event);
35033ea4b24SSong Liu 	mutex_unlock(&event_mutex);
35133ea4b24SSong Liu 	destroy_local_trace_uprobe(p_event->tp_event);
35233ea4b24SSong Liu }
35333ea4b24SSong Liu #endif /* CONFIG_UPROBE_EVENTS */
35433ea4b24SSong Liu 
355a4eaf7f1SPeter Zijlstra int perf_trace_add(struct perf_event *p_event, int flags)
35697d5a220SFrederic Weisbecker {
3572425bcb9SSteven Rostedt (Red Hat) 	struct trace_event_call *tp_event = p_event->tp_event;
358466c81c4SPeter Zijlstra 
359466c81c4SPeter Zijlstra 	if (!(flags & PERF_EF_START))
360466c81c4SPeter Zijlstra 		p_event->hw.state = PERF_HES_STOPPED;
361466c81c4SPeter Zijlstra 
362466c81c4SPeter Zijlstra 	/*
363466c81c4SPeter Zijlstra 	 * If TRACE_REG_PERF_ADD returns false; no custom action was performed
364466c81c4SPeter Zijlstra 	 * and we need to take the default action of enqueueing our event on
365466c81c4SPeter Zijlstra 	 * the right per-cpu hlist.
366466c81c4SPeter Zijlstra 	 */
367466c81c4SPeter Zijlstra 	if (!tp_event->class->reg(tp_event, TRACE_REG_PERF_ADD, p_event)) {
3686016ee13SNamhyung Kim 		struct hlist_head __percpu *pcpu_list;
3691c024ecaSPeter Zijlstra 		struct hlist_head *list;
37097d5a220SFrederic Weisbecker 
3716016ee13SNamhyung Kim 		pcpu_list = tp_event->perf_events;
3726016ee13SNamhyung Kim 		if (WARN_ON_ONCE(!pcpu_list))
3731c024ecaSPeter Zijlstra 			return -EINVAL;
37497d5a220SFrederic Weisbecker 
3756016ee13SNamhyung Kim 		list = this_cpu_ptr(pcpu_list);
3761c024ecaSPeter Zijlstra 		hlist_add_head_rcu(&p_event->hlist_entry, list);
377466c81c4SPeter Zijlstra 	}
3781c024ecaSPeter Zijlstra 
379466c81c4SPeter Zijlstra 	return 0;
3801c024ecaSPeter Zijlstra }
3811c024ecaSPeter Zijlstra 
382a4eaf7f1SPeter Zijlstra void perf_trace_del(struct perf_event *p_event, int flags)
3831c024ecaSPeter Zijlstra {
3842425bcb9SSteven Rostedt (Red Hat) 	struct trace_event_call *tp_event = p_event->tp_event;
385466c81c4SPeter Zijlstra 
386466c81c4SPeter Zijlstra 	/*
387466c81c4SPeter Zijlstra 	 * If TRACE_REG_PERF_DEL returns false; no custom action was performed
388466c81c4SPeter Zijlstra 	 * and we need to take the default action of dequeueing our event from
389466c81c4SPeter Zijlstra 	 * the right per-cpu hlist.
390466c81c4SPeter Zijlstra 	 */
391466c81c4SPeter Zijlstra 	if (!tp_event->class->reg(tp_event, TRACE_REG_PERF_DEL, p_event))
3921c024ecaSPeter Zijlstra 		hlist_del_rcu(&p_event->hlist_entry);
3931c024ecaSPeter Zijlstra }
3941c024ecaSPeter Zijlstra 
3951e1dcd93SAlexei Starovoitov void *perf_trace_buf_alloc(int size, struct pt_regs **regs, int *rctxp)
39697d5a220SFrederic Weisbecker {
3971c024ecaSPeter Zijlstra 	char *raw_data;
3981e1dcd93SAlexei Starovoitov 	int rctx;
39997d5a220SFrederic Weisbecker 
400eb1e7961SFrederic Weisbecker 	BUILD_BUG_ON(PERF_MAX_TRACE_SIZE % sizeof(unsigned long));
401eb1e7961SFrederic Weisbecker 
402cd92bf61SOleg Nesterov 	if (WARN_ONCE(size > PERF_MAX_TRACE_SIZE,
403cd92bf61SOleg Nesterov 		      "perf buffer not large enough"))
404cd92bf61SOleg Nesterov 		return NULL;
405cd92bf61SOleg Nesterov 
4061e1dcd93SAlexei Starovoitov 	*rctxp = rctx = perf_swevent_get_recursion_context();
4071e1dcd93SAlexei Starovoitov 	if (rctx < 0)
4081c024ecaSPeter Zijlstra 		return NULL;
40997d5a220SFrederic Weisbecker 
41086038c5eSPeter Zijlstra (Intel) 	if (regs)
4111e1dcd93SAlexei Starovoitov 		*regs = this_cpu_ptr(&__perf_regs[rctx]);
4121e1dcd93SAlexei Starovoitov 	raw_data = this_cpu_ptr(perf_trace_buf[rctx]);
41397d5a220SFrederic Weisbecker 
41497d5a220SFrederic Weisbecker 	/* zero the dead bytes from align to not leak stack to user */
415eb1e7961SFrederic Weisbecker 	memset(&raw_data[size - sizeof(u64)], 0, sizeof(u64));
4161e1dcd93SAlexei Starovoitov 	return raw_data;
4171e1dcd93SAlexei Starovoitov }
4181e1dcd93SAlexei Starovoitov EXPORT_SYMBOL_GPL(perf_trace_buf_alloc);
4191e1dcd93SAlexei Starovoitov NOKPROBE_SYMBOL(perf_trace_buf_alloc);
42097d5a220SFrederic Weisbecker 
4211e1dcd93SAlexei Starovoitov void perf_trace_buf_update(void *record, u16 type)
4221e1dcd93SAlexei Starovoitov {
4231e1dcd93SAlexei Starovoitov 	struct trace_entry *entry = record;
4241e1dcd93SAlexei Starovoitov 	int pc = preempt_count();
4251e1dcd93SAlexei Starovoitov 	unsigned long flags;
4261e1dcd93SAlexei Starovoitov 
42787f44bbcSPeter Zijlstra 	local_save_flags(flags);
42846710f3aSCong Wang 	tracing_generic_entry_update(entry, type, flags, pc);
42997d5a220SFrederic Weisbecker }
4301e1dcd93SAlexei Starovoitov NOKPROBE_SYMBOL(perf_trace_buf_update);
431ced39002SJiri Olsa 
432ced39002SJiri Olsa #ifdef CONFIG_FUNCTION_TRACER
433ced39002SJiri Olsa static void
4342f5f6ad9SSteven Rostedt perf_ftrace_function_call(unsigned long ip, unsigned long parent_ip,
435a1e2e31dSSteven Rostedt 			  struct ftrace_ops *ops, struct pt_regs *pt_regs)
436ced39002SJiri Olsa {
437ced39002SJiri Olsa 	struct ftrace_entry *entry;
438466c81c4SPeter Zijlstra 	struct perf_event *event;
439466c81c4SPeter Zijlstra 	struct hlist_head head;
440ced39002SJiri Olsa 	struct pt_regs regs;
441ced39002SJiri Olsa 	int rctx;
4425d15a624SSteven Rostedt (VMware) 	int bit;
443ced39002SJiri Olsa 
444*5d029b03SSteven Rostedt (VMware) 	if (!rcu_is_watching())
445*5d029b03SSteven Rostedt (VMware) 		return;
446*5d029b03SSteven Rostedt (VMware) 
447466c81c4SPeter Zijlstra 	if ((unsigned long)ops->private != smp_processor_id())
448b8ebfd3fSOleg Nesterov 		return;
449b8ebfd3fSOleg Nesterov 
4505d15a624SSteven Rostedt (VMware) 	bit = ftrace_test_recursion_trylock();
4515d15a624SSteven Rostedt (VMware) 	if (bit < 0)
4525d15a624SSteven Rostedt (VMware) 		return;
4535d15a624SSteven Rostedt (VMware) 
454466c81c4SPeter Zijlstra 	event = container_of(ops, struct perf_event, ftrace_ops);
455466c81c4SPeter Zijlstra 
456466c81c4SPeter Zijlstra 	/*
457466c81c4SPeter Zijlstra 	 * @event->hlist entry is NULL (per INIT_HLIST_NODE), and all
458466c81c4SPeter Zijlstra 	 * the perf code does is hlist_for_each_entry_rcu(), so we can
459466c81c4SPeter Zijlstra 	 * get away with simply setting the @head.first pointer in order
460466c81c4SPeter Zijlstra 	 * to create a singular list.
461466c81c4SPeter Zijlstra 	 */
462466c81c4SPeter Zijlstra 	head.first = &event->hlist_entry;
463466c81c4SPeter Zijlstra 
464ced39002SJiri Olsa #define ENTRY_SIZE (ALIGN(sizeof(struct ftrace_entry) + sizeof(u32), \
465ced39002SJiri Olsa 		    sizeof(u64)) - sizeof(u32))
466ced39002SJiri Olsa 
467ced39002SJiri Olsa 	BUILD_BUG_ON(ENTRY_SIZE > PERF_MAX_TRACE_SIZE);
468ced39002SJiri Olsa 
469ec5e099dSAlexei Starovoitov 	memset(&regs, 0, sizeof(regs));
470ced39002SJiri Olsa 	perf_fetch_caller_regs(&regs);
471ced39002SJiri Olsa 
4721e1dcd93SAlexei Starovoitov 	entry = perf_trace_buf_alloc(ENTRY_SIZE, NULL, &rctx);
473ced39002SJiri Olsa 	if (!entry)
4745d15a624SSteven Rostedt (VMware) 		goto out;
475ced39002SJiri Olsa 
476ced39002SJiri Olsa 	entry->ip = ip;
477ced39002SJiri Olsa 	entry->parent_ip = parent_ip;
4781e1dcd93SAlexei Starovoitov 	perf_trace_buf_submit(entry, ENTRY_SIZE, rctx, TRACE_FN,
479466c81c4SPeter Zijlstra 			      1, &regs, &head, NULL);
480ced39002SJiri Olsa 
4815d15a624SSteven Rostedt (VMware) out:
4825d15a624SSteven Rostedt (VMware) 	ftrace_test_recursion_unlock(bit);
483ced39002SJiri Olsa #undef ENTRY_SIZE
484ced39002SJiri Olsa }
485ced39002SJiri Olsa 
486ced39002SJiri Olsa static int perf_ftrace_function_register(struct perf_event *event)
487ced39002SJiri Olsa {
488ced39002SJiri Olsa 	struct ftrace_ops *ops = &event->ftrace_ops;
489ced39002SJiri Olsa 
490ced39002SJiri Olsa 	ops->func    = perf_ftrace_function_call;
491466c81c4SPeter Zijlstra 	ops->private = (void *)(unsigned long)nr_cpu_ids;
492466c81c4SPeter Zijlstra 
493ced39002SJiri Olsa 	return register_ftrace_function(ops);
494ced39002SJiri Olsa }
495ced39002SJiri Olsa 
496ced39002SJiri Olsa static int perf_ftrace_function_unregister(struct perf_event *event)
497ced39002SJiri Olsa {
498ced39002SJiri Olsa 	struct ftrace_ops *ops = &event->ftrace_ops;
4995500fa51SJiri Olsa 	int ret = unregister_ftrace_function(ops);
5005500fa51SJiri Olsa 	ftrace_free_filter(ops);
5015500fa51SJiri Olsa 	return ret;
502ced39002SJiri Olsa }
503ced39002SJiri Olsa 
5042425bcb9SSteven Rostedt (Red Hat) int perf_ftrace_event_register(struct trace_event_call *call,
505ced39002SJiri Olsa 			       enum trace_reg type, void *data)
506ced39002SJiri Olsa {
507466c81c4SPeter Zijlstra 	struct perf_event *event = data;
508466c81c4SPeter Zijlstra 
509ced39002SJiri Olsa 	switch (type) {
510ced39002SJiri Olsa 	case TRACE_REG_REGISTER:
511ced39002SJiri Olsa 	case TRACE_REG_UNREGISTER:
512ced39002SJiri Olsa 		break;
513ced39002SJiri Olsa 	case TRACE_REG_PERF_REGISTER:
514ced39002SJiri Olsa 	case TRACE_REG_PERF_UNREGISTER:
515ced39002SJiri Olsa 		return 0;
516ced39002SJiri Olsa 	case TRACE_REG_PERF_OPEN:
517ced39002SJiri Olsa 		return perf_ftrace_function_register(data);
518ced39002SJiri Olsa 	case TRACE_REG_PERF_CLOSE:
519ced39002SJiri Olsa 		return perf_ftrace_function_unregister(data);
520ced39002SJiri Olsa 	case TRACE_REG_PERF_ADD:
521466c81c4SPeter Zijlstra 		event->ftrace_ops.private = (void *)(unsigned long)smp_processor_id();
522466c81c4SPeter Zijlstra 		return 1;
523ced39002SJiri Olsa 	case TRACE_REG_PERF_DEL:
524466c81c4SPeter Zijlstra 		event->ftrace_ops.private = (void *)(unsigned long)nr_cpu_ids;
525466c81c4SPeter Zijlstra 		return 1;
526ced39002SJiri Olsa 	}
527ced39002SJiri Olsa 
528ced39002SJiri Olsa 	return -EINVAL;
529ced39002SJiri Olsa }
530ced39002SJiri Olsa #endif /* CONFIG_FUNCTION_TRACER */
531