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