Move headers under include/
[lttng-modules.git] / lttng-syscalls.c
CommitLineData
b7cdc182 1/* SPDX-License-Identifier: (GPL-2.0-only or LGPL-2.1-only)
9f36eaed 2 *
259b6cb3
MD
3 * lttng-syscalls.c
4 *
2faf7d1b 5 * LTTng syscall probes.
259b6cb3 6 *
886d51a3 7 * Copyright (C) 2010-2012 Mathieu Desnoyers <mathieu.desnoyers@efficios.com>
259b6cb3
MD
8 */
9
10#include <linux/module.h>
11#include <linux/slab.h>
6333ace3 12#include <linux/compat.h>
abc0446a 13#include <linux/err.h>
80f87dd2 14#include <linux/bitmap.h>
7ca580f8
MD
15#include <linux/in.h>
16#include <linux/in6.h>
2d2464bd 17#include <linux/seq_file.h>
d4291869 18#include <linux/stringify.h>
082d4946
MD
19#include <linux/file.h>
20#include <linux/anon_inodes.h>
c8dfb724 21#include <linux/fcntl.h>
259b6cb3
MD
22#include <asm/ptrace.h>
23#include <asm/syscall.h>
24
b5304713
MD
25#include <lttng/bitfield.h>
26#include <lttng/lttng-events.h>
27#include <lttng/lttng-tracepoint.h>
259b6cb3 28
b97bc8bf
MD
29#define LTTNG_SYSCALL_NR_ARGS 6
30
6333ace3 31#ifndef CONFIG_COMPAT
bfa949bf
MD
32# ifndef is_compat_task
33# define is_compat_task() (0)
34# endif
6333ace3
MD
35#endif
36
1aa3298b
MD
37/* in_compat_syscall appears in kernel 4.6. */
38#ifndef in_compat_syscall
39 #define in_compat_syscall() is_compat_task()
40#endif
41
5b7ac358
MD
42enum sc_type {
43 SC_TYPE_ENTRY,
44 SC_TYPE_EXIT,
45 SC_TYPE_COMPAT_ENTRY,
46 SC_TYPE_COMPAT_EXIT,
47};
48
d4291869
MD
49#define SYSCALL_ENTRY_TOK syscall_entry_
50#define COMPAT_SYSCALL_ENTRY_TOK compat_syscall_entry_
51#define SYSCALL_EXIT_TOK syscall_exit_
52#define COMPAT_SYSCALL_EXIT_TOK compat_syscall_exit_
53
54#define SYSCALL_ENTRY_STR __stringify(SYSCALL_ENTRY_TOK)
55#define COMPAT_SYSCALL_ENTRY_STR __stringify(COMPAT_SYSCALL_ENTRY_TOK)
56#define SYSCALL_EXIT_STR __stringify(SYSCALL_EXIT_TOK)
57#define COMPAT_SYSCALL_EXIT_STR __stringify(COMPAT_SYSCALL_EXIT_TOK)
5b7ac358 58
a93244f8 59static
2faf7d1b 60void syscall_entry_probe(void *__data, struct pt_regs *regs, long id);
5b7ac358
MD
61static
62void syscall_exit_probe(void *__data, struct pt_regs *regs, long ret);
259b6cb3 63
3a523f5b
MD
64/*
65 * Forward declarations for old kernels.
66 */
67struct mmsghdr;
68struct rlimit64;
69struct oldold_utsname;
70struct old_utsname;
71struct sel_arg_struct;
72struct mmap_arg_struct;
c0b71117 73struct file_handle;
a292e6f1 74struct user_msghdr;
3a523f5b 75
9eb15e8b
MJ
76/*
77 * Forward declaration for kernels >= 5.6
78 */
79struct timex;
edfdcb68
MJ
80struct timeval;
81struct itimerval;
82struct itimerspec;
83
edfdcb68 84typedef __kernel_old_time_t time_t;
9eb15e8b 85
80f87dd2
MD
86#ifdef IA32_NR_syscalls
87#define NR_compat_syscalls IA32_NR_syscalls
88#else
89#define NR_compat_syscalls NR_syscalls
90#endif
91
259b6cb3
MD
92/*
93 * Create LTTng tracepoint probes.
94 */
95#define LTTNG_PACKAGE_BUILD
96#define CREATE_TRACE_POINTS
2655f9ad 97#define TP_MODULE_NOINIT
c075712b 98#define TRACE_INCLUDE_PATH instrumentation/syscalls/headers
259b6cb3 99
a93244f8
MD
100#define PARAMS(args...) args
101
5b7ac358 102/* Handle unknown syscalls */
72a52753 103#undef TRACE_SYSTEM
5b7ac358 104#define TRACE_SYSTEM syscalls_unknown
241ae9a8 105#include <instrumentation/syscalls/headers/syscalls_unknown.h>
5b7ac358
MD
106#undef TRACE_SYSTEM
107
fc4f7161
MD
108#define SC_ENTER
109
fc4f7161
MD
110#undef sc_exit
111#define sc_exit(...)
b75d00c4
MD
112#undef sc_in
113#define sc_in(...) __VA_ARGS__
114#undef sc_out
115#define sc_out(...)
116#undef sc_inout
117#define sc_inout(...) __VA_ARGS__
5b7ac358
MD
118
119/* Hijack probe callback for system call enter */
a93244f8 120#undef TP_PROBE_CB
259b6cb3 121#define TP_PROBE_CB(_template) &syscall_entry_probe
57ede728 122#define SC_LTTNG_TRACEPOINT_EVENT(_name, _proto, _args, _fields) \
d4291869 123 LTTNG_TRACEPOINT_EVENT(syscall_entry_##_name, PARAMS(_proto), PARAMS(_args), \
57ede728 124 PARAMS(_fields))
265822ae 125#define SC_LTTNG_TRACEPOINT_EVENT_CODE(_name, _proto, _args, _locvar, _code_pre, _fields, _code_post) \
d4291869 126 LTTNG_TRACEPOINT_EVENT_CODE(syscall_entry_##_name, PARAMS(_proto), PARAMS(_args), \
265822ae
MD
127 PARAMS(_locvar), PARAMS(_code_pre), \
128 PARAMS(_fields), PARAMS(_code_post))
57ede728
MD
129#define SC_LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS(_name, _fields) \
130 LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS(syscall_entry_##_name, PARAMS(_fields))
cb3ef14c 131#define SC_LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS(_template, _name) \
d4291869 132 LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS(syscall_entry_##_template, syscall_entry_##_name)
141ddf28
MD
133/* Enumerations only defined at first inclusion. */
134#define SC_LTTNG_TRACEPOINT_ENUM(_name, _values) \
135 LTTNG_TRACEPOINT_ENUM(_name, PARAMS(_values))
a93244f8 136#undef TRACE_SYSTEM
d4291869 137#define TRACE_SYSTEM syscall_entry_integers
5b7ac358 138#define TRACE_INCLUDE_FILE syscalls_integers
241ae9a8 139#include <instrumentation/syscalls/headers/syscalls_integers.h>
5b7ac358 140#undef TRACE_INCLUDE_FILE
a93244f8 141#undef TRACE_SYSTEM
d4291869 142#define TRACE_SYSTEM syscall_entry_pointers
5b7ac358 143#define TRACE_INCLUDE_FILE syscalls_pointers
241ae9a8 144#include <instrumentation/syscalls/headers/syscalls_pointers.h>
5b7ac358 145#undef TRACE_INCLUDE_FILE
a93244f8 146#undef TRACE_SYSTEM
141ddf28 147#undef SC_LTTNG_TRACEPOINT_ENUM
cb3ef14c
MD
148#undef SC_LTTNG_TRACEPOINT_EVENT_CODE
149#undef SC_LTTNG_TRACEPOINT_EVENT
150#undef SC_LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS
151#undef SC_LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS
5b7ac358 152#undef TP_PROBE_CB
3bc29f0a
MD
153#undef _TRACE_SYSCALLS_INTEGERS_H
154#undef _TRACE_SYSCALLS_POINTERS_H
5b7ac358
MD
155
156/* Hijack probe callback for compat system call enter */
a93244f8 157#define TP_PROBE_CB(_template) &syscall_entry_probe
771af27e 158#define LTTNG_SC_COMPAT
57ede728 159#define SC_LTTNG_TRACEPOINT_EVENT(_name, _proto, _args, _fields) \
d4291869 160 LTTNG_TRACEPOINT_EVENT(compat_syscall_entry_##_name, PARAMS(_proto), PARAMS(_args), \
57ede728 161 PARAMS(_fields))
265822ae 162#define SC_LTTNG_TRACEPOINT_EVENT_CODE(_name, _proto, _args, _locvar, _code_pre, _fields, _code_post) \
d4291869 163 LTTNG_TRACEPOINT_EVENT_CODE(compat_syscall_entry_##_name, PARAMS(_proto), PARAMS(_args), \
265822ae 164 PARAMS(_locvar), PARAMS(_code_pre), PARAMS(_fields), PARAMS(_code_post))
57ede728
MD
165#define SC_LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS(_name, _fields) \
166 LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS(compat_syscall_entry_##_name, PARAMS(_fields))
cb3ef14c 167#define SC_LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS(_template, _name) \
d4291869
MD
168 LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS(compat_syscall_entry_##_template, \
169 compat_syscall_entry_##_name)
141ddf28
MD
170/* Enumerations only defined at inital inclusion (not here). */
171#define SC_LTTNG_TRACEPOINT_ENUM(_name, _values)
d4291869 172#define TRACE_SYSTEM compat_syscall_entry_integers
5b7ac358 173#define TRACE_INCLUDE_FILE compat_syscalls_integers
241ae9a8 174#include <instrumentation/syscalls/headers/compat_syscalls_integers.h>
5b7ac358
MD
175#undef TRACE_INCLUDE_FILE
176#undef TRACE_SYSTEM
d4291869 177#define TRACE_SYSTEM compat_syscall_entry_pointers
5b7ac358 178#define TRACE_INCLUDE_FILE compat_syscalls_pointers
241ae9a8 179#include <instrumentation/syscalls/headers/compat_syscalls_pointers.h>
5b7ac358
MD
180#undef TRACE_INCLUDE_FILE
181#undef TRACE_SYSTEM
141ddf28 182#undef SC_LTTNG_TRACEPOINT_ENUM
cb3ef14c
MD
183#undef SC_LTTNG_TRACEPOINT_EVENT_CODE
184#undef SC_LTTNG_TRACEPOINT_EVENT
185#undef SC_LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS
186#undef SC_LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS
5b7ac358 187#undef TP_PROBE_CB
3bc29f0a
MD
188#undef _TRACE_SYSCALLS_INTEGERS_H
189#undef _TRACE_SYSCALLS_POINTERS_H
771af27e 190#undef LTTNG_SC_COMPAT
5b7ac358 191
fc4f7161
MD
192#undef SC_ENTER
193
194#define SC_EXIT
195
fc4f7161
MD
196#undef sc_exit
197#define sc_exit(...) __VA_ARGS__
b75d00c4
MD
198#undef sc_in
199#define sc_in(...)
200#undef sc_out
201#define sc_out(...) __VA_ARGS__
202#undef sc_inout
203#define sc_inout(...) __VA_ARGS__
5b7ac358
MD
204
205/* Hijack probe callback for system call exit */
206#define TP_PROBE_CB(_template) &syscall_exit_probe
57ede728 207#define SC_LTTNG_TRACEPOINT_EVENT(_name, _proto, _args, _fields) \
3bc29f0a 208 LTTNG_TRACEPOINT_EVENT(syscall_exit_##_name, PARAMS(_proto), PARAMS(_args), \
57ede728 209 PARAMS(_fields))
265822ae 210#define SC_LTTNG_TRACEPOINT_EVENT_CODE(_name, _proto, _args, _locvar, _code_pre, _fields, _code_post) \
3bc29f0a 211 LTTNG_TRACEPOINT_EVENT_CODE(syscall_exit_##_name, PARAMS(_proto), PARAMS(_args), \
265822ae 212 PARAMS(_locvar), PARAMS(_code_pre), PARAMS(_fields), PARAMS(_code_post))
57ede728
MD
213#define SC_LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS(_name, _fields) \
214 LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS(syscall_exit_##_name, PARAMS(_fields))
cb3ef14c
MD
215#define SC_LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS(_template, _name) \
216 LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS(syscall_exit_##_template, \
5b7ac358 217 syscall_exit_##_name)
141ddf28
MD
218/* Enumerations only defined at inital inclusion (not here). */
219#define SC_LTTNG_TRACEPOINT_ENUM(_name, _values)
5b7ac358
MD
220#define TRACE_SYSTEM syscall_exit_integers
221#define TRACE_INCLUDE_FILE syscalls_integers
241ae9a8 222#include <instrumentation/syscalls/headers/syscalls_integers.h>
5b7ac358
MD
223#undef TRACE_INCLUDE_FILE
224#undef TRACE_SYSTEM
225#define TRACE_SYSTEM syscall_exit_pointers
226#define TRACE_INCLUDE_FILE syscalls_pointers
241ae9a8 227#include <instrumentation/syscalls/headers/syscalls_pointers.h>
5b7ac358
MD
228#undef TRACE_INCLUDE_FILE
229#undef TRACE_SYSTEM
141ddf28 230#undef SC_LTTNG_TRACEPOINT_ENUM
cb3ef14c
MD
231#undef SC_LTTNG_TRACEPOINT_EVENT_CODE
232#undef SC_LTTNG_TRACEPOINT_EVENT
233#undef SC_LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS
234#undef SC_LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS
5b7ac358 235#undef TP_PROBE_CB
3bc29f0a
MD
236#undef _TRACE_SYSCALLS_INTEGERS_H
237#undef _TRACE_SYSCALLS_POINTERS_H
5b7ac358
MD
238
239
240/* Hijack probe callback for compat system call exit */
241#define TP_PROBE_CB(_template) &syscall_exit_probe
771af27e 242#define LTTNG_SC_COMPAT
57ede728 243#define SC_LTTNG_TRACEPOINT_EVENT(_name, _proto, _args, _fields) \
3bc29f0a 244 LTTNG_TRACEPOINT_EVENT(compat_syscall_exit_##_name, PARAMS(_proto), PARAMS(_args), \
57ede728 245 PARAMS(_fields))
265822ae 246#define SC_LTTNG_TRACEPOINT_EVENT_CODE(_name, _proto, _args, _locvar, _code_pre, _fields, _code_post) \
3bc29f0a 247 LTTNG_TRACEPOINT_EVENT_CODE(compat_syscall_exit_##_name, PARAMS(_proto), PARAMS(_args), \
265822ae 248 PARAMS(_locvar), PARAMS(_code_pre), PARAMS(_fields), PARAMS(_code_post))
57ede728
MD
249#define SC_LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS(_name, _fields) \
250 LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS(compat_syscall_exit_##_name, PARAMS(_fields))
cb3ef14c 251#define SC_LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS(_template, _name) \
3bc29f0a 252 LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS(compat_syscall_exit_##_template, \
5b7ac358 253 compat_syscall_exit_##_name)
141ddf28
MD
254/* Enumerations only defined at inital inclusion (not here). */
255#define SC_LTTNG_TRACEPOINT_ENUM(_name, _values)
5b7ac358
MD
256#define TRACE_SYSTEM compat_syscall_exit_integers
257#define TRACE_INCLUDE_FILE compat_syscalls_integers
241ae9a8 258#include <instrumentation/syscalls/headers/compat_syscalls_integers.h>
5b7ac358 259#undef TRACE_INCLUDE_FILE
a93244f8 260#undef TRACE_SYSTEM
5b7ac358
MD
261#define TRACE_SYSTEM compat_syscall_exit_pointers
262#define TRACE_INCLUDE_FILE compat_syscalls_pointers
241ae9a8 263#include <instrumentation/syscalls/headers/compat_syscalls_pointers.h>
5b7ac358 264#undef TRACE_INCLUDE_FILE
a93244f8 265#undef TRACE_SYSTEM
141ddf28 266#undef SC_LTTNG_TRACEPOINT_ENUM
cb3ef14c
MD
267#undef SC_LTTNG_TRACEPOINT_EVENT_CODE
268#undef SC_LTTNG_TRACEPOINT_EVENT
269#undef SC_LTTNG_TRACEPOINT_EVENT_CLASS_NOARGS
270#undef SC_LTTNG_TRACEPOINT_EVENT_INSTANCE_NOARGS
a93244f8 271#undef TP_PROBE_CB
3bc29f0a
MD
272#undef _TRACE_SYSCALLS_INTEGERS_H
273#undef _TRACE_SYSCALLS_POINTERS_H
771af27e 274#undef LTTNG_SC_COMPAT
5b7ac358 275
fc4f7161 276#undef SC_EXIT
259b6cb3 277
2655f9ad 278#undef TP_MODULE_NOINIT
259b6cb3
MD
279#undef LTTNG_PACKAGE_BUILD
280#undef CREATE_TRACE_POINTS
281
a93244f8
MD
282struct trace_syscall_entry {
283 void *func;
284 const struct lttng_event_desc *desc;
285 const struct lttng_event_field *fields;
286 unsigned int nrargs;
287};
288
289#define CREATE_SYSCALL_TABLE
290
fc4f7161
MD
291#define SC_ENTER
292
293#undef sc_exit
294#define sc_exit(...)
295
259b6cb3 296#undef TRACE_SYSCALL_TABLE
f7bdf4db 297#define TRACE_SYSCALL_TABLE(_template, _name, _nr, _nrargs) \
259b6cb3 298 [ _nr ] = { \
d4291869 299 .func = __event_probe__syscall_entry_##_template, \
259b6cb3 300 .nrargs = (_nrargs), \
d4291869
MD
301 .fields = __event_fields___syscall_entry_##_template, \
302 .desc = &__event_desc___syscall_entry_##_name, \
259b6cb3
MD
303 },
304
5b7ac358 305/* Syscall enter tracing table */
49c50022 306static const struct trace_syscall_entry sc_table[] = {
241ae9a8
MD
307#include <instrumentation/syscalls/headers/syscalls_integers.h>
308#include <instrumentation/syscalls/headers/syscalls_pointers.h>
259b6cb3
MD
309};
310
a93244f8
MD
311#undef TRACE_SYSCALL_TABLE
312#define TRACE_SYSCALL_TABLE(_template, _name, _nr, _nrargs) \
313 [ _nr ] = { \
d4291869 314 .func = __event_probe__compat_syscall_entry_##_template, \
a93244f8 315 .nrargs = (_nrargs), \
d4291869
MD
316 .fields = __event_fields___compat_syscall_entry_##_template, \
317 .desc = &__event_desc___compat_syscall_entry_##_name, \
a93244f8
MD
318 },
319
5b7ac358 320/* Compat syscall enter table */
a93244f8 321const struct trace_syscall_entry compat_sc_table[] = {
241ae9a8
MD
322#include <instrumentation/syscalls/headers/compat_syscalls_integers.h>
323#include <instrumentation/syscalls/headers/compat_syscalls_pointers.h>
a93244f8 324};
259b6cb3 325
fc4f7161
MD
326#undef SC_ENTER
327
328#define SC_EXIT
329
330#undef sc_exit
331#define sc_exit(...) __VA_ARGS__
332
5b7ac358
MD
333#undef TRACE_SYSCALL_TABLE
334#define TRACE_SYSCALL_TABLE(_template, _name, _nr, _nrargs) \
335 [ _nr ] = { \
336 .func = __event_probe__syscall_exit_##_template, \
337 .nrargs = (_nrargs), \
338 .fields = __event_fields___syscall_exit_##_template, \
339 .desc = &__event_desc___syscall_exit_##_name, \
340 },
341
342/* Syscall exit table */
343static const struct trace_syscall_entry sc_exit_table[] = {
241ae9a8
MD
344#include <instrumentation/syscalls/headers/syscalls_integers.h>
345#include <instrumentation/syscalls/headers/syscalls_pointers.h>
5b7ac358
MD
346};
347
348#undef TRACE_SYSCALL_TABLE
349#define TRACE_SYSCALL_TABLE(_template, _name, _nr, _nrargs) \
350 [ _nr ] = { \
351 .func = __event_probe__compat_syscall_exit_##_template, \
352 .nrargs = (_nrargs), \
353 .fields = __event_fields___compat_syscall_exit_##_template, \
354 .desc = &__event_desc___compat_syscall_exit_##_name, \
355 },
356
357/* Compat syscall exit table */
358const struct trace_syscall_entry compat_sc_exit_table[] = {
241ae9a8
MD
359#include <instrumentation/syscalls/headers/compat_syscalls_integers.h>
360#include <instrumentation/syscalls/headers/compat_syscalls_pointers.h>
5b7ac358
MD
361};
362
fc4f7161
MD
363#undef SC_EXIT
364
a93244f8 365#undef CREATE_SYSCALL_TABLE
2faf7d1b 366
80f87dd2
MD
367struct lttng_syscall_filter {
368 DECLARE_BITMAP(sc, NR_syscalls);
369 DECLARE_BITMAP(sc_compat, NR_compat_syscalls);
370};
371
a90917c3 372static void syscall_entry_unknown(struct lttng_event *event,
f405cfce
MD
373 struct pt_regs *regs, unsigned int id)
374{
1b7b9c65 375 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
f405cfce 376
b97bc8bf 377 syscall_get_arguments(current, regs, args);
1aa3298b 378 if (unlikely(in_compat_syscall()))
d4291869 379 __event_probe__compat_syscall_entry_unknown(event, id, args);
a93244f8 380 else
d4291869 381 __event_probe__syscall_entry_unknown(event, id, args);
f405cfce
MD
382}
383
2faf7d1b 384void syscall_entry_probe(void *__data, struct pt_regs *regs, long id)
259b6cb3 385{
a90917c3
MD
386 struct lttng_channel *chan = __data;
387 struct lttng_event *event, *unknown_event;
49c50022
MD
388 const struct trace_syscall_entry *table, *entry;
389 size_t table_len;
259b6cb3 390
1aa3298b 391 if (unlikely(in_compat_syscall())) {
80f87dd2
MD
392 struct lttng_syscall_filter *filter;
393
39848b23 394 filter = rcu_dereference_raw_check(chan->sc_filter);
80f87dd2 395 if (filter) {
74f7b56a 396 if (id < 0 || id >= NR_compat_syscalls
80f87dd2
MD
397 || !test_bit(id, filter->sc_compat)) {
398 /* System call filtered out. */
399 return;
400 }
401 }
49c50022 402 table = compat_sc_table;
a93244f8 403 table_len = ARRAY_SIZE(compat_sc_table);
49c50022
MD
404 unknown_event = chan->sc_compat_unknown;
405 } else {
80f87dd2
MD
406 struct lttng_syscall_filter *filter;
407
39848b23 408 filter = rcu_dereference_raw_check(chan->sc_filter);
80f87dd2 409 if (filter) {
74f7b56a 410 if (id < 0 || id >= NR_syscalls
80f87dd2
MD
411 || !test_bit(id, filter->sc)) {
412 /* System call filtered out. */
413 return;
414 }
415 }
49c50022
MD
416 table = sc_table;
417 table_len = ARRAY_SIZE(sc_table);
418 unknown_event = chan->sc_unknown;
b76dc1a0 419 }
74f7b56a 420 if (unlikely(id < 0 || id >= table_len)) {
49c50022 421 syscall_entry_unknown(unknown_event, regs, id);
259b6cb3 422 return;
f405cfce 423 }
1aa3298b 424 if (unlikely(in_compat_syscall()))
49c50022
MD
425 event = chan->compat_sc_table[id];
426 else
427 event = chan->sc_table[id];
f405cfce 428 if (unlikely(!event)) {
49c50022 429 syscall_entry_unknown(unknown_event, regs, id);
f405cfce
MD
430 return;
431 }
49c50022 432 entry = &table[id];
f405cfce 433 WARN_ON_ONCE(!entry);
259b6cb3
MD
434
435 switch (entry->nrargs) {
436 case 0:
437 {
438 void (*fptr)(void *__data) = entry->func;
439
440 fptr(event);
441 break;
442 }
443 case 1:
444 {
445 void (*fptr)(void *__data, unsigned long arg0) = entry->func;
1b7b9c65 446 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
259b6cb3 447
b97bc8bf 448 syscall_get_arguments(current, regs, args);
259b6cb3
MD
449 fptr(event, args[0]);
450 break;
451 }
452 case 2:
453 {
454 void (*fptr)(void *__data,
455 unsigned long arg0,
456 unsigned long arg1) = entry->func;
1b7b9c65 457 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
259b6cb3 458
b97bc8bf 459 syscall_get_arguments(current, regs, args);
259b6cb3
MD
460 fptr(event, args[0], args[1]);
461 break;
462 }
463 case 3:
464 {
465 void (*fptr)(void *__data,
466 unsigned long arg0,
467 unsigned long arg1,
468 unsigned long arg2) = entry->func;
1b7b9c65 469 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
259b6cb3 470
b97bc8bf 471 syscall_get_arguments(current, regs, args);
259b6cb3
MD
472 fptr(event, args[0], args[1], args[2]);
473 break;
474 }
475 case 4:
476 {
477 void (*fptr)(void *__data,
478 unsigned long arg0,
479 unsigned long arg1,
480 unsigned long arg2,
481 unsigned long arg3) = entry->func;
1b7b9c65 482 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
259b6cb3 483
b97bc8bf 484 syscall_get_arguments(current, regs, args);
259b6cb3
MD
485 fptr(event, args[0], args[1], args[2], args[3]);
486 break;
487 }
488 case 5:
489 {
490 void (*fptr)(void *__data,
491 unsigned long arg0,
492 unsigned long arg1,
493 unsigned long arg2,
494 unsigned long arg3,
495 unsigned long arg4) = entry->func;
1b7b9c65 496 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
259b6cb3 497
b97bc8bf 498 syscall_get_arguments(current, regs, args);
259b6cb3
MD
499 fptr(event, args[0], args[1], args[2], args[3], args[4]);
500 break;
501 }
502 case 6:
503 {
504 void (*fptr)(void *__data,
505 unsigned long arg0,
506 unsigned long arg1,
507 unsigned long arg2,
508 unsigned long arg3,
509 unsigned long arg4,
510 unsigned long arg5) = entry->func;
1b7b9c65 511 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
259b6cb3 512
b97bc8bf 513 syscall_get_arguments(current, regs, args);
259b6cb3
MD
514 fptr(event, args[0], args[1], args[2],
515 args[3], args[4], args[5]);
516 break;
517 }
518 default:
519 break;
520 }
521}
522
5b7ac358 523static void syscall_exit_unknown(struct lttng_event *event,
74f7b56a 524 struct pt_regs *regs, int id, long ret)
5b7ac358 525{
1b7b9c65 526 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
5b7ac358 527
b97bc8bf 528 syscall_get_arguments(current, regs, args);
1aa3298b 529 if (unlikely(in_compat_syscall()))
5b7ac358
MD
530 __event_probe__compat_syscall_exit_unknown(event, id, ret,
531 args);
532 else
533 __event_probe__syscall_exit_unknown(event, id, ret, args);
534}
535
536void syscall_exit_probe(void *__data, struct pt_regs *regs, long ret)
537{
538 struct lttng_channel *chan = __data;
539 struct lttng_event *event, *unknown_event;
540 const struct trace_syscall_entry *table, *entry;
541 size_t table_len;
542 long id;
543
544 id = syscall_get_nr(current, regs);
1aa3298b 545 if (unlikely(in_compat_syscall())) {
5b7ac358
MD
546 struct lttng_syscall_filter *filter;
547
39848b23 548 filter = rcu_dereference_raw_check(chan->sc_filter);
5b7ac358 549 if (filter) {
74f7b56a 550 if (id < 0 || id >= NR_compat_syscalls
5b7ac358
MD
551 || !test_bit(id, filter->sc_compat)) {
552 /* System call filtered out. */
553 return;
554 }
555 }
556 table = compat_sc_exit_table;
557 table_len = ARRAY_SIZE(compat_sc_exit_table);
558 unknown_event = chan->compat_sc_exit_unknown;
559 } else {
560 struct lttng_syscall_filter *filter;
561
39848b23 562 filter = rcu_dereference_raw_check(chan->sc_filter);
5b7ac358 563 if (filter) {
74f7b56a 564 if (id < 0 || id >= NR_syscalls
5b7ac358
MD
565 || !test_bit(id, filter->sc)) {
566 /* System call filtered out. */
567 return;
568 }
569 }
570 table = sc_exit_table;
571 table_len = ARRAY_SIZE(sc_exit_table);
572 unknown_event = chan->sc_exit_unknown;
573 }
74f7b56a 574 if (unlikely(id < 0 || id >= table_len)) {
5b7ac358
MD
575 syscall_exit_unknown(unknown_event, regs, id, ret);
576 return;
577 }
1aa3298b 578 if (unlikely(in_compat_syscall()))
5b7ac358
MD
579 event = chan->compat_sc_exit_table[id];
580 else
581 event = chan->sc_exit_table[id];
582 if (unlikely(!event)) {
583 syscall_exit_unknown(unknown_event, regs, id, ret);
584 return;
585 }
586 entry = &table[id];
587 WARN_ON_ONCE(!entry);
588
589 switch (entry->nrargs) {
590 case 0:
591 {
fc4f7161 592 void (*fptr)(void *__data, long ret) = entry->func;
5b7ac358 593
fc4f7161 594 fptr(event, ret);
5b7ac358
MD
595 break;
596 }
597 case 1:
598 {
599 void (*fptr)(void *__data,
fc4f7161 600 long ret,
5b7ac358 601 unsigned long arg0) = entry->func;
1b7b9c65 602 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
5b7ac358 603
b97bc8bf 604 syscall_get_arguments(current, regs, args);
fc4f7161 605 fptr(event, ret, args[0]);
5b7ac358
MD
606 break;
607 }
608 case 2:
609 {
610 void (*fptr)(void *__data,
fc4f7161 611 long ret,
5b7ac358
MD
612 unsigned long arg0,
613 unsigned long arg1) = entry->func;
1b7b9c65 614 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
5b7ac358 615
b97bc8bf 616 syscall_get_arguments(current, regs, args);
fc4f7161 617 fptr(event, ret, args[0], args[1]);
5b7ac358
MD
618 break;
619 }
620 case 3:
621 {
622 void (*fptr)(void *__data,
fc4f7161 623 long ret,
5b7ac358
MD
624 unsigned long arg0,
625 unsigned long arg1,
626 unsigned long arg2) = entry->func;
1b7b9c65 627 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
5b7ac358 628
b97bc8bf 629 syscall_get_arguments(current, regs, args);
fc4f7161 630 fptr(event, ret, args[0], args[1], args[2]);
5b7ac358
MD
631 break;
632 }
633 case 4:
634 {
635 void (*fptr)(void *__data,
fc4f7161 636 long ret,
5b7ac358
MD
637 unsigned long arg0,
638 unsigned long arg1,
639 unsigned long arg2,
640 unsigned long arg3) = entry->func;
1b7b9c65 641 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
5b7ac358 642
b97bc8bf 643 syscall_get_arguments(current, regs, args);
fc4f7161 644 fptr(event, ret, args[0], args[1], args[2], args[3]);
5b7ac358
MD
645 break;
646 }
647 case 5:
648 {
649 void (*fptr)(void *__data,
fc4f7161 650 long ret,
5b7ac358
MD
651 unsigned long arg0,
652 unsigned long arg1,
653 unsigned long arg2,
654 unsigned long arg3,
655 unsigned long arg4) = entry->func;
1b7b9c65 656 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
5b7ac358 657
b97bc8bf 658 syscall_get_arguments(current, regs, args);
fc4f7161 659 fptr(event, ret, args[0], args[1], args[2], args[3], args[4]);
5b7ac358
MD
660 break;
661 }
662 case 6:
663 {
664 void (*fptr)(void *__data,
fc4f7161 665 long ret,
5b7ac358
MD
666 unsigned long arg0,
667 unsigned long arg1,
668 unsigned long arg2,
669 unsigned long arg3,
670 unsigned long arg4,
671 unsigned long arg5) = entry->func;
1b7b9c65 672 unsigned long args[LTTNG_SYSCALL_NR_ARGS];
5b7ac358 673
b97bc8bf 674 syscall_get_arguments(current, regs, args);
fc4f7161 675 fptr(event, ret, args[0], args[1], args[2],
5b7ac358
MD
676 args[3], args[4], args[5]);
677 break;
678 }
679 default:
680 break;
681 }
682}
683
33a39a3c
MD
684/*
685 * noinline to diminish caller stack size.
686 * Should be called with sessions lock held.
687 */
49c50022
MD
688static
689int fill_table(const struct trace_syscall_entry *table, size_t table_len,
5b7ac358
MD
690 struct lttng_event **chan_table, struct lttng_channel *chan,
691 void *filter, enum sc_type type)
259b6cb3 692{
2a0c4816 693 const struct lttng_event_desc *desc;
259b6cb3 694 unsigned int i;
49c50022
MD
695
696 /* Allocate events for each syscall, insert into table */
697 for (i = 0; i < table_len; i++) {
698 struct lttng_kernel_event ev;
2a0c4816 699 desc = table[i].desc;
49c50022
MD
700
701 if (!desc) {
702 /* Unknown syscall */
703 continue;
704 }
705 /*
706 * Skip those already populated by previous failed
707 * register for this channel.
708 */
709 if (chan_table[i])
710 continue;
711 memset(&ev, 0, sizeof(ev));
5b7ac358
MD
712 switch (type) {
713 case SC_TYPE_ENTRY:
714 strncpy(ev.name, SYSCALL_ENTRY_STR,
715 LTTNG_KERNEL_SYM_NAME_LEN);
716 break;
717 case SC_TYPE_EXIT:
718 strncpy(ev.name, SYSCALL_EXIT_STR,
719 LTTNG_KERNEL_SYM_NAME_LEN);
720 break;
721 case SC_TYPE_COMPAT_ENTRY:
722 strncpy(ev.name, COMPAT_SYSCALL_ENTRY_STR,
723 LTTNG_KERNEL_SYM_NAME_LEN);
724 break;
725 case SC_TYPE_COMPAT_EXIT:
726 strncpy(ev.name, COMPAT_SYSCALL_EXIT_STR,
727 LTTNG_KERNEL_SYM_NAME_LEN);
728 break;
729 default:
730 BUG_ON(1);
731 break;
732 }
733 strncat(ev.name, desc->name,
734 LTTNG_KERNEL_SYM_NAME_LEN - strlen(ev.name) - 1);
f8695253 735 ev.name[LTTNG_KERNEL_SYM_NAME_LEN - 1] = '\0';
33a39a3c
MD
736 ev.instrumentation = LTTNG_KERNEL_SYSCALL;
737 chan_table[i] = _lttng_event_create(chan, &ev, filter,
3c997079 738 desc, ev.instrumentation);
abc0446a
MD
739 WARN_ON_ONCE(!chan_table[i]);
740 if (IS_ERR(chan_table[i])) {
49c50022
MD
741 /*
742 * If something goes wrong in event registration
743 * after the first one, we have no choice but to
744 * leave the previous events in there, until
745 * deleted by session teardown.
746 */
abc0446a 747 return PTR_ERR(chan_table[i]);
49c50022
MD
748 }
749 }
750 return 0;
751}
752
33a39a3c
MD
753/*
754 * Should be called with sessions lock held.
755 */
a90917c3 756int lttng_syscalls_register(struct lttng_channel *chan, void *filter)
49c50022 757{
2a0c4816 758 struct lttng_kernel_event ev;
259b6cb3
MD
759 int ret;
760
259b6cb3
MD
761 if (!chan->sc_table) {
762 /* create syscall table mapping syscall to events */
a90917c3 763 chan->sc_table = kzalloc(sizeof(struct lttng_event *)
259b6cb3
MD
764 * ARRAY_SIZE(sc_table), GFP_KERNEL);
765 if (!chan->sc_table)
766 return -ENOMEM;
767 }
5b7ac358
MD
768 if (!chan->sc_exit_table) {
769 /* create syscall table mapping syscall to events */
770 chan->sc_exit_table = kzalloc(sizeof(struct lttng_event *)
771 * ARRAY_SIZE(sc_exit_table), GFP_KERNEL);
772 if (!chan->sc_exit_table)
773 return -ENOMEM;
774 }
775
259b6cb3 776
49c50022
MD
777#ifdef CONFIG_COMPAT
778 if (!chan->compat_sc_table) {
779 /* create syscall table mapping compat syscall to events */
a90917c3 780 chan->compat_sc_table = kzalloc(sizeof(struct lttng_event *)
a93244f8 781 * ARRAY_SIZE(compat_sc_table), GFP_KERNEL);
49c50022
MD
782 if (!chan->compat_sc_table)
783 return -ENOMEM;
784 }
5b7ac358
MD
785
786 if (!chan->compat_sc_exit_table) {
787 /* create syscall table mapping compat syscall to events */
788 chan->compat_sc_exit_table = kzalloc(sizeof(struct lttng_event *)
789 * ARRAY_SIZE(compat_sc_exit_table), GFP_KERNEL);
790 if (!chan->compat_sc_exit_table)
791 return -ENOMEM;
792 }
49c50022 793#endif
f405cfce 794 if (!chan->sc_unknown) {
f405cfce 795 const struct lttng_event_desc *desc =
d4291869 796 &__event_desc___syscall_entry_unknown;
2f804c0a 797
f405cfce 798 memset(&ev, 0, sizeof(ev));
f8695253
MD
799 strncpy(ev.name, desc->name, LTTNG_KERNEL_SYM_NAME_LEN);
800 ev.name[LTTNG_KERNEL_SYM_NAME_LEN - 1] = '\0';
33a39a3c
MD
801 ev.instrumentation = LTTNG_KERNEL_SYSCALL;
802 chan->sc_unknown = _lttng_event_create(chan, &ev, filter,
3c997079
MD
803 desc,
804 ev.instrumentation);
abc0446a
MD
805 WARN_ON_ONCE(!chan->sc_unknown);
806 if (IS_ERR(chan->sc_unknown)) {
807 return PTR_ERR(chan->sc_unknown);
f405cfce
MD
808 }
809 }
810
b76dc1a0 811 if (!chan->sc_compat_unknown) {
b76dc1a0 812 const struct lttng_event_desc *desc =
d4291869 813 &__event_desc___compat_syscall_entry_unknown;
b76dc1a0
MD
814
815 memset(&ev, 0, sizeof(ev));
f8695253
MD
816 strncpy(ev.name, desc->name, LTTNG_KERNEL_SYM_NAME_LEN);
817 ev.name[LTTNG_KERNEL_SYM_NAME_LEN - 1] = '\0';
33a39a3c
MD
818 ev.instrumentation = LTTNG_KERNEL_SYSCALL;
819 chan->sc_compat_unknown = _lttng_event_create(chan, &ev, filter,
3c997079
MD
820 desc,
821 ev.instrumentation);
abc0446a
MD
822 WARN_ON_ONCE(!chan->sc_unknown);
823 if (IS_ERR(chan->sc_compat_unknown)) {
824 return PTR_ERR(chan->sc_compat_unknown);
b76dc1a0
MD
825 }
826 }
827
5b7ac358 828 if (!chan->compat_sc_exit_unknown) {
2f804c0a 829 const struct lttng_event_desc *desc =
5b7ac358 830 &__event_desc___compat_syscall_exit_unknown;
2f804c0a
MD
831
832 memset(&ev, 0, sizeof(ev));
f8695253
MD
833 strncpy(ev.name, desc->name, LTTNG_KERNEL_SYM_NAME_LEN);
834 ev.name[LTTNG_KERNEL_SYM_NAME_LEN - 1] = '\0';
33a39a3c
MD
835 ev.instrumentation = LTTNG_KERNEL_SYSCALL;
836 chan->compat_sc_exit_unknown = _lttng_event_create(chan, &ev,
3c997079
MD
837 filter, desc,
838 ev.instrumentation);
5b7ac358
MD
839 WARN_ON_ONCE(!chan->compat_sc_exit_unknown);
840 if (IS_ERR(chan->compat_sc_exit_unknown)) {
841 return PTR_ERR(chan->compat_sc_exit_unknown);
842 }
843 }
844
845 if (!chan->sc_exit_unknown) {
846 const struct lttng_event_desc *desc =
847 &__event_desc___syscall_exit_unknown;
848
849 memset(&ev, 0, sizeof(ev));
850 strncpy(ev.name, desc->name, LTTNG_KERNEL_SYM_NAME_LEN);
851 ev.name[LTTNG_KERNEL_SYM_NAME_LEN - 1] = '\0';
33a39a3c
MD
852 ev.instrumentation = LTTNG_KERNEL_SYSCALL;
853 chan->sc_exit_unknown = _lttng_event_create(chan, &ev, filter,
3c997079 854 desc, ev.instrumentation);
5b7ac358
MD
855 WARN_ON_ONCE(!chan->sc_exit_unknown);
856 if (IS_ERR(chan->sc_exit_unknown)) {
857 return PTR_ERR(chan->sc_exit_unknown);
2f804c0a
MD
858 }
859 }
860
49c50022 861 ret = fill_table(sc_table, ARRAY_SIZE(sc_table),
5b7ac358
MD
862 chan->sc_table, chan, filter, SC_TYPE_ENTRY);
863 if (ret)
864 return ret;
865 ret = fill_table(sc_exit_table, ARRAY_SIZE(sc_exit_table),
866 chan->sc_exit_table, chan, filter, SC_TYPE_EXIT);
49c50022
MD
867 if (ret)
868 return ret;
5b7ac358 869
49c50022 870#ifdef CONFIG_COMPAT
a93244f8 871 ret = fill_table(compat_sc_table, ARRAY_SIZE(compat_sc_table),
5b7ac358
MD
872 chan->compat_sc_table, chan, filter,
873 SC_TYPE_COMPAT_ENTRY);
874 if (ret)
875 return ret;
876 ret = fill_table(compat_sc_exit_table, ARRAY_SIZE(compat_sc_exit_table),
877 chan->compat_sc_exit_table, chan, filter,
878 SC_TYPE_COMPAT_EXIT);
49c50022
MD
879 if (ret)
880 return ret;
881#endif
80f87dd2 882 if (!chan->sys_enter_registered) {
0d260d3d 883 ret = lttng_tracepoint_probe_register("sys_enter",
80f87dd2
MD
884 (void *) syscall_entry_probe, chan);
885 if (ret)
886 return ret;
887 chan->sys_enter_registered = 1;
888 }
63728b02
MD
889 /*
890 * We change the name of sys_exit tracepoint due to namespace
891 * conflict with sys_exit syscall entry.
892 */
80f87dd2 893 if (!chan->sys_exit_registered) {
0d260d3d 894 ret = lttng_tracepoint_probe_register("sys_exit",
5b7ac358 895 (void *) syscall_exit_probe, chan);
80f87dd2 896 if (ret) {
0d260d3d 897 WARN_ON_ONCE(lttng_tracepoint_probe_unregister("sys_enter",
80f87dd2
MD
898 (void *) syscall_entry_probe, chan));
899 return ret;
900 }
901 chan->sys_exit_registered = 1;
63728b02 902 }
259b6cb3
MD
903 return ret;
904}
905
906/*
907 * Only called at session destruction.
908 */
a90917c3 909int lttng_syscalls_unregister(struct lttng_channel *chan)
259b6cb3
MD
910{
911 int ret;
912
913 if (!chan->sc_table)
914 return 0;
80f87dd2 915 if (chan->sys_enter_registered) {
0d260d3d 916 ret = lttng_tracepoint_probe_unregister("sys_enter",
2d9cd7f3 917 (void *) syscall_entry_probe, chan);
80f87dd2
MD
918 if (ret)
919 return ret;
920 chan->sys_enter_registered = 0;
921 }
922 if (chan->sys_exit_registered) {
0d260d3d 923 ret = lttng_tracepoint_probe_unregister("sys_exit",
2d9cd7f3 924 (void *) syscall_exit_probe, chan);
80f87dd2
MD
925 if (ret)
926 return ret;
927 chan->sys_exit_registered = 0;
928 }
a90917c3 929 /* lttng_event destroy will be performed by lttng_session_destroy() */
259b6cb3 930 kfree(chan->sc_table);
5b7ac358 931 kfree(chan->sc_exit_table);
49c50022
MD
932#ifdef CONFIG_COMPAT
933 kfree(chan->compat_sc_table);
5b7ac358 934 kfree(chan->compat_sc_exit_table);
49c50022 935#endif
80f87dd2
MD
936 kfree(chan->sc_filter);
937 return 0;
938}
939
940static
941int get_syscall_nr(const char *syscall_name)
942{
943 int syscall_nr = -1;
944 int i;
945
946 for (i = 0; i < ARRAY_SIZE(sc_table); i++) {
947 const struct trace_syscall_entry *entry;
5b7ac358 948 const char *it_name;
80f87dd2
MD
949
950 entry = &sc_table[i];
951 if (!entry->desc)
952 continue;
5b7ac358
MD
953 it_name = entry->desc->name;
954 it_name += strlen(SYSCALL_ENTRY_STR);
955 if (!strcmp(syscall_name, it_name)) {
80f87dd2
MD
956 syscall_nr = i;
957 break;
958 }
959 }
960 return syscall_nr;
961}
962
963static
964int get_compat_syscall_nr(const char *syscall_name)
965{
966 int syscall_nr = -1;
967 int i;
968
969 for (i = 0; i < ARRAY_SIZE(compat_sc_table); i++) {
970 const struct trace_syscall_entry *entry;
5b7ac358 971 const char *it_name;
80f87dd2
MD
972
973 entry = &compat_sc_table[i];
974 if (!entry->desc)
975 continue;
5b7ac358
MD
976 it_name = entry->desc->name;
977 it_name += strlen(COMPAT_SYSCALL_ENTRY_STR);
978 if (!strcmp(syscall_name, it_name)) {
80f87dd2
MD
979 syscall_nr = i;
980 break;
981 }
982 }
983 return syscall_nr;
984}
985
12e579db
MD
986static
987uint32_t get_sc_tables_len(void)
988{
989 return ARRAY_SIZE(sc_table) + ARRAY_SIZE(compat_sc_table);
990}
991
80f87dd2
MD
992int lttng_syscall_filter_enable(struct lttng_channel *chan,
993 const char *name)
994{
995 int syscall_nr, compat_syscall_nr, ret;
996 struct lttng_syscall_filter *filter;
997
998 WARN_ON_ONCE(!chan->sc_table);
999
1000 if (!name) {
1001 /* Enable all system calls by removing filter */
1002 if (chan->sc_filter) {
1003 filter = chan->sc_filter;
1004 rcu_assign_pointer(chan->sc_filter, NULL);
1005 synchronize_trace();
1006 kfree(filter);
1007 }
1008 chan->syscall_all = 1;
1009 return 0;
1010 }
1011
1012 if (!chan->sc_filter) {
1013 if (chan->syscall_all) {
1014 /*
1015 * All syscalls are already enabled.
1016 */
1017 return -EEXIST;
1018 }
1019 filter = kzalloc(sizeof(struct lttng_syscall_filter),
1020 GFP_KERNEL);
1021 if (!filter)
1022 return -ENOMEM;
1023 } else {
1024 filter = chan->sc_filter;
1025 }
1026 syscall_nr = get_syscall_nr(name);
1027 compat_syscall_nr = get_compat_syscall_nr(name);
1028 if (syscall_nr < 0 && compat_syscall_nr < 0) {
1029 ret = -ENOENT;
1030 goto error;
1031 }
1032 if (syscall_nr >= 0) {
1033 if (test_bit(syscall_nr, filter->sc)) {
1034 ret = -EEXIST;
1035 goto error;
1036 }
1037 bitmap_set(filter->sc, syscall_nr, 1);
1038 }
1039 if (compat_syscall_nr >= 0) {
1040 if (test_bit(compat_syscall_nr, filter->sc_compat)) {
1041 ret = -EEXIST;
1042 goto error;
1043 }
1044 bitmap_set(filter->sc_compat, compat_syscall_nr, 1);
1045 }
1046 if (!chan->sc_filter)
1047 rcu_assign_pointer(chan->sc_filter, filter);
1048 return 0;
1049
1050error:
1051 if (!chan->sc_filter)
1052 kfree(filter);
1053 return ret;
1054}
1055
1056int lttng_syscall_filter_disable(struct lttng_channel *chan,
1057 const char *name)
1058{
1059 int syscall_nr, compat_syscall_nr, ret;
1060 struct lttng_syscall_filter *filter;
1061
1062 WARN_ON_ONCE(!chan->sc_table);
1063
1064 if (!chan->sc_filter) {
bcde0d5b
MD
1065 if (!chan->syscall_all)
1066 return -EEXIST;
80f87dd2
MD
1067 filter = kzalloc(sizeof(struct lttng_syscall_filter),
1068 GFP_KERNEL);
1069 if (!filter)
1070 return -ENOMEM;
1071 /* Trace all system calls, then apply disable. */
1072 bitmap_set(filter->sc, 0, NR_syscalls);
1073 bitmap_set(filter->sc_compat, 0, NR_compat_syscalls);
1074 } else {
1075 filter = chan->sc_filter;
1076 }
1077
72814741 1078 if (!name) {
404e87bf
MD
1079 /* Fail if all syscalls are already disabled. */
1080 if (bitmap_empty(filter->sc, NR_syscalls)
1081 && bitmap_empty(filter->sc_compat,
1082 NR_compat_syscalls)) {
1083 ret = -EEXIST;
1084 goto error;
1085 }
1086
72814741
MD
1087 /* Disable all system calls */
1088 bitmap_clear(filter->sc, 0, NR_syscalls);
1089 bitmap_clear(filter->sc_compat, 0, NR_compat_syscalls);
1090 goto apply_filter;
1091 }
80f87dd2
MD
1092 syscall_nr = get_syscall_nr(name);
1093 compat_syscall_nr = get_compat_syscall_nr(name);
1094 if (syscall_nr < 0 && compat_syscall_nr < 0) {
1095 ret = -ENOENT;
1096 goto error;
1097 }
1098 if (syscall_nr >= 0) {
cecef7f8 1099 if (!test_bit(syscall_nr, filter->sc)) {
80f87dd2
MD
1100 ret = -EEXIST;
1101 goto error;
1102 }
cecef7f8 1103 bitmap_clear(filter->sc, syscall_nr, 1);
80f87dd2
MD
1104 }
1105 if (compat_syscall_nr >= 0) {
cecef7f8 1106 if (!test_bit(compat_syscall_nr, filter->sc_compat)) {
80f87dd2
MD
1107 ret = -EEXIST;
1108 goto error;
1109 }
cecef7f8 1110 bitmap_clear(filter->sc_compat, compat_syscall_nr, 1);
80f87dd2 1111 }
72814741 1112apply_filter:
80f87dd2
MD
1113 if (!chan->sc_filter)
1114 rcu_assign_pointer(chan->sc_filter, filter);
1115 chan->syscall_all = 0;
259b6cb3 1116 return 0;
80f87dd2
MD
1117
1118error:
1119 if (!chan->sc_filter)
1120 kfree(filter);
1121 return ret;
259b6cb3 1122}
2d2464bd
MD
1123
1124static
1125const struct trace_syscall_entry *syscall_list_get_entry(loff_t *pos)
1126{
1127 const struct trace_syscall_entry *entry;
1128 int iter = 0;
1129
1130 for (entry = sc_table;
1131 entry < sc_table + ARRAY_SIZE(sc_table);
1132 entry++) {
1133 if (iter++ >= *pos)
1134 return entry;
1135 }
1136 for (entry = compat_sc_table;
1137 entry < compat_sc_table + ARRAY_SIZE(compat_sc_table);
1138 entry++) {
1139 if (iter++ >= *pos)
1140 return entry;
1141 }
1142 /* End of list */
1143 return NULL;
1144}
1145
1146static
1147void *syscall_list_start(struct seq_file *m, loff_t *pos)
1148{
1149 return (void *) syscall_list_get_entry(pos);
1150}
1151
1152static
1153void *syscall_list_next(struct seq_file *m, void *p, loff_t *ppos)
1154{
1155 (*ppos)++;
1156 return (void *) syscall_list_get_entry(ppos);
1157}
1158
1159static
1160void syscall_list_stop(struct seq_file *m, void *p)
1161{
1162}
1163
12e579db
MD
1164static
1165int get_sc_table(const struct trace_syscall_entry *entry,
1166 const struct trace_syscall_entry **table,
1167 unsigned int *bitness)
1168{
1169 if (entry >= sc_table && entry < sc_table + ARRAY_SIZE(sc_table)) {
1170 if (bitness)
1171 *bitness = BITS_PER_LONG;
1172 if (table)
1173 *table = sc_table;
1174 return 0;
1175 }
1176 if (!(entry >= compat_sc_table
1177 && entry < compat_sc_table + ARRAY_SIZE(compat_sc_table))) {
1178 return -EINVAL;
1179 }
1180 if (bitness)
1181 *bitness = 32;
1182 if (table)
1183 *table = compat_sc_table;
1184 return 0;
1185}
1186
2d2464bd
MD
1187static
1188int syscall_list_show(struct seq_file *m, void *p)
1189{
1190 const struct trace_syscall_entry *table, *entry = p;
1191 unsigned int bitness;
d4291869 1192 unsigned long index;
12e579db 1193 int ret;
d4291869 1194 const char *name;
2d2464bd 1195
12e579db
MD
1196 ret = get_sc_table(entry, &table, &bitness);
1197 if (ret)
1198 return ret;
f4855b46
MD
1199 if (!entry->desc)
1200 return 0;
d4291869
MD
1201 if (table == sc_table) {
1202 index = entry - table;
1203 name = &entry->desc->name[strlen(SYSCALL_ENTRY_STR)];
1204 } else {
1205 index = (entry - table) + ARRAY_SIZE(sc_table);
1206 name = &entry->desc->name[strlen(COMPAT_SYSCALL_ENTRY_STR)];
1207 }
12e579db 1208 seq_printf(m, "syscall { index = %lu; name = %s; bitness = %u; };\n",
d4291869 1209 index, name, bitness);
2d2464bd
MD
1210 return 0;
1211}
1212
1213static
1214const struct seq_operations lttng_syscall_list_seq_ops = {
1215 .start = syscall_list_start,
1216 .next = syscall_list_next,
1217 .stop = syscall_list_stop,
1218 .show = syscall_list_show,
1219};
1220
1221static
1222int lttng_syscall_list_open(struct inode *inode, struct file *file)
1223{
1224 return seq_open(file, &lttng_syscall_list_seq_ops);
1225}
1226
1227const struct file_operations lttng_syscall_list_fops = {
1228 .owner = THIS_MODULE,
1229 .open = lttng_syscall_list_open,
1230 .read = seq_read,
1231 .llseek = seq_lseek,
1232 .release = seq_release,
1233};
12e579db
MD
1234
1235long lttng_channel_syscall_mask(struct lttng_channel *channel,
1236 struct lttng_kernel_syscall_mask __user *usyscall_mask)
1237{
1238 uint32_t len, sc_tables_len, bitmask_len;
1239 int ret = 0, bit;
1240 char *tmp_mask;
1241 struct lttng_syscall_filter *filter;
1242
1243 ret = get_user(len, &usyscall_mask->len);
1244 if (ret)
1245 return ret;
1246 sc_tables_len = get_sc_tables_len();
1247 bitmask_len = ALIGN(sc_tables_len, 8) >> 3;
1248 if (len < sc_tables_len) {
1249 return put_user(sc_tables_len, &usyscall_mask->len);
1250 }
1251 /* Array is large enough, we can copy array to user-space. */
1252 tmp_mask = kzalloc(bitmask_len, GFP_KERNEL);
1253 if (!tmp_mask)
1254 return -ENOMEM;
1255 filter = channel->sc_filter;
1256
1257 for (bit = 0; bit < ARRAY_SIZE(sc_table); bit++) {
e2129868 1258 char state;
2f25059d
MD
1259
1260 if (channel->sc_table) {
1261 if (filter)
1262 state = test_bit(bit, filter->sc);
1263 else
1264 state = 1;
1265 } else {
1266 state = 0;
1267 }
1268 bt_bitfield_write_be(tmp_mask, char, bit, 1, state);
12e579db
MD
1269 }
1270 for (; bit < sc_tables_len; bit++) {
e2129868 1271 char state;
2f25059d
MD
1272
1273 if (channel->compat_sc_table) {
1274 if (filter)
1275 state = test_bit(bit - ARRAY_SIZE(sc_table),
1276 filter->sc_compat);
1277 else
1278 state = 1;
1279 } else {
1280 state = 0;
1281 }
1282 bt_bitfield_write_be(tmp_mask, char, bit, 1, state);
12e579db
MD
1283 }
1284 if (copy_to_user(usyscall_mask->mask, tmp_mask, bitmask_len))
1285 ret = -EFAULT;
1286 kfree(tmp_mask);
1287 return ret;
1288}
082d4946
MD
1289
1290int lttng_abi_syscall_list(void)
1291{
1292 struct file *syscall_list_file;
1293 int file_fd, ret;
1294
34522739 1295 file_fd = get_unused_fd_flags(0);
082d4946
MD
1296 if (file_fd < 0) {
1297 ret = file_fd;
1298 goto fd_error;
1299 }
1300
1301 syscall_list_file = anon_inode_getfile("[lttng_syscall_list]",
1302 &lttng_syscall_list_fops,
1303 NULL, O_RDWR);
1304 if (IS_ERR(syscall_list_file)) {
1305 ret = PTR_ERR(syscall_list_file);
1306 goto file_error;
1307 }
1308 ret = lttng_syscall_list_fops.open(NULL, syscall_list_file);
1309 if (ret < 0)
1310 goto open_error;
1311 fd_install(file_fd, syscall_list_file);
082d4946
MD
1312 return file_fd;
1313
1314open_error:
1315 fput(syscall_list_file);
1316file_error:
1317 put_unused_fd(file_fd);
1318fd_error:
1319 return ret;
1320}
This page took 0.101959 seconds and 4 git commands to generate.